diff --git a/.gitignore b/.gitignore index a471ffe03ff2c..d69f9330bc92d 100644 --- a/.gitignore +++ b/.gitignore @@ -1,4 +1,3 @@ -pkg/ tivan .vagrant telegraf diff --git a/Godeps/Godeps.json b/Godeps/Godeps.json new file mode 100644 index 0000000000000..a4544d3aa3338 --- /dev/null +++ b/Godeps/Godeps.json @@ -0,0 +1,210 @@ +{ + "ImportPath": "github.com/influxdb/telegraf", + "GoVersion": "go1.4.2", + "Packages": [ + "./..." + ], + "Deps": [ + { + "ImportPath": "github.com/Shopify/sarama", + "Comment": "v1.4.3-45-g5b18996", + "Rev": "5b18996ef1cd555a60562ae4c5d7843ae137e12d" + }, + { + "ImportPath": "github.com/Sirupsen/logrus", + "Comment": "v0.8.6-7-g9c060de", + "Rev": "9c060de643590dae45da9d7c26276463bfc46fa0" + }, + { + "ImportPath": "github.com/armon/go-metrics", + "Rev": "b2d95e5291cdbc26997d1301a5e467ecbb240e25" + }, + { + "ImportPath": "github.com/boltdb/bolt", + "Comment": "v1.0-117-g0f053fa", + "Rev": "0f053fabc06119583d61937a0a06ef0ba0f1b301" + }, + { + "ImportPath": "github.com/cenkalti/backoff", + "Rev": "4dc77674aceaabba2c7e3da25d4c823edfb73f99" + }, + { + "ImportPath": "github.com/dancannon/gorethink/encoding", + "Comment": "v1.x.x-1-g786f12a", + "Rev": "786f12ae730ea93485c4eb2c44b3ede6e1e8745f" + }, + { + "ImportPath": "github.com/dancannon/gorethink/ql2", + "Comment": "v1.x.x-1-g786f12a", + "Rev": "786f12ae730ea93485c4eb2c44b3ede6e1e8745f" + }, + { + "ImportPath": "github.com/dancannon/gorethink/types", + "Comment": "v1.x.x-1-g786f12a", + "Rev": "786f12ae730ea93485c4eb2c44b3ede6e1e8745f" + }, + { + "ImportPath": "github.com/eapache/go-resiliency/breaker", + "Comment": "v1.0.0-1-ged0319b", + "Rev": "ed0319b32e66e3295db52695ba3ee493e823fbfe" + }, + { + "ImportPath": "github.com/eapache/queue", + "Comment": "v1.0.2", + "Rev": "ded5959c0d4e360646dc9e9908cff48666781367" + }, + { + "ImportPath": "github.com/fsouza/go-dockerclient", + "Rev": "42d06e2b125654477366c320dcea99107a86e9c2" + }, + { + "ImportPath": "github.com/go-sql-driver/mysql", + "Comment": "v1.2-118-g3dd7008", + "Rev": "3dd7008ac1529aca1bcd8a9db75228a71ba23cac" + }, + { + "ImportPath": "github.com/gogo/protobuf/proto", + "Rev": "cabd153b69f71bab8b89fd667a2d9bb28c92ceb4" + }, + { + "ImportPath": "github.com/golang/protobuf/proto", + "Rev": "73aaaa9eb61d74fbf7e256ca586a3a565b308eea" + }, + { + "ImportPath": "github.com/golang/snappy", + "Rev": "723cc1e459b8eea2dea4583200fd60757d40097a" + }, + { + "ImportPath": "github.com/gonuts/go-shellquote", + "Rev": "e842a11b24c6abfb3dd27af69a17f482e4b483c2" + }, + { + "ImportPath": "github.com/hashicorp/go-msgpack/codec", + "Rev": "fa3f63826f7c23912c15263591e65d54d080b458" + }, + { + "ImportPath": "github.com/hashicorp/raft", + "Rev": "9b586e29edf1ed085b11da7772479ee45c433996" + }, + { + "ImportPath": "github.com/hashicorp/raft-boltdb", + "Rev": "d1e82c1ec3f15ee991f7cc7ffd5b67ff6f5bbaee" + }, + { + "ImportPath": "github.com/influxdb/influxdb/client", + "Comment": "v0.9.1-rc1-536-g1548f62", + "Rev": "1548f6289f2f0d96178f23e14bf72f6dae0eb437" + }, + { + "ImportPath": "github.com/influxdb/influxdb/influxql", + "Comment": "v0.9.1-rc1-536-g1548f62", + "Rev": "1548f6289f2f0d96178f23e14bf72f6dae0eb437" + }, + { + "ImportPath": "github.com/influxdb/influxdb/meta", + "Comment": "v0.9.1-rc1-536-g1548f62", + "Rev": "1548f6289f2f0d96178f23e14bf72f6dae0eb437" + }, + { + "ImportPath": "github.com/influxdb/influxdb/snapshot", + "Comment": "v0.9.1-rc1-536-g1548f62", + "Rev": "1548f6289f2f0d96178f23e14bf72f6dae0eb437" + }, + { + "ImportPath": "github.com/influxdb/influxdb/toml", + "Comment": "v0.9.1-rc1-536-g1548f62", + "Rev": "1548f6289f2f0d96178f23e14bf72f6dae0eb437" + }, + { + "ImportPath": "github.com/influxdb/influxdb/tsdb", + "Comment": "v0.9.1-rc1-536-g1548f62", + "Rev": "1548f6289f2f0d96178f23e14bf72f6dae0eb437" + }, + { + "ImportPath": "github.com/lib/pq", + "Comment": "go1.0-cutoff-59-gb269bd0", + "Rev": "b269bd035a727d6c1081f76e7a239a1b00674c40" + }, + { + "ImportPath": "github.com/matttproud/golang_protobuf_extensions/pbutil", + "Rev": "fc2b8d3a73c4867e51861bbdd5ae3c1f0869dd6a" + }, + { + "ImportPath": "github.com/naoina/go-stringutil", + "Rev": "360db0db4b01d34e12a2ec042c09e7d37fece761" + }, + { + "ImportPath": "github.com/naoina/toml", + "Rev": "5811abcabb29d6af0fdf060f96d328962bd3cd5e" + }, + { + "ImportPath": "github.com/prometheus/client_golang/extraction", + "Comment": "0.7.0-22-gbbd006b", + "Rev": "bbd006bc5e64ea2c807381d50263be5f230b427d" + }, + { + "ImportPath": "github.com/prometheus/client_golang/model", + "Comment": "0.7.0-22-gbbd006b", + "Rev": "bbd006bc5e64ea2c807381d50263be5f230b427d" + }, + { + "ImportPath": "github.com/prometheus/client_golang/text", + "Comment": "0.7.0-22-gbbd006b", + "Rev": "bbd006bc5e64ea2c807381d50263be5f230b427d" + }, + { + "ImportPath": "github.com/prometheus/client_model/go", + "Comment": "model-0.0.2-12-gfa8ad6f", + "Rev": "fa8ad6fec33561be4280a8f0514318c79d7f6cb6" + }, + { + "ImportPath": "github.com/samuel/go-zookeeper/zk", + "Rev": "5bb5cfc093ad18a28148c578f8632cfdb4d802e4" + }, + { + "ImportPath": "github.com/stretchr/objx", + "Rev": "cbeaeb16a013161a98496fad62933b1d21786672" + }, + { + "ImportPath": "github.com/stretchr/testify/assert", + "Comment": "v1.0-21-gf552045", + "Rev": "f5520455607c0233cb6d7b056f71b22c1d265ef1" + }, + { + "ImportPath": "github.com/stretchr/testify/mock", + "Comment": "v1.0-21-gf552045", + "Rev": "f5520455607c0233cb6d7b056f71b22c1d265ef1" + }, + { + "ImportPath": "github.com/stretchr/testify/require", + "Comment": "v1.0-21-gf552045", + "Rev": "f5520455607c0233cb6d7b056f71b22c1d265ef1" + }, + { + "ImportPath": "github.com/wvanbergen/kafka/consumergroup", + "Rev": "b0e5c20a0d7c3ccfd37a5965ae30a3a0fd15945d" + }, + { + "ImportPath": "github.com/wvanbergen/kazoo-go", + "Rev": "02a3868e9b87153285439cd27a39c0a2984a13af" + }, + { + "ImportPath": "golang.org/x/crypto/bcrypt", + "Rev": "173ce04bfaf66c7bb0fa9d5c0bfd93e773909dbd" + }, + { + "ImportPath": "golang.org/x/crypto/blowfish", + "Rev": "173ce04bfaf66c7bb0fa9d5c0bfd93e773909dbd" + }, + { + "ImportPath": "gopkg.in/dancannon/gorethink.v1", + "Comment": "v1.x.x", + "Rev": "8aca6ba2cc6e873299617d730fac0d7f6593113a" + }, + { + "ImportPath": "gopkg.in/mgo.v2", + "Comment": "r2015.06.03-3-g3569c88", + "Rev": "3569c88678d88179dcbd68d02ab081cbca3cd4d0" + } + ] +} diff --git a/Godeps/Readme b/Godeps/Readme new file mode 100644 index 0000000000000..4cdaa53d56d71 --- /dev/null +++ b/Godeps/Readme @@ -0,0 +1,5 @@ +This directory tree is generated automatically by godep. + +Please do not edit. + +See https://github.com/tools/godep for more information. diff --git a/Godeps/_workspace/.gitignore b/Godeps/_workspace/.gitignore new file mode 100644 index 0000000000000..f037d684ef2bd --- /dev/null +++ b/Godeps/_workspace/.gitignore @@ -0,0 +1,2 @@ +/pkg +/bin diff --git a/Godeps/_workspace/pkg/darwin_amd64/github.com/fsouza/go-dockerclient.a b/Godeps/_workspace/pkg/darwin_amd64/github.com/fsouza/go-dockerclient.a new file mode 100644 index 0000000000000..9ecf800aeda86 --- /dev/null +++ b/Godeps/_workspace/pkg/darwin_amd64/github.com/fsouza/go-dockerclient.a @@ -0,0 +1,1806 @@ +! +__.PKGDEF 0 0 0 644 98955 ` +go object darwin amd64 go1.4.2 X:precisestack + +$$ +package docker + import net "net" + import ioutil "io/ioutil" + import sync "sync" + import runtime "runtime" + import tls "crypto/tls" + import bufio "bufio" + import time "time" + import archive "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive" + import url "net/url" + import errors "errors" + import http "net/http" + import io "io" + import math "math" + import fileutils "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils" + import x509 "crypto/x509" + import base64 "encoding/base64" + import filepath "path/filepath" + import os "os" + import strconv "strconv" + import strings "strings" + import httputil "net/http/httputil" + import homedir "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir" + import stdcopy "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy" + import fmt "fmt" + import reflect "reflect" + import atomic "sync/atomic" + import opts "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts" + import json "encoding/json" + import bytes "bytes" + import path "path" + var @"".AuthParseError error + type @"".AuthConfiguration struct { Username string "json:\"username,omitempty\""; Password string "json:\"password,omitempty\""; Email string "json:\"email,omitempty\""; ServerAddress string "json:\"serveraddress,omitempty\"" } + type @"".AuthConfigurations struct { Configs map[string]@"".AuthConfiguration "json:\"configs\"" } + func @"".NewAuthConfigurationsFromDockerCfg () (? *@"".AuthConfigurations, ? error) + type @"io".Reader interface { Read(@"io".p []byte) (@"io".n int, @"io".err error) } + func @"".NewAuthConfigurations (@"".r·3 @"io".Reader) (? *@"".AuthConfigurations, ? error) + type @"".ChangeType int + const @"".ChangeModify @"".ChangeType = 0x0 + const @"".ChangeAdd @"".ChangeType = 0x1 + const @"".ChangeDelete @"".ChangeType = 0x2 + type @"".Change struct { Path string; Kind @"".ChangeType } + func (@"".change·2 *@"".Change) String () (? string) + var @"".ErrInvalidEndpoint error + var @"".ErrConnectionRefused error + type @"".APIVersion []int + func (@"".version·2 @"".APIVersion "esc:0x0") GreaterThan (@"".other·3 @"".APIVersion "esc:0x0") (? bool) + func (@"".version·2 @"".APIVersion "esc:0x0") GreaterThanOrEqualTo (@"".other·3 @"".APIVersion "esc:0x0") (? bool) + func (@"".version·2 @"".APIVersion "esc:0x0") LessThan (@"".other·3 @"".APIVersion "esc:0x0") (? bool) + func (@"".version·2 @"".APIVersion "esc:0x0") LessThanOrEqualTo (@"".other·3 @"".APIVersion "esc:0x0") (? bool) + func (@"".version·2 @"".APIVersion "esc:0x0") String () (? string) + func (@"".version·2 @"".APIVersion "esc:0x0") @"".compare (@"".other·3 @"".APIVersion "esc:0x0") (? int) + func @"".NewAPIVersion (@"".input·3 string) (? @"".APIVersion, ? error) + type @"io".Writer interface { Write(@"io".p []byte) (@"io".n int, @"io".err error) } + type @"net/http".keyValues struct { @"net/http".key string; @"net/http".values []string } + type @"net/http".headerSorter struct { @"net/http".kvs []@"net/http".keyValues } + func (@"net/http".s·2 *@"net/http".headerSorter "esc:0x0") Len () (? int) { return len(@"net/http".s·2.@"net/http".kvs) } + func (@"net/http".s·2 *@"net/http".headerSorter "esc:0x0") Less (@"net/http".i·3 int, @"net/http".j·4 int) (? bool) { return @"net/http".s·2.@"net/http".kvs[@"net/http".i·3].@"net/http".key < @"net/http".s·2.@"net/http".kvs[@"net/http".j·4].@"net/http".key } + func (@"net/http".s·1 *@"net/http".headerSorter "esc:0x0") Swap (@"net/http".i·2 int, @"net/http".j·3 int) { @"net/http".s·1.@"net/http".kvs[@"net/http".i·2], @"net/http".s·1.@"net/http".kvs[@"net/http".j·3] = @"net/http".s·1.@"net/http".kvs[@"net/http".j·3], @"net/http".s·1.@"net/http".kvs[@"net/http".i·2] } + type @"net/http".Header map[string][]string + func (@"net/http".h·1 @"net/http".Header "esc:0x0") Add (@"net/http".key·2 string, @"net/http".value·3 string) + func (@"net/http".h·1 @"net/http".Header "esc:0x0") Del (@"net/http".key·2 string "esc:0x0") + func (@"net/http".h·2 @"net/http".Header "esc:0x0") Get (@"net/http".key·3 string "esc:0x0") (? string) + func (@"net/http".h·1 @"net/http".Header "esc:0x0") Set (@"net/http".key·2 string, @"net/http".value·3 string) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") Write (@"net/http".w·3 @"io".Writer) (? error) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") WriteSubset (@"net/http".w·3 @"io".Writer, @"net/http".exclude·4 map[string]bool "esc:0x0") (? error) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") @"net/http".clone () (? @"net/http".Header) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") @"net/http".get (@"net/http".key·3 string "esc:0x0") (? string) { var @"net/http".v·4 []string; ; @"net/http".v·4 = @"net/http".h·2[@"net/http".key·3]; if len(@"net/http".v·4) > 0x0 { return @"net/http".v·4[0x0] }; return "" } + func (@"net/http".h·3 @"net/http".Header "esc:0x0") @"net/http".sortedKeyValues (@"net/http".exclude·4 map[string]bool "esc:0x0") (@"net/http".kvs·1 []@"net/http".keyValues, @"net/http".hs·2 *@"net/http".headerSorter) + type @"io".ReadCloser interface { Close() (? error); Read(@"io".p []byte) (@"io".n int, @"io".err error) } + type @"net/url".Userinfo struct { @"net/url".username string; @"net/url".password string; @"net/url".passwordSet bool } + func (@"net/url".u·3 *@"net/url".Userinfo "esc:0x1") Password () (? string, ? bool) { if @"net/url".u·3.@"net/url".passwordSet { return @"net/url".u·3.@"net/url".password, true }; return "", false } + func (@"net/url".u·2 *@"net/url".Userinfo "esc:0x1") String () (? string) + func (@"net/url".u·2 *@"net/url".Userinfo "esc:0x1") Username () (? string) { return @"net/url".u·2.@"net/url".username } + type @"net/url".Values map[string][]string + func (@"net/url".v·1 @"net/url".Values "esc:0x0") Add (@"net/url".key·2 string, @"net/url".value·3 string) { @"net/url".v·1[@"net/url".key·2] = append(@"net/url".v·1[@"net/url".key·2], @"net/url".value·3) } + func (@"net/url".v·1 @"net/url".Values "esc:0x0") Del (@"net/url".key·2 string "esc:0x0") { delete(@"net/url".v·1, @"net/url".key·2) } + func (@"net/url".v·2 @"net/url".Values "esc:0x0") Encode () (? string) + func (@"net/url".v·2 @"net/url".Values "esc:0x0") Get (@"net/url".key·3 string "esc:0x0") (? string) { if @"net/url".v·2 == nil { return "" }; var @"net/url".vs·4 []string; ; var @"net/url".ok·5 bool; ; @"net/url".vs·4, @"net/url".ok·5 = @"net/url".v·2[@"net/url".key·3]; if !@"net/url".ok·5 || len(@"net/url".vs·4) == 0x0 { return "" }; return @"net/url".vs·4[0x0] } + func (@"net/url".v·1 @"net/url".Values "esc:0x0") Set (@"net/url".key·2 string, @"net/url".value·3 string) { @"net/url".v·1[@"net/url".key·2] = ([]string{ 0x0:@"net/url".value·3 }) } + type @"net/url".URL struct { Scheme string; Opaque string; User *@"net/url".Userinfo; Host string; Path string; RawQuery string; Fragment string } + func (@"net/url".u·2 *@"net/url".URL "esc:0x0") IsAbs () (? bool) { return @"net/url".u·2.Scheme != "" } + func (@"net/url".u·3 *@"net/url".URL "esc:0x2") Parse (@"net/url".ref·4 string) (? *@"net/url".URL, ? error) + func (@"net/url".u·2 *@"net/url".URL) Query () (? @"net/url".Values) + func (@"net/url".u·2 *@"net/url".URL "esc:0x1") RequestURI () (? string) + func (@"net/url".u·2 *@"net/url".URL "esc:0x2") ResolveReference (@"net/url".ref·3 *@"net/url".URL "esc:0x2") (? *@"net/url".URL) + func (@"net/url".u·2 *@"net/url".URL "esc:0x0") String () (? string) + import multipart "mime/multipart" // indirect + import textproto "net/textproto" // indirect + type @"net/textproto".MIMEHeader map[string][]string + func (@"net/textproto".h·1 @"net/textproto".MIMEHeader "esc:0x0") Add (@"net/textproto".key·2 string, @"net/textproto".value·3 string) + func (@"net/textproto".h·1 @"net/textproto".MIMEHeader "esc:0x0") Del (@"net/textproto".key·2 string "esc:0x0") + func (@"net/textproto".h·2 @"net/textproto".MIMEHeader "esc:0x0") Get (@"net/textproto".key·3 string "esc:0x0") (? string) + func (@"net/textproto".h·1 @"net/textproto".MIMEHeader "esc:0x0") Set (@"net/textproto".key·2 string, @"net/textproto".value·3 string) + type @"mime/multipart".File interface { Close() (? error); Read(@"io".p []byte) (@"io".n int, @"io".err error); ReadAt(@"io".p []byte, @"io".off int64) (@"io".n int, @"io".err error); Seek(@"io".offset int64, @"io".whence int) (? int64, ? error) } + type @"mime/multipart".FileHeader struct { Filename string; Header @"net/textproto".MIMEHeader; @"mime/multipart".content []byte; @"mime/multipart".tmpfile string } + func (@"mime/multipart".fh·3 *@"mime/multipart".FileHeader) Open () (? @"mime/multipart".File, ? error) + type @"mime/multipart".Form struct { Value map[string][]string; File map[string][]*@"mime/multipart".FileHeader } + func (@"mime/multipart".f·2 *@"mime/multipart".Form "esc:0x0") RemoveAll () (? error) + type @"crypto/x509".SignatureAlgorithm int + type @"crypto/x509".PublicKeyAlgorithm int + import big "math/big" // indirect + type @"math/big".Word uintptr + type @"math/big".divisor struct { @"math/big".bbb @"math/big".nat; @"math/big".nbits int; @"math/big".ndigits int } + import rand "math/rand" // indirect + type @"math/rand".Source interface { Int63() (? int64); Seed(@"math/rand".seed int64) } + type @"math/rand".Rand struct { @"math/rand".src @"math/rand".Source } + func (@"math/rand".r·2 *@"math/rand".Rand) ExpFloat64 () (? float64) + func (@"math/rand".r·2 *@"math/rand".Rand) Float32 () (? float32) + func (@"math/rand".r·2 *@"math/rand".Rand) Float64 () (? float64) + func (@"math/rand".r·2 *@"math/rand".Rand) Int () (? int) + func (@"math/rand".r·2 *@"math/rand".Rand) Int31 () (? int32) + func (@"math/rand".r·2 *@"math/rand".Rand) Int31n (@"math/rand".n·3 int32) (? int32) + func (@"math/rand".r·2 *@"math/rand".Rand) Int63 () (? int64) + func (@"math/rand".r·2 *@"math/rand".Rand) Int63n (@"math/rand".n·3 int64) (? int64) + func (@"math/rand".r·2 *@"math/rand".Rand) Intn (@"math/rand".n·3 int) (? int) + func (@"math/rand".r·2 *@"math/rand".Rand) NormFloat64 () (? float64) + func (@"math/rand".r·2 *@"math/rand".Rand) Perm (@"math/rand".n·3 int) (? []int) + func (@"math/rand".r·1 *@"math/rand".Rand) Seed (@"math/rand".seed·2 int64) + func (@"math/rand".r·2 *@"math/rand".Rand) Uint32 () (? uint32) + type @"io".RuneScanner interface { ReadRune() (@"io".r rune, @"io".size int, @"io".err error); UnreadRune() (? error) } + type @"math/big".nat []@"math/big".Word + func (@"math/big".z·2 @"math/big".nat) @"math/big".add (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".and (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".andNot (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x0") @"math/big".bit (@"math/big".i·3 uint) (? uint) { var @"math/big".j·4 int; ; @"math/big".j·4 = int(@"math/big".i·3 / 0x40); if @"math/big".j·4 >= len(@"math/big".z·2) { return 0x0 }; return uint(@"math/big".z·2[@"math/big".j·4] >> (@"math/big".i·3 % 0x40) & @"math/big".Word(0x1)) } + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".bitLen () (? int) + func (@"math/big".z·2 @"math/big".nat "esc:0x0") @"math/big".bytes (@"math/big".buf·3 []byte "esc:0x0") (@"math/big".i·1 int) + func (@"math/big".z·1 @"math/big".nat "esc:0x0") @"math/big".clear () + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".cmp (@"math/big".y·3 @"math/big".nat "esc:0x0") (@"math/big".r·1 int) + func (@"math/big".q·1 @"math/big".nat) @"math/big".convertWords (@"math/big".s·2 []byte "esc:0x0", @"math/big".charset·3 string "esc:0x0", @"math/big".b·4 @"math/big".Word, @"math/big".ndigits·5 int, @"math/big".bb·6 @"math/big".Word, @"math/big".table·7 []@"math/big".divisor "esc:0x0") + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".decimalString () (? string) + func (@"math/big".z·3 @"math/big".nat) @"math/big".div (@"math/big".z2·4 @"math/big".nat, @"math/big".u·5 @"math/big".nat, @"math/big".v·6 @"math/big".nat) (@"math/big".q·1 @"math/big".nat, @"math/big".r·2 @"math/big".nat) + func (@"math/big".z·3 @"math/big".nat "esc:0x2") @"math/big".divLarge (@"math/big".u·4 @"math/big".nat, @"math/big".uIn·5 @"math/big".nat, @"math/big".v·6 @"math/big".nat) (@"math/big".q·1 @"math/big".nat, @"math/big".r·2 @"math/big".nat) + func (@"math/big".z·3 @"math/big".nat) @"math/big".divW (@"math/big".x·4 @"math/big".nat, @"math/big".y·5 @"math/big".Word) (@"math/big".q·1 @"math/big".nat, @"math/big".r·2 @"math/big".Word) + func (@"math/big".z·2 @"math/big".nat) @"math/big".expNN (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat "esc:0x0", @"math/big".m·5 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".expNNWindowed (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat "esc:0x0", @"math/big".m·5 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".expWW (@"math/big".x·3 @"math/big".Word, @"math/big".y·4 @"math/big".Word) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".make (@"math/big".n·3 int) (? @"math/big".nat) + func (@"math/big".x·2 @"math/big".nat) @"math/big".modW (@"math/big".d·3 @"math/big".Word) (@"math/big".r·1 @"math/big".Word) + func (@"math/big".z·2 @"math/big".nat) @"math/big".mul (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".mulAddWW (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".Word, @"math/big".r·5 @"math/big".Word) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".mulRange (@"math/big".a·3 uint64, @"math/big".b·4 uint64) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".norm () (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".or (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".n·2 @"math/big".nat) @"math/big".probablyPrime (@"math/big".reps·3 int) (? bool) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".random (@"math/big".rand·3 *@"math/rand".Rand, @"math/big".limit·4 @"math/big".nat "esc:0x0", @"math/big".n·5 int) (? @"math/big".nat) + func (@"math/big".z·4 @"math/big".nat) @"math/big".scan (@"math/big".r·5 @"io".RuneScanner, @"math/big".base·6 int) (? @"math/big".nat, ? int, ? error) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".set (@"math/big".x·3 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setBit (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".i·4 uint, @"math/big".b·5 uint) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setBytes (@"math/big".buf·3 []byte "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setUint64 (@"math/big".x·3 uint64) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setWord (@"math/big".x·3 @"math/big".Word) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".shl (@"math/big".x·3 @"math/big".nat, @"math/big".s·4 uint) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".shr (@"math/big".x·3 @"math/big".nat, @"math/big".s·4 uint) (? @"math/big".nat) + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".string (@"math/big".charset·3 string "esc:0x0") (? string) + func (@"math/big".z·2 @"math/big".nat) @"math/big".sub (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".trailingZeroBits () (? uint) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".xor (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + type @"fmt".State interface { Flag(@"fmt".c int) (? bool); Precision() (@"fmt".prec int, @"fmt".ok bool); Width() (@"fmt".wid int, @"fmt".ok bool); Write(@"fmt".b []byte) (@"fmt".ret int, @"fmt".err error) } + type @"fmt".ScanState interface { Read(@"fmt".buf []byte) (@"fmt".n int, @"fmt".err error); ReadRune() (@"fmt".r rune, @"fmt".size int, @"fmt".err error); SkipSpace(); Token(@"fmt".skipSpace bool, @"fmt".f func(? rune) (? bool)) (@"fmt".token []byte, @"fmt".err error); UnreadRune() (? error); Width() (@"fmt".wid int, @"fmt".ok bool) } + type @"math/big".Int struct { @"math/big".neg bool; @"math/big".abs @"math/big".nat } + func (@"math/big".z·2 *@"math/big".Int) Abs (@"math/big".x·3 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Add (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) And (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) AndNot (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Binomial (@"math/big".n·3 int64, @"math/big".k·4 int64) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int) Bit (@"math/big".i·3 int) (? uint) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") BitLen () (? int) + func (@"math/big".x·2 *@"math/big".Int "esc:0x1") Bits () (? []@"math/big".Word) { return @"math/big".x·2.@"math/big".abs } + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Bytes () (? []byte) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Cmp (@"math/big".y·3 *@"math/big".Int "esc:0x0") (@"math/big".r·1 int) + func (@"math/big".z·2 *@"math/big".Int) Div (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int) DivMod (@"math/big".x·4 *@"math/big".Int, @"math/big".y·5 *@"math/big".Int, @"math/big".m·6 *@"math/big".Int) (? *@"math/big".Int, ? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Exp (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int "esc:0x0", @"math/big".m·5 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".x·1 *@"math/big".Int "esc:0x0") Format (@"math/big".s·2 @"fmt".State, @"math/big".ch·3 rune) + func (@"math/big".z·2 *@"math/big".Int) GCD (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int, @"math/big".a·5 *@"math/big".Int, @"math/big".b·6 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) GobDecode (@"math/big".buf·3 []byte "esc:0x0") (? error) + func (@"math/big".x·3 *@"math/big".Int "esc:0x0") GobEncode () (? []byte, ? error) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Int64 () (? int64) + func (@"math/big".z·2 *@"math/big".Int) Lsh (@"math/big".x·3 *@"math/big".Int, @"math/big".n·4 uint) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int "esc:0x0") MarshalJSON () (? []byte, ? error) + func (@"math/big".z·3 *@"math/big".Int "esc:0x0") MarshalText () (@"math/big".text·1 []byte, @"math/big".err·2 error) + func (@"math/big".z·2 *@"math/big".Int) Mod (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) ModInverse (@"math/big".g·3 *@"math/big".Int, @"math/big".n·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Mul (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) MulRange (@"math/big".a·3 int64, @"math/big".b·4 int64) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Neg (@"math/big".x·3 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Not (@"math/big".x·3 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Or (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int) ProbablyPrime (@"math/big".n·3 int) (? bool) + func (@"math/big".z·2 *@"math/big".Int) Quo (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int) QuoRem (@"math/big".x·4 *@"math/big".Int, @"math/big".y·5 *@"math/big".Int, @"math/big".r·6 *@"math/big".Int) (? *@"math/big".Int, ? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Rand (@"math/big".rnd·3 *@"math/rand".Rand, @"math/big".n·4 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Rem (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Rsh (@"math/big".x·3 *@"math/big".Int, @"math/big".n·4 uint) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Scan (@"math/big".s·3 @"fmt".ScanState, @"math/big".ch·4 rune) (? error) + func (@"math/big".z·2 *@"math/big".Int) Set (@"math/big".x·3 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) SetBit (@"math/big".x·3 *@"math/big".Int, @"math/big".i·4 int, @"math/big".b·5 uint) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int "esc:0x2") SetBits (@"math/big".abs·3 []@"math/big".Word) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) SetBytes (@"math/big".buf·3 []byte "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) SetInt64 (@"math/big".x·3 int64) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int) SetString (@"math/big".s·4 string, @"math/big".base·5 int) (? *@"math/big".Int, ? bool) + func (@"math/big".z·2 *@"math/big".Int) SetUint64 (@"math/big".x·3 uint64) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Sign () (? int) { if len(@"math/big".x·2.@"math/big".abs) == 0x0 { return 0x0 }; if @"math/big".x·2.@"math/big".neg { return -0x1 }; return 0x1 } + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") String () (? string) + func (@"math/big".z·2 *@"math/big".Int) Sub (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Uint64 () (? uint64) + func (@"math/big".z·2 *@"math/big".Int) UnmarshalJSON (@"math/big".text·3 []byte) (? error) + func (@"math/big".z·2 *@"math/big".Int) UnmarshalText (@"math/big".text·3 []byte) (? error) + func (@"math/big".z·2 *@"math/big".Int) Xor (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) @"math/big".binaryGCD (@"math/big".a·3 *@"math/big".Int, @"math/big".b·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·4 *@"math/big".Int) @"math/big".scan (@"math/big".r·5 @"io".RuneScanner, @"math/big".base·6 int) (? *@"math/big".Int, ? int, ? error) + import pkix "crypto/x509/pkix" // indirect + import asn1 "encoding/asn1" // indirect + type @"encoding/asn1".ObjectIdentifier []int + func (@"encoding/asn1".oi·2 @"encoding/asn1".ObjectIdentifier "esc:0x0") Equal (@"encoding/asn1".other·3 @"encoding/asn1".ObjectIdentifier "esc:0x0") (? bool) + func (@"encoding/asn1".oi·2 @"encoding/asn1".ObjectIdentifier "esc:0x0") String () (? string) + type @"crypto/x509/pkix".AttributeTypeAndValue struct { Type @"encoding/asn1".ObjectIdentifier; Value interface {} } + type @"crypto/x509/pkix".RelativeDistinguishedNameSET []@"crypto/x509/pkix".AttributeTypeAndValue + type @"crypto/x509/pkix".RDNSequence []@"crypto/x509/pkix".RelativeDistinguishedNameSET + type @"crypto/x509/pkix".Name struct { Country []string; Organization []string; OrganizationalUnit []string; Locality []string; Province []string; StreetAddress []string; PostalCode []string; SerialNumber string; CommonName string; Names []@"crypto/x509/pkix".AttributeTypeAndValue } + func (@"crypto/x509/pkix".n·1 *@"crypto/x509/pkix".Name) FillFromRDNSequence (@"crypto/x509/pkix".rdns·2 *@"crypto/x509/pkix".RDNSequence "esc:0x0") + func (@"crypto/x509/pkix".n·2 @"crypto/x509/pkix".Name) ToRDNSequence () (@"crypto/x509/pkix".ret·1 @"crypto/x509/pkix".RDNSequence) + type @"time".zone struct { @"time".name string; @"time".offset int; @"time".isDST bool } + type @"time".zoneTrans struct { @"time".when int64; @"time".index uint8; @"time".isstd bool; @"time".isutc bool } + type @"time".Location struct { @"time".name string; @"time".zone []@"time".zone; @"time".tx []@"time".zoneTrans; @"time".cacheStart int64; @"time".cacheEnd int64; @"time".cacheZone *@"time".zone } + func (@"time".l·2 *@"time".Location "esc:0x0") String () (? string) + func (@"time".l·2 *@"time".Location "esc:0x0") @"time".firstZoneUsed () (? bool) + func (@"time".l·2 *@"time".Location "esc:0x2") @"time".get () (? *@"time".Location) + func (@"time".l·6 *@"time".Location "esc:0x1") @"time".lookup (@"time".sec·7 int64) (@"time".name·1 string, @"time".offset·2 int, @"time".isDST·3 bool, @"time".start·4 int64, @"time".end·5 int64) + func (@"time".l·2 *@"time".Location "esc:0x0") @"time".lookupFirstZone () (? int) + func (@"time".l·4 *@"time".Location "esc:0x0") @"time".lookupName (@"time".name·5 string "esc:0x0", @"time".unix·6 int64) (@"time".offset·1 int, @"time".isDST·2 bool, @"time".ok·3 bool) + type @"time".Duration int64 + func (@"time".d·2 @"time".Duration) Hours () (? float64) { var @"time".hour·3 @"time".Duration; ; @"time".hour·3 = @"time".d·2 / @"time".Duration(0x34630B8A000); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0x34630B8A000); return float64(@"time".hour·3) + float64(@"time".nsec·4) * 0x9C5FFF26ED75Fp-93 } + func (@"time".d·2 @"time".Duration) Minutes () (? float64) { var @"time".min·3 @"time".Duration; ; @"time".min·3 = @"time".d·2 / @"time".Duration(0xDF8475800); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0xDF8475800); return float64(@"time".min·3) + float64(@"time".nsec·4) * 0x9299FF347E9E9p-87 } + func (@"time".d·2 @"time".Duration) Nanoseconds () (? int64) { return int64(@"time".d·2) } + func (@"time".d·2 @"time".Duration) Seconds () (? float64) { var @"time".sec·3 @"time".Duration; ; @"time".sec·3 = @"time".d·2 / @"time".Duration(0x3B9ACA00); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0x3B9ACA00); return float64(@"time".sec·3) + float64(@"time".nsec·4) * 0x112E0BE826D695p-82 } + func (@"time".d·2 @"time".Duration) String () (? string) + type @"time".Month int + func (@"time".m·2 @"time".Month) String () (? string) { return @"time".months[@"time".m·2 - @"time".Month(0x1)] } + type @"time".Weekday int + func (@"time".d·2 @"time".Weekday) String () (? string) { return @"time".days[@"time".d·2] } + type @"time".Time struct { @"time".sec int64; @"time".nsec int32; @"time".loc *@"time".Location } + func (@"time".t·2 @"time".Time "esc:0x2") Add (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x2") AddDate (@"time".years·3 int, @"time".months·4 int, @"time".days·5 int) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") After (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec > @"time".u·3.@"time".sec || @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec > @"time".u·3.@"time".nsec } + func (@"time".t·2 @"time".Time "esc:0x0") Before (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec < @"time".u·3.@"time".sec || @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec < @"time".u·3.@"time".nsec } + func (@"time".t·4 @"time".Time "esc:0x0") Clock () (@"time".hour·1 int, @"time".min·2 int, @"time".sec·3 int) + func (@"time".t·4 @"time".Time "esc:0x0") Date () (@"time".year·1 int, @"time".month·2 @"time".Month, @"time".day·3 int) + func (@"time".t·2 @"time".Time "esc:0x0") Day () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") Equal (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec == @"time".u·3.@"time".nsec } + func (@"time".t·2 @"time".Time "esc:0x0") Format (@"time".layout·3 string "esc:0x0") (? string) + func (@"time".t·2 *@"time".Time "esc:0x0") GobDecode (@"time".data·3 []byte "esc:0x0") (? error) + func (@"time".t·3 @"time".Time "esc:0x0") GobEncode () (? []byte, ? error) + func (@"time".t·2 @"time".Time "esc:0x0") Hour () (? int) + func (@"time".t·3 @"time".Time "esc:0x0") ISOWeek () (@"time".year·1 int, @"time".week·2 int) + func (@"time".t·2 @"time".Time "esc:0x2") In (@"time".loc·3 *@"time".Location "esc:0x2") (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") IsZero () (? bool) { return @"time".t·2.@"time".sec == 0x0 && @"time".t·2.@"time".nsec == 0x0 } + func (@"time".t·2 @"time".Time "esc:0x2") Local () (? @"time".Time) { @"time".t·2.@"time".loc = @"time".Local; return @"time".t·2 } + func (@"time".t·2 @"time".Time "esc:0x2") Location () (? *@"time".Location) { var @"time".l·3 *@"time".Location; ; @"time".l·3 = @"time".t·2.@"time".loc; if @"time".l·3 == nil { @"time".l·3 = @"time".UTC }; return @"time".l·3 } + func (@"time".t·3 @"time".Time "esc:0x0") MarshalBinary () (? []byte, ? error) + func (@"time".t·3 @"time".Time "esc:0x0") MarshalJSON () (? []byte, ? error) + func (@"time".t·3 @"time".Time "esc:0x0") MarshalText () (? []byte, ? error) + func (@"time".t·2 @"time".Time "esc:0x0") Minute () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") Month () (? @"time".Month) + func (@"time".t·2 @"time".Time "esc:0x0") Nanosecond () (? int) { return int(@"time".t·2.@"time".nsec) } + func (@"time".t·2 @"time".Time "esc:0x2") Round (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") Second () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") String () (? string) + func (@"time".t·2 @"time".Time "esc:0x0") Sub (@"time".u·3 @"time".Time "esc:0x0") (? @"time".Duration) + func (@"time".t·2 @"time".Time "esc:0x2") Truncate (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x2") UTC () (? @"time".Time) { @"time".t·2.@"time".loc = @"time".UTC; return @"time".t·2 } + func (@"time".t·2 @"time".Time "esc:0x0") Unix () (? int64) { return @"time".t·2.@"time".sec + -0xE7791F700 } + func (@"time".t·2 @"time".Time "esc:0x0") UnixNano () (? int64) { return (@"time".t·2.@"time".sec + -0xE7791F700) * 0x3B9ACA00 + int64(@"time".t·2.@"time".nsec) } + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalBinary (@"time".data·3 []byte "esc:0x0") (? error) + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalJSON (@"time".data·3 []byte "esc:0x0") (@"time".err·1 error) + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalText (@"time".data·3 []byte "esc:0x0") (@"time".err·1 error) + func (@"time".t·2 @"time".Time "esc:0x0") Weekday () (? @"time".Weekday) + func (@"time".t·2 @"time".Time "esc:0x0") Year () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") YearDay () (? int) + func (@"time".t·3 @"time".Time "esc:0x0") Zone () (@"time".name·1 string, @"time".offset·2 int) + func (@"time".t·2 @"time".Time "esc:0x0") @"time".abs () (? uint64) + func (@"time".t·5 @"time".Time "esc:0x0") @"time".date (@"time".full·6 bool) (@"time".year·1 int, @"time".month·2 @"time".Month, @"time".day·3 int, @"time".yday·4 int) + func (@"time".t·4 @"time".Time "esc:0x1") @"time".locabs () (@"time".name·1 string, @"time".offset·2 int, @"time".abs·3 uint64) + type @"crypto/x509".KeyUsage int + type @"crypto/x509/pkix".Extension struct { Id @"encoding/asn1".ObjectIdentifier; Critical bool "asn1:\"optional\""; Value []byte } + type @"crypto/x509".ExtKeyUsage int + type @"net".IPMask []byte + func (@"net".m·3 @"net".IPMask "esc:0x0") Size () (@"net".ones·1 int, @"net".bits·2 int) + func (@"net".m·2 @"net".IPMask "esc:0x0") String () (? string) + type @"net".IP []byte + func (@"net".ip·2 @"net".IP "esc:0x0") DefaultMask () (? @"net".IPMask) + func (@"net".ip·2 @"net".IP "esc:0x0") Equal (@"net".x·3 @"net".IP "esc:0x0") (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsGlobalUnicast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsInterfaceLocalMulticast () (? bool) { return len(@"net".ip·2) == 0x10 && @"net".ip·2[0x0] == byte(0xFF) && @"net".ip·2[0x1] & byte(0xF) == byte(0x1) } + func (@"net".ip·2 @"net".IP "esc:0x0") IsLinkLocalMulticast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsLinkLocalUnicast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsLoopback () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsMulticast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsUnspecified () (? bool) + func (@"net".ip·3 @"net".IP "esc:0x0") MarshalText () (? []byte, ? error) + func (@"net".ip·2 @"net".IP "esc:0x0") Mask (@"net".mask·3 @"net".IPMask "esc:0x0") (? @"net".IP) + func (@"net".ip·2 @"net".IP "esc:0x0") String () (? string) + func (@"net".ip·2 @"net".IP "esc:0x2") To16 () (? @"net".IP) + func (@"net".ip·2 @"net".IP "esc:0x2") To4 () (? @"net".IP) + func (@"net".ip·2 *@"net".IP "esc:0x0") UnmarshalText (@"net".text·3 []byte "esc:0x0") (? error) + type @"encoding/asn1".RawContent []byte + type @"encoding/asn1".RawValue struct { Class int; Tag int; IsCompound bool; Bytes []byte; FullBytes []byte } + type @"crypto/x509/pkix".AlgorithmIdentifier struct { Algorithm @"encoding/asn1".ObjectIdentifier; Parameters @"encoding/asn1".RawValue "asn1:\"optional\"" } + type @"crypto/x509/pkix".RevokedCertificate struct { SerialNumber *@"math/big".Int; RevocationTime @"time".Time; Extensions []@"crypto/x509/pkix".Extension "asn1:\"optional\"" } + type @"crypto/x509/pkix".TBSCertificateList struct { Raw @"encoding/asn1".RawContent; Version int "asn1:\"optional,default:2\""; Signature @"crypto/x509/pkix".AlgorithmIdentifier; Issuer @"crypto/x509/pkix".RDNSequence; ThisUpdate @"time".Time; NextUpdate @"time".Time "asn1:\"optional\""; RevokedCertificates []@"crypto/x509/pkix".RevokedCertificate "asn1:\"optional\""; Extensions []@"crypto/x509/pkix".Extension "asn1:\"tag:0,optional,explicit\"" } + type @"encoding/asn1".BitString struct { Bytes []byte; BitLength int } + func (@"encoding/asn1".b·2 @"encoding/asn1".BitString "esc:0x0") At (@"encoding/asn1".i·3 int) (? int) { if @"encoding/asn1".i·3 < 0x0 || @"encoding/asn1".i·3 >= @"encoding/asn1".b·2.BitLength { return 0x0 }; var @"encoding/asn1".x·4 int; ; @"encoding/asn1".x·4 = @"encoding/asn1".i·3 / 0x8; var @"encoding/asn1".y·5 uint; ; @"encoding/asn1".y·5 = 0x7 - uint(@"encoding/asn1".i·3 % 0x8); return int(@"encoding/asn1".b·2.Bytes[@"encoding/asn1".x·4] >> @"encoding/asn1".y·5) & 0x1 } + func (@"encoding/asn1".b·2 @"encoding/asn1".BitString "esc:0x2") RightAlign () (? []byte) + type @"crypto/x509/pkix".CertificateList struct { TBSCertList @"crypto/x509/pkix".TBSCertificateList; SignatureAlgorithm @"crypto/x509/pkix".AlgorithmIdentifier; SignatureValue @"encoding/asn1".BitString } + func (@"crypto/x509/pkix".certList·2 *@"crypto/x509/pkix".CertificateList "esc:0x0") HasExpired (@"crypto/x509/pkix".now·3 @"time".Time "esc:0x0") (? bool) + type @"crypto/x509".CertPool struct { @"crypto/x509".bySubjectKeyId map[string][]int; @"crypto/x509".byName map[string][]int; @"crypto/x509".certs []*@"crypto/x509".Certificate } + func (@"crypto/x509".s·1 *@"crypto/x509".CertPool) AddCert (@"crypto/x509".cert·2 *@"crypto/x509".Certificate) + func (@"crypto/x509".s·2 *@"crypto/x509".CertPool) AppendCertsFromPEM (@"crypto/x509".pemCerts·3 []byte) (@"crypto/x509".ok·1 bool) + func (@"crypto/x509".s·2 *@"crypto/x509".CertPool "esc:0x0") Subjects () (@"crypto/x509".res·1 [][]byte) + func (@"crypto/x509".s·4 *@"crypto/x509".CertPool "esc:0x0") @"crypto/x509".findVerifiedParents (@"crypto/x509".cert·5 *@"crypto/x509".Certificate) (@"crypto/x509".parents·1 []int, @"crypto/x509".errCert·2 *@"crypto/x509".Certificate, @"crypto/x509".err·3 error) + type @"crypto/x509".VerifyOptions struct { DNSName string; Intermediates *@"crypto/x509".CertPool; Roots *@"crypto/x509".CertPool; CurrentTime @"time".Time; KeyUsages []@"crypto/x509".ExtKeyUsage } + type @"crypto/x509".Certificate struct { Raw []byte; RawTBSCertificate []byte; RawSubjectPublicKeyInfo []byte; RawSubject []byte; RawIssuer []byte; Signature []byte; SignatureAlgorithm @"crypto/x509".SignatureAlgorithm; PublicKeyAlgorithm @"crypto/x509".PublicKeyAlgorithm; PublicKey interface {}; Version int; SerialNumber *@"math/big".Int; Issuer @"crypto/x509/pkix".Name; Subject @"crypto/x509/pkix".Name; NotBefore @"time".Time; NotAfter @"time".Time; KeyUsage @"crypto/x509".KeyUsage; Extensions []@"crypto/x509/pkix".Extension; ExtraExtensions []@"crypto/x509/pkix".Extension; ExtKeyUsage []@"crypto/x509".ExtKeyUsage; UnknownExtKeyUsage []@"encoding/asn1".ObjectIdentifier; BasicConstraintsValid bool; IsCA bool; MaxPathLen int; MaxPathLenZero bool; SubjectKeyId []byte; AuthorityKeyId []byte; OCSPServer []string; IssuingCertificateURL []string; DNSNames []string; EmailAddresses []string; IPAddresses []@"net".IP; PermittedDNSDomainsCritical bool; PermittedDNSDomains []string; CRLDistributionPoints []string; PolicyIdentifiers []@"encoding/asn1".ObjectIdentifier } + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate) CheckCRLSignature (@"crypto/x509".crl·3 *@"crypto/x509/pkix".CertificateList) (@"crypto/x509".err·1 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate) CheckSignature (@"crypto/x509".algo·3 @"crypto/x509".SignatureAlgorithm, @"crypto/x509".signed·4 []byte, @"crypto/x509".signature·5 []byte) (@"crypto/x509".err·1 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate) CheckSignatureFrom (@"crypto/x509".parent·3 *@"crypto/x509".Certificate) (@"crypto/x509".err·1 error) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate) CreateCRL (@"crypto/x509".rand·4 @"io".Reader, @"crypto/x509".priv·5 interface {}, @"crypto/x509".revokedCerts·6 []@"crypto/x509/pkix".RevokedCertificate, @"crypto/x509".now·7 @"time".Time, @"crypto/x509".expiry·8 @"time".Time) (@"crypto/x509".crlBytes·1 []byte, @"crypto/x509".err·2 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate "esc:0x0") Equal (@"crypto/x509".other·3 *@"crypto/x509".Certificate "esc:0x0") (? bool) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate) Verify (@"crypto/x509".opts·4 @"crypto/x509".VerifyOptions "esc:0x4") (@"crypto/x509".chains·1 [][]*@"crypto/x509".Certificate, @"crypto/x509".err·2 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate "esc:0x2") VerifyHostname (@"crypto/x509".h·3 string "esc:0x2") (? error) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate) @"crypto/x509".buildChains (@"crypto/x509".cache·4 map[int][][]*@"crypto/x509".Certificate "esc:0x0", @"crypto/x509".currentChain·5 []*@"crypto/x509".Certificate "esc:0x0", @"crypto/x509".opts·6 *@"crypto/x509".VerifyOptions "esc:0x0") (@"crypto/x509".chains·1 [][]*@"crypto/x509".Certificate, @"crypto/x509".err·2 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate "esc:0x2") @"crypto/x509".isValid (@"crypto/x509".certType·3 int, @"crypto/x509".currentChain·4 []*@"crypto/x509".Certificate "esc:0x0", @"crypto/x509".opts·5 *@"crypto/x509".VerifyOptions "esc:0x0") (? error) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate "esc:0x0") @"crypto/x509".systemVerify (@"crypto/x509".opts·4 *@"crypto/x509".VerifyOptions "esc:0x0") (@"crypto/x509".chains·1 [][]*@"crypto/x509".Certificate, @"crypto/x509".err·2 error) { return nil, nil } + type @"crypto/tls".ConnectionState struct { Version uint16; HandshakeComplete bool; DidResume bool; CipherSuite uint16; NegotiatedProtocol string; NegotiatedProtocolIsMutual bool; ServerName string; PeerCertificates []*@"crypto/x509".Certificate; VerifiedChains [][]*@"crypto/x509".Certificate; TLSUnique []byte } + type @"net/http".Cookie struct { Name string; Value string; Path string; Domain string; Expires @"time".Time; RawExpires string; MaxAge int; Secure bool; HttpOnly bool; Raw string; Unparsed []string } + func (@"net/http".c·2 *@"net/http".Cookie) String () (? string) + type @"bufio".Reader struct { @"bufio".buf []byte; @"bufio".rd @"io".Reader; @"bufio".r int; @"bufio".w int; @"bufio".err error; @"bufio".lastByte int; @"bufio".lastRuneSize int } + func (@"bufio".b·2 *@"bufio".Reader "esc:0x0") Buffered () (? int) { return @"bufio".b·2.@"bufio".w - @"bufio".b·2.@"bufio".r } + func (@"bufio".b·3 *@"bufio".Reader) Peek (@"bufio".n·4 int) (? []byte, ? error) + func (@"bufio".b·3 *@"bufio".Reader) Read (@"bufio".p·4 []byte) (@"bufio".n·1 int, @"bufio".err·2 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadByte () (@"bufio".c·1 byte, @"bufio".err·2 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadBytes (@"bufio".delim·4 byte) (@"bufio".line·1 []byte, @"bufio".err·2 error) + func (@"bufio".b·4 *@"bufio".Reader) ReadLine () (@"bufio".line·1 []byte, @"bufio".isPrefix·2 bool, @"bufio".err·3 error) + func (@"bufio".b·4 *@"bufio".Reader) ReadRune () (@"bufio".r·1 rune, @"bufio".size·2 int, @"bufio".err·3 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadSlice (@"bufio".delim·4 byte) (@"bufio".line·1 []byte, @"bufio".err·2 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadString (@"bufio".delim·4 byte) (@"bufio".line·1 string, @"bufio".err·2 error) + func (@"bufio".b·1 *@"bufio".Reader) Reset (@"bufio".r·2 @"io".Reader) + func (@"bufio".b·2 *@"bufio".Reader "esc:0x0") UnreadByte () (? error) + func (@"bufio".b·2 *@"bufio".Reader "esc:0x0") UnreadRune () (? error) { if @"bufio".b·2.@"bufio".lastRuneSize < 0x0 || @"bufio".b·2.@"bufio".r < @"bufio".b·2.@"bufio".lastRuneSize { return @"bufio".ErrInvalidUnreadRune }; @"bufio".b·2.@"bufio".r -= @"bufio".b·2.@"bufio".lastRuneSize; @"bufio".b·2.@"bufio".lastByte = -0x1; @"bufio".b·2.@"bufio".lastRuneSize = -0x1; return nil } + func (@"bufio".b·3 *@"bufio".Reader) WriteTo (@"bufio".w·4 @"io".Writer) (@"bufio".n·1 int64, @"bufio".err·2 error) + func (@"bufio".b·1 *@"bufio".Reader) @"bufio".fill () + func (@"bufio".b·2 *@"bufio".Reader "esc:0x1") @"bufio".readErr () (? error) { var @"bufio".err·3 error; ; @"bufio".err·3 = @"bufio".b·2.@"bufio".err; @"bufio".b·2.@"bufio".err = nil; return @"bufio".err·3 } + func (@"bufio".b·1 *@"bufio".Reader "esc:0x0") @"bufio".reset (@"bufio".buf·2 []byte, @"bufio".r·3 @"io".Reader) { *@"bufio".b·1 = (@"bufio".Reader{ @"bufio".buf:@"bufio".buf·2, @"bufio".rd:@"bufio".r·3, @"bufio".lastByte:-0x1, @"bufio".lastRuneSize:-0x1 }) } + func (@"bufio".b·3 *@"bufio".Reader) @"bufio".writeBuf (@"bufio".w·4 @"io".Writer) (? int64, ? error) + type @"bytes".readOp int + type @"bytes".Buffer struct { @"bytes".buf []byte; @"bytes".off int; @"bytes".runeBytes [4]byte; @"bytes".bootstrap [64]byte; @"bytes".lastRead @"bytes".readOp } + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x1") Bytes () (? []byte) { return @"bytes".b·2.@"bytes".buf[@"bytes".b·2.@"bytes".off:] } + func (@"bytes".b·1 *@"bytes".Buffer) Grow (@"bytes".n·2 int) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") Len () (? int) { return len(@"bytes".b·2.@"bytes".buf) - @"bytes".b·2.@"bytes".off } + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x1") Next (@"bytes".n·3 int) (? []byte) + func (@"bytes".b·3 *@"bytes".Buffer) Read (@"bytes".p·4 []byte "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) ReadByte () (@"bytes".c·1 byte, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x0") ReadBytes (@"bytes".delim·4 byte) (@"bytes".line·1 []byte, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) ReadFrom (@"bytes".r·4 @"io".Reader) (@"bytes".n·1 int64, @"bytes".err·2 error) + func (@"bytes".b·4 *@"bytes".Buffer) ReadRune () (@"bytes".r·1 rune, @"bytes".size·2 int, @"bytes".err·3 error) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x0") ReadString (@"bytes".delim·4 byte) (@"bytes".line·1 string, @"bytes".err·2 error) + func (@"bytes".b·1 *@"bytes".Buffer) Reset () + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") String () (? string) { if @"bytes".b·2 == nil { return "" }; return string(@"bytes".b·2.@"bytes".buf[@"bytes".b·2.@"bytes".off:]) } + func (@"bytes".b·1 *@"bytes".Buffer) Truncate (@"bytes".n·2 int) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") UnreadByte () (? error) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") UnreadRune () (? error) + func (@"bytes".b·3 *@"bytes".Buffer) Write (@"bytes".p·4 []byte "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·2 *@"bytes".Buffer) WriteByte (@"bytes".c·3 byte) (? error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteRune (@"bytes".r·4 rune) (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteString (@"bytes".s·4 string "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteTo (@"bytes".w·4 @"io".Writer) (@"bytes".n·1 int64, @"bytes".err·2 error) + func (@"bytes".b·2 *@"bytes".Buffer) @"bytes".grow (@"bytes".n·3 int) (? int) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x1") @"bytes".readSlice (@"bytes".delim·4 byte) (@"bytes".line·1 []byte, @"bytes".err·2 error) + type @"mime/multipart".Part struct { Header @"net/textproto".MIMEHeader; @"mime/multipart".buffer *@"bytes".Buffer; @"mime/multipart".mr *@"mime/multipart".Reader; @"mime/multipart".bytesRead int; @"mime/multipart".disposition string; @"mime/multipart".dispositionParams map[string]string; @"mime/multipart".r @"io".Reader } + func (@"mime/multipart".p·2 *@"mime/multipart".Part) Close () (? error) + func (@"mime/multipart".p·2 *@"mime/multipart".Part "esc:0x0") FileName () (? string) + func (@"mime/multipart".p·2 *@"mime/multipart".Part "esc:0x0") FormName () (? string) + func (@"mime/multipart".p·3 *@"mime/multipart".Part) Read (@"mime/multipart".d·4 []byte) (@"mime/multipart".n·1 int, @"mime/multipart".err·2 error) + func (@"mime/multipart".p·1 *@"mime/multipart".Part "esc:0x0") @"mime/multipart".parseContentDisposition () + func (@"mime/multipart".bp·2 *@"mime/multipart".Part) @"mime/multipart".populateHeaders () (? error) + type @"mime/multipart".Reader struct { @"mime/multipart".bufReader *@"bufio".Reader; @"mime/multipart".currentPart *@"mime/multipart".Part; @"mime/multipart".partsRead int; @"mime/multipart".nl []byte; @"mime/multipart".nlDashBoundary []byte; @"mime/multipart".dashBoundaryDash []byte; @"mime/multipart".dashBoundary []byte } + func (@"mime/multipart".r·3 *@"mime/multipart".Reader) NextPart () (? *@"mime/multipart".Part, ? error) + func (@"mime/multipart".r·3 *@"mime/multipart".Reader) ReadForm (@"mime/multipart".maxMemory·4 int64) (@"mime/multipart".f·1 *@"mime/multipart".Form, @"mime/multipart".err·2 error) + func (@"mime/multipart".mr·2 *@"mime/multipart".Reader) @"mime/multipart".isBoundaryDelimiterLine (@"mime/multipart".line·3 []byte "esc:0x0") (@"mime/multipart".ret·1 bool) + func (@"mime/multipart".mr·2 *@"mime/multipart".Reader "esc:0x0") @"mime/multipart".isFinalBoundary (@"mime/multipart".line·3 []byte "esc:0x0") (? bool) + func (@"mime/multipart".mr·2 *@"mime/multipart".Reader "esc:0x0") @"mime/multipart".peekBufferIsEmptyPart (@"mime/multipart".peek·3 []byte "esc:0x0") (? bool) + type @"net/http".Request struct { Method string; URL *@"net/url".URL; Proto string; ProtoMajor int; ProtoMinor int; Header @"net/http".Header; Body @"io".ReadCloser; ContentLength int64; TransferEncoding []string; Close bool; Host string; Form @"net/url".Values; PostForm @"net/url".Values; MultipartForm *@"mime/multipart".Form; Trailer @"net/http".Header; RemoteAddr string; RequestURI string; TLS *@"crypto/tls".ConnectionState } + func (@"net/http".r·1 *@"net/http".Request "esc:0x0") AddCookie (@"net/http".c·2 *@"net/http".Cookie) + func (@"net/http".r·4 *@"net/http".Request "esc:0x0") BasicAuth () (@"net/http".username·1 string, @"net/http".password·2 string, @"net/http".ok·3 bool) + func (@"net/http".r·3 *@"net/http".Request "esc:0x0") Cookie (@"net/http".name·4 string "esc:0x0") (? *@"net/http".Cookie, ? error) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") Cookies () (? []*@"net/http".Cookie) + func (@"net/http".r·4 *@"net/http".Request) FormFile (@"net/http".key·5 string "esc:0x0") (? @"mime/multipart".File, ? *@"mime/multipart".FileHeader, ? error) + func (@"net/http".r·2 *@"net/http".Request) FormValue (@"net/http".key·3 string "esc:0x0") (? string) + func (@"net/http".r·3 *@"net/http".Request) MultipartReader () (? *@"mime/multipart".Reader, ? error) + func (@"net/http".r·2 *@"net/http".Request) ParseForm () (? error) + func (@"net/http".r·2 *@"net/http".Request) ParseMultipartForm (@"net/http".maxMemory·3 int64) (? error) + func (@"net/http".r·2 *@"net/http".Request) PostFormValue (@"net/http".key·3 string "esc:0x0") (? string) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") ProtoAtLeast (@"net/http".major·3 int, @"net/http".minor·4 int) (? bool) { return @"net/http".r·2.ProtoMajor > @"net/http".major·3 || @"net/http".r·2.ProtoMajor == @"net/http".major·3 && @"net/http".r·2.ProtoMinor >= @"net/http".minor·4 } + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") Referer () (? string) + func (@"net/http".r·1 *@"net/http".Request "esc:0x0") SetBasicAuth (@"net/http".username·2 string "esc:0x0", @"net/http".password·3 string "esc:0x0") + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") UserAgent () (? string) + func (@"net/http".r·2 *@"net/http".Request) Write (@"net/http".w·3 @"io".Writer) (? error) + func (@"net/http".r·2 *@"net/http".Request) WriteProxy (@"net/http".w·3 @"io".Writer) (? error) + func (@"net/http".r·1 *@"net/http".Request) @"net/http".closeBody () + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") @"net/http".expectsContinue () (? bool) + func (@"net/http".r·3 *@"net/http".Request) @"net/http".multipartReader () (? *@"mime/multipart".Reader, ? error) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") @"net/http".wantsClose () (? bool) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") @"net/http".wantsHttp10KeepAlive () (? bool) + func (@"net/http".req·2 *@"net/http".Request) @"net/http".write (@"net/http".w·3 @"io".Writer, @"net/http".usingProxy·4 bool, @"net/http".extraHeaders·5 @"net/http".Header "esc:0x0") (? error) + type @"net/http".Response struct { Status string; StatusCode int; Proto string; ProtoMajor int; ProtoMinor int; Header @"net/http".Header; Body @"io".ReadCloser; ContentLength int64; TransferEncoding []string; Close bool; Trailer @"net/http".Header; Request *@"net/http".Request; TLS *@"crypto/tls".ConnectionState } + func (@"net/http".r·2 *@"net/http".Response "esc:0x0") Cookies () (? []*@"net/http".Cookie) + func (@"net/http".r·3 *@"net/http".Response "esc:0x1") Location () (? *@"net/url".URL, ? error) + func (@"net/http".r·2 *@"net/http".Response "esc:0x0") ProtoAtLeast (@"net/http".major·3 int, @"net/http".minor·4 int) (? bool) { return @"net/http".r·2.ProtoMajor > @"net/http".major·3 || @"net/http".r·2.ProtoMajor == @"net/http".major·3 && @"net/http".r·2.ProtoMinor >= @"net/http".minor·4 } + func (@"net/http".r·2 *@"net/http".Response) Write (@"net/http".w·3 @"io".Writer) (? error) + type @"net/http".RoundTripper interface { RoundTrip(? *@"net/http".Request) (? *@"net/http".Response, ? error) } + type @"net/http".CookieJar interface { Cookies(@"net/http".u *@"net/url".URL) (? []*@"net/http".Cookie); SetCookies(@"net/http".u *@"net/url".URL, @"net/http".cookies []*@"net/http".Cookie) } + type @"net/http".Client struct { Transport @"net/http".RoundTripper; CheckRedirect func(@"net/http".req *@"net/http".Request, @"net/http".via []*@"net/http".Request) (? error); Jar @"net/http".CookieJar; Timeout @"time".Duration } + func (@"net/http".c·3 *@"net/http".Client) Do (@"net/http".req·4 *@"net/http".Request) (@"net/http".resp·1 *@"net/http".Response, @"net/http".err·2 error) + func (@"net/http".c·3 *@"net/http".Client) Get (@"net/http".url·4 string) (@"net/http".resp·1 *@"net/http".Response, @"net/http".err·2 error) + func (@"net/http".c·3 *@"net/http".Client) Head (@"net/http".url·4 string) (@"net/http".resp·1 *@"net/http".Response, @"net/http".err·2 error) + func (@"net/http".c·3 *@"net/http".Client) Post (@"net/http".url·4 string, @"net/http".bodyType·5 string, @"net/http".body·6 @"io".Reader) (@"net/http".resp·1 *@"net/http".Response, @"net/http".err·2 error) + func (@"net/http".c·3 *@"net/http".Client) PostForm (@"net/http".url·4 string, @"net/http".data·5 @"net/url".Values "esc:0x0") (@"net/http".resp·1 *@"net/http".Response, @"net/http".err·2 error) + func (@"net/http".c·3 *@"net/http".Client) @"net/http".doFollowingRedirects (@"net/http".ireq·4 *@"net/http".Request, @"net/http".shouldRedirect·5 func(? int) (? bool) "esc:0x0") (@"net/http".resp·1 *@"net/http".Response, @"net/http".err·2 error) + func (@"net/http".c·3 *@"net/http".Client) @"net/http".send (@"net/http".req·4 *@"net/http".Request) (? *@"net/http".Response, ? error) + func (@"net/http".c·2 *@"net/http".Client "esc:0x1") @"net/http".transport () (? @"net/http".RoundTripper) { if @"net/http".c·2.Transport != nil { return @"net/http".c·2.Transport }; return @"net/http".DefaultTransport } + import crypto "crypto" // indirect + type @"crypto".PrivateKey interface {} + type @"crypto/tls".Certificate struct { Certificate [][]byte; PrivateKey @"crypto".PrivateKey; OCSPStaple []byte; Leaf *@"crypto/x509".Certificate } + type @"crypto/tls".CurveID uint16 + type @"crypto/tls".ClientHelloInfo struct { CipherSuites []uint16; ServerName string; SupportedCurves []@"crypto/tls".CurveID; SupportedPoints []uint8 } + type @"crypto/tls".ClientAuthType int + type @"crypto/tls".ClientSessionState struct { @"crypto/tls".sessionTicket []uint8; @"crypto/tls".vers uint16; @"crypto/tls".cipherSuite uint16; @"crypto/tls".masterSecret []byte; @"crypto/tls".serverCertificates []*@"crypto/x509".Certificate } + type @"crypto/tls".ClientSessionCache interface { Get(@"crypto/tls".sessionKey string) (@"crypto/tls".session *@"crypto/tls".ClientSessionState, @"crypto/tls".ok bool); Put(@"crypto/tls".sessionKey string, @"crypto/tls".cs *@"crypto/tls".ClientSessionState) } + type @"sync".Mutex struct { @"sync".state int32; @"sync".sema uint32 } + func (@"sync".m·1 *@"sync".Mutex) Lock () + func (@"sync".m·1 *@"sync".Mutex) Unlock () + type @"sync".Once struct { @"sync".m @"sync".Mutex; @"sync".done uint32 } + func (@"sync".o·1 *@"sync".Once) Do (@"sync".f·2 func() "esc:0x0") + type @"crypto/tls".Config struct { Rand @"io".Reader; Time func() (? @"time".Time); Certificates []@"crypto/tls".Certificate; NameToCertificate map[string]*@"crypto/tls".Certificate; GetCertificate func(@"crypto/tls".clientHello *@"crypto/tls".ClientHelloInfo) (? *@"crypto/tls".Certificate, ? error); RootCAs *@"crypto/x509".CertPool; NextProtos []string; ServerName string; ClientAuth @"crypto/tls".ClientAuthType; ClientCAs *@"crypto/x509".CertPool; InsecureSkipVerify bool; CipherSuites []uint16; PreferServerCipherSuites bool; SessionTicketsDisabled bool; SessionTicketKey [32]byte; ClientSessionCache @"crypto/tls".ClientSessionCache; MinVersion uint16; MaxVersion uint16; CurvePreferences []@"crypto/tls".CurveID; @"crypto/tls".serverInitOnce @"sync".Once } + func (@"crypto/tls".c·1 *@"crypto/tls".Config) BuildNameToCertificate () + func (@"crypto/tls".c·2 *@"crypto/tls".Config "esc:0x1") @"crypto/tls".cipherSuites () (? []uint16) + func (@"crypto/tls".c·2 *@"crypto/tls".Config "esc:0x1") @"crypto/tls".curvePreferences () (? []@"crypto/tls".CurveID) { if @"crypto/tls".c·2 == nil || len(@"crypto/tls".c·2.CurvePreferences) == 0x0 { return @"crypto/tls".defaultCurvePreferences }; return @"crypto/tls".c·2.CurvePreferences } + func (@"crypto/tls".c·3 *@"crypto/tls".Config "esc:0x1") @"crypto/tls".getCertificate (@"crypto/tls".clientHello·4 *@"crypto/tls".ClientHelloInfo) (? *@"crypto/tls".Certificate, ? error) + func (@"crypto/tls".c·2 *@"crypto/tls".Config "esc:0x0") @"crypto/tls".maxVersion () (? uint16) { if @"crypto/tls".c·2 == nil || @"crypto/tls".c·2.MaxVersion == 0x0 { return 0x303 }; return @"crypto/tls".c·2.MaxVersion } + func (@"crypto/tls".c·2 *@"crypto/tls".Config "esc:0x0") @"crypto/tls".minVersion () (? uint16) { if @"crypto/tls".c·2 == nil || @"crypto/tls".c·2.MinVersion == 0x0 { return 0x300 }; return @"crypto/tls".c·2.MinVersion } + func (@"crypto/tls".c·3 *@"crypto/tls".Config "esc:0x0") @"crypto/tls".mutualVersion (@"crypto/tls".vers·4 uint16) (? uint16, ? bool) + func (@"crypto/tls".c·2 *@"crypto/tls".Config "esc:0x1") @"crypto/tls".rand () (? @"io".Reader) { var @"crypto/tls".r·3 @"io".Reader; ; @"crypto/tls".r·3 = @"crypto/tls".c·2.Rand; if @"crypto/tls".r·3 == nil { return @"crypto/rand".Reader }; return @"crypto/tls".r·3 } + func (@"crypto/tls".c·1 *@"crypto/tls".Config) @"crypto/tls".serverInit () + func (@"crypto/tls".c·2 *@"crypto/tls".Config "esc:0x0") @"crypto/tls".time () (? @"time".Time) + type @"sync".Locker interface { Lock(); Unlock() } + type @"sync".RWMutex struct { @"sync".w @"sync".Mutex; @"sync".writerSem uint32; @"sync".readerSem uint32; @"sync".readerCount int32; @"sync".readerWait int32 } + func (@"sync".rw·1 *@"sync".RWMutex) Lock () + func (@"sync".rw·1 *@"sync".RWMutex) RLock () + func (@"sync".rw·2 *@"sync".RWMutex "esc:0x2") RLocker () (? @"sync".Locker) { return (*@"sync".rlocker)(@"sync".rw·2) } + func (@"sync".rw·1 *@"sync".RWMutex) RUnlock () + func (@"sync".rw·1 *@"sync".RWMutex) Unlock () + type @"sync".WaitGroup struct { @"sync".m @"sync".Mutex; @"sync".counter int32; @"sync".waiters int32; @"sync".sema *uint32 } + func (@"sync".wg·1 *@"sync".WaitGroup) Add (@"sync".delta·2 int) + func (@"sync".wg·1 *@"sync".WaitGroup) Done () + func (@"sync".wg·1 *@"sync".WaitGroup) Wait () + type @"".APIEvents struct { Status string "json:\"Status,omitempty\" yaml:\"Status,omitempty\""; ID string "json:\"ID,omitempty\" yaml:\"ID,omitempty\""; From string "json:\"From,omitempty\" yaml:\"From,omitempty\""; Time int64 "json:\"Time,omitempty\" yaml:\"Time,omitempty\"" } + type @"".eventMonitoringState struct { ? @"sync".RWMutex; ? @"sync".WaitGroup; @"".enabled bool; @"".lastSeen *int64; C chan *@"".APIEvents; @"".errC chan error; @"".listeners []chan<- *@"".APIEvents } + func (@"".eventState·2 *@"".eventMonitoringState) @"".addListener (@"".listener·3 chan<- *@"".APIEvents) (? error) + func (@"".eventState·1 *@"".eventMonitoringState) @"".closeListeners () + func (@"".eventState·2 *@"".eventMonitoringState) @"".connectWithRetry (@"".c·3 *@"".Client) (? error) + func (@"".eventState·2 *@"".eventMonitoringState) @"".disableEventMonitoring () (? error) + func (@"".eventState·2 *@"".eventMonitoringState) @"".enableEventMonitoring (@"".c·3 *@"".Client) (? error) + func (@"".eventState·2 *@"".eventMonitoringState) @"".isEnabled () (? bool) + func (@"".eventState·1 *@"".eventMonitoringState) @"".monitorEvents (@"".c·2 *@"".Client) + func (@"".eventState·2 *@"".eventMonitoringState) @"".noListeners () (? bool) + func (@"".eventState·2 *@"".eventMonitoringState) @"".removeListener (@"".listener·3 chan<- *@"".APIEvents "esc:0x0") (? error) + func (@"".eventState·1 *@"".eventMonitoringState) @"".sendEvent (@"".event·2 *@"".APIEvents) + func (@"".eventState·1 *@"".eventMonitoringState) @"".updateLastSeen (@"".e·2 *@"".APIEvents "esc:0x0") + type @"".doOptions struct { @"".data interface {}; @"".forceJSON bool } + type @"".streamOptions struct { @"".setRawTerminal bool; @"".rawJSONStream bool; @"".useJSONDecoder bool; @"".headers map[string]string; @"".in @"io".Reader; @"".stdout @"io".Writer; @"".stderr @"io".Writer; @"".timeout @"time".Duration } + type @"".hijackOptions struct { @"".success chan struct {}; @"".setRawTerminal bool; @"".in @"io".Reader; @"".stdout @"io".Writer; @"".stderr @"io".Writer; @"".data interface {} } + type @"".APIPort struct { PrivatePort int64 "json:\"PrivatePort,omitempty\" yaml:\"PrivatePort,omitempty\""; PublicPort int64 "json:\"PublicPort,omitempty\" yaml:\"PublicPort,omitempty\""; Type string "json:\"Type,omitempty\" yaml:\"Type,omitempty\""; IP string "json:\"IP,omitempty\" yaml:\"IP,omitempty\"" } + type @"".APIContainers struct { ID string "json:\"Id\" yaml:\"Id\""; Image string "json:\"Image,omitempty\" yaml:\"Image,omitempty\""; Command string "json:\"Command,omitempty\" yaml:\"Command,omitempty\""; Created int64 "json:\"Created,omitempty\" yaml:\"Created,omitempty\""; Status string "json:\"Status,omitempty\" yaml:\"Status,omitempty\""; Ports []@"".APIPort "json:\"Ports,omitempty\" yaml:\"Ports,omitempty\""; SizeRw int64 "json:\"SizeRw,omitempty\" yaml:\"SizeRw,omitempty\""; SizeRootFs int64 "json:\"SizeRootFs,omitempty\" yaml:\"SizeRootFs,omitempty\""; Names []string "json:\"Names,omitempty\" yaml:\"Names,omitempty\"" } + type @"".ListContainersOptions struct { All bool; Size bool; Limit int; Since string; Before string; Filters map[string][]string } + type @"".RenameContainerOptions struct { ID string "qs:\"-\""; Name string "json:\"name,omitempty\" yaml:\"name,omitempty\"" } + type @"".Port string + func (@"".p·2 @"".Port "esc:0x0") Port () (? string) + func (@"".p·2 @"".Port "esc:0x0") Proto () (? string) + type @"".Config struct { Hostname string "json:\"Hostname,omitempty\" yaml:\"Hostname,omitempty\""; Domainname string "json:\"Domainname,omitempty\" yaml:\"Domainname,omitempty\""; User string "json:\"User,omitempty\" yaml:\"User,omitempty\""; Memory int64 "json:\"Memory,omitempty\" yaml:\"Memory,omitempty\""; MemorySwap int64 "json:\"MemorySwap,omitempty\" yaml:\"MemorySwap,omitempty\""; CPUShares int64 "json:\"CpuShares,omitempty\" yaml:\"CpuShares,omitempty\""; CPUSet string "json:\"Cpuset,omitempty\" yaml:\"Cpuset,omitempty\""; AttachStdin bool "json:\"AttachStdin,omitempty\" yaml:\"AttachStdin,omitempty\""; AttachStdout bool "json:\"AttachStdout,omitempty\" yaml:\"AttachStdout,omitempty\""; AttachStderr bool "json:\"AttachStderr,omitempty\" yaml:\"AttachStderr,omitempty\""; PortSpecs []string "json:\"PortSpecs,omitempty\" yaml:\"PortSpecs,omitempty\""; ExposedPorts map[@"".Port]struct {} "json:\"ExposedPorts,omitempty\" yaml:\"ExposedPorts,omitempty\""; Tty bool "json:\"Tty,omitempty\" yaml:\"Tty,omitempty\""; OpenStdin bool "json:\"OpenStdin,omitempty\" yaml:\"OpenStdin,omitempty\""; StdinOnce bool "json:\"StdinOnce,omitempty\" yaml:\"StdinOnce,omitempty\""; Env []string "json:\"Env,omitempty\" yaml:\"Env,omitempty\""; Cmd []string "json:\"Cmd\" yaml:\"Cmd\""; DNS []string "json:\"Dns,omitempty\" yaml:\"Dns,omitempty\""; Image string "json:\"Image,omitempty\" yaml:\"Image,omitempty\""; Volumes map[string]struct {} "json:\"Volumes,omitempty\" yaml:\"Volumes,omitempty\""; VolumesFrom string "json:\"VolumesFrom,omitempty\" yaml:\"VolumesFrom,omitempty\""; WorkingDir string "json:\"WorkingDir,omitempty\" yaml:\"WorkingDir,omitempty\""; MacAddress string "json:\"MacAddress,omitempty\" yaml:\"MacAddress,omitempty\""; Entrypoint []string "json:\"Entrypoint\" yaml:\"Entrypoint\""; NetworkDisabled bool "json:\"NetworkDisabled,omitempty\" yaml:\"NetworkDisabled,omitempty\""; SecurityOpts []string "json:\"SecurityOpts,omitempty\" yaml:\"SecurityOpts,omitempty\""; OnBuild []string "json:\"OnBuild,omitempty\" yaml:\"OnBuild,omitempty\""; Labels map[string]string "json:\"Labels,omitempty\" yaml:\"Labels,omitempty\"" } + type @"".State struct { Running bool "json:\"Running,omitempty\" yaml:\"Running,omitempty\""; Paused bool "json:\"Paused,omitempty\" yaml:\"Paused,omitempty\""; Restarting bool "json:\"Restarting,omitempty\" yaml:\"Restarting,omitempty\""; OOMKilled bool "json:\"OOMKilled,omitempty\" yaml:\"OOMKilled,omitempty\""; Pid int "json:\"Pid,omitempty\" yaml:\"Pid,omitempty\""; ExitCode int "json:\"ExitCode,omitempty\" yaml:\"ExitCode,omitempty\""; Error string "json:\"Error,omitempty\" yaml:\"Error,omitempty\""; StartedAt @"time".Time "json:\"StartedAt,omitempty\" yaml:\"StartedAt,omitempty\""; FinishedAt @"time".Time "json:\"FinishedAt,omitempty\" yaml:\"FinishedAt,omitempty\"" } + func (@"".s·2 *@"".State "esc:0x0") String () (? string) + type @"".SwarmNode struct { ID string "json:\"ID,omitempty\" yaml:\"ID,omitempty\""; IP string "json:\"IP,omitempty\" yaml:\"IP,omitempty\""; Addr string "json:\"Addr,omitempty\" yaml:\"Addr,omitempty\""; Name string "json:\"Name,omitempty\" yaml:\"Name,omitempty\""; CPUs int64 "json:\"CPUs,omitempty\" yaml:\"CPUs,omitempty\""; Memory int64 "json:\"Memory,omitempty\" yaml:\"Memory,omitempty\""; Labels map[string]string "json:\"Labels,omitempty\" yaml:\"Labels,omitempty\"" } + type @"".PortMapping map[string]string + type @"".PortBinding struct { HostIP string "json:\"HostIP,omitempty\" yaml:\"HostIP,omitempty\""; HostPort string "json:\"HostPort,omitempty\" yaml:\"HostPort,omitempty\"" } + type @"".NetworkSettings struct { IPAddress string "json:\"IPAddress,omitempty\" yaml:\"IPAddress,omitempty\""; IPPrefixLen int "json:\"IPPrefixLen,omitempty\" yaml:\"IPPrefixLen,omitempty\""; MacAddress string "json:\"MacAddress,omitempty\" yaml:\"MacAddress,omitempty\""; Gateway string "json:\"Gateway,omitempty\" yaml:\"Gateway,omitempty\""; Bridge string "json:\"Bridge,omitempty\" yaml:\"Bridge,omitempty\""; PortMapping map[string]@"".PortMapping "json:\"PortMapping,omitempty\" yaml:\"PortMapping,omitempty\""; Ports map[@"".Port][]@"".PortBinding "json:\"Ports,omitempty\" yaml:\"Ports,omitempty\""; NetworkID string "json:\"NetworkID,omitempty\" yaml:\"NetworkID,omitempty\""; EndpointID string "json:\"EndpointID,omitempty\" yaml:\"EndpointID,omitempty\""; SandboxKey string "json:\"SandboxKey,omitempty\" yaml:\"SandboxKey,omitempty\""; GlobalIPv6Address string "json:\"GlobalIPv6Address,omitempty\" yaml:\"GlobalIPv6Address,omitempty\""; GlobalIPv6PrefixLen int "json:\"GlobalIPv6PrefixLen,omitempty\" yaml:\"GlobalIPv6PrefixLen,omitempty\""; IPv6Gateway string "json:\"IPv6Gateway,omitempty\" yaml:\"IPv6Gateway,omitempty\""; LinkLocalIPv6Address string "json:\"LinkLocalIPv6Address,omitempty\" yaml:\"LinkLocalIPv6Address,omitempty\""; LinkLocalIPv6PrefixLen int "json:\"LinkLocalIPv6PrefixLen,omitempty\" yaml:\"LinkLocalIPv6PrefixLen,omitempty\""; SecondaryIPAddresses []string "json:\"SecondaryIPAddresses,omitempty\" yaml:\"SecondaryIPAddresses,omitempty\""; SecondaryIPv6Addresses []string "json:\"SecondaryIPv6Addresses,omitempty\" yaml:\"SecondaryIPv6Addresses,omitempty\"" } + func (@"".settings·2 *@"".NetworkSettings "esc:0x0") PortMappingAPI () (? []@"".APIPort) + type @"".KeyValuePair struct { Key string "json:\"Key,omitempty\" yaml:\"Key,omitempty\""; Value string "json:\"Value,omitempty\" yaml:\"Value,omitempty\"" } + type @"".RestartPolicy struct { Name string "json:\"Name,omitempty\" yaml:\"Name,omitempty\""; MaximumRetryCount int "json:\"MaximumRetryCount,omitempty\" yaml:\"MaximumRetryCount,omitempty\"" } + type @"".Device struct { PathOnHost string "json:\"PathOnHost,omitempty\" yaml:\"PathOnHost,omitempty\""; PathInContainer string "json:\"PathInContainer,omitempty\" yaml:\"PathInContainer,omitempty\""; CgroupPermissions string "json:\"CgroupPermissions,omitempty\" yaml:\"CgroupPermissions,omitempty\"" } + type @"".LogConfig struct { Type string "json:\"Type,omitempty\" yaml:\"Type,omitempty\""; Config map[string]string "json:\"Config,omitempty\" yaml:\"Config,omitempty\"" } + type @"".ULimit struct { Name string "json:\"Name,omitempty\" yaml:\"Name,omitempty\""; Soft int64 "json:\"Soft,omitempty\" yaml:\"Soft,omitempty\""; Hard int64 "json:\"Hard,omitempty\" yaml:\"Hard,omitempty\"" } + type @"".HostConfig struct { Binds []string "json:\"Binds,omitempty\" yaml:\"Binds,omitempty\""; CapAdd []string "json:\"CapAdd,omitempty\" yaml:\"CapAdd,omitempty\""; CapDrop []string "json:\"CapDrop,omitempty\" yaml:\"CapDrop,omitempty\""; ContainerIDFile string "json:\"ContainerIDFile,omitempty\" yaml:\"ContainerIDFile,omitempty\""; LxcConf []@"".KeyValuePair "json:\"LxcConf,omitempty\" yaml:\"LxcConf,omitempty\""; Privileged bool "json:\"Privileged,omitempty\" yaml:\"Privileged,omitempty\""; PortBindings map[@"".Port][]@"".PortBinding "json:\"PortBindings,omitempty\" yaml:\"PortBindings,omitempty\""; Links []string "json:\"Links,omitempty\" yaml:\"Links,omitempty\""; PublishAllPorts bool "json:\"PublishAllPorts,omitempty\" yaml:\"PublishAllPorts,omitempty\""; DNS []string "json:\"Dns,omitempty\" yaml:\"Dns,omitempty\""; DNSSearch []string "json:\"DnsSearch,omitempty\" yaml:\"DnsSearch,omitempty\""; ExtraHosts []string "json:\"ExtraHosts,omitempty\" yaml:\"ExtraHosts,omitempty\""; VolumesFrom []string "json:\"VolumesFrom,omitempty\" yaml:\"VolumesFrom,omitempty\""; NetworkMode string "json:\"NetworkMode,omitempty\" yaml:\"NetworkMode,omitempty\""; IpcMode string "json:\"IpcMode,omitempty\" yaml:\"IpcMode,omitempty\""; PidMode string "json:\"PidMode,omitempty\" yaml:\"PidMode,omitempty\""; UTSMode string "json:\"UTSMode,omitempty\" yaml:\"UTSMode,omitempty\""; RestartPolicy @"".RestartPolicy "json:\"RestartPolicy,omitempty\" yaml:\"RestartPolicy,omitempty\""; Devices []@"".Device "json:\"Devices,omitempty\" yaml:\"Devices,omitempty\""; LogConfig @"".LogConfig "json:\"LogConfig,omitempty\" yaml:\"LogConfig,omitempty\""; ReadonlyRootfs bool "json:\"ReadonlyRootfs,omitempty\" yaml:\"ReadonlyRootfs,omitempty\""; SecurityOpt []string "json:\"SecurityOpt,omitempty\" yaml:\"SecurityOpt,omitempty\""; CgroupParent string "json:\"CgroupParent,omitempty\" yaml:\"CgroupParent,omitempty\""; Memory int64 "json:\"Memory,omitempty\" yaml:\"Memory,omitempty\""; MemorySwap int64 "json:\"MemorySwap,omitempty\" yaml:\"MemorySwap,omitempty\""; CPUShares int64 "json:\"CpuShares,omitempty\" yaml:\"CpuShares,omitempty\""; CPUSet string "json:\"Cpuset,omitempty\" yaml:\"Cpuset,omitempty\""; CPUQuota int64 "json:\"CpuQuota,omitempty\" yaml:\"CpuQuota,omitempty\""; CPUPeriod int64 "json:\"CpuPeriod,omitempty\" yaml:\"CpuPeriod,omitempty\""; Ulimits []@"".ULimit "json:\"Ulimits,omitempty\" yaml:\"Ulimits,omitempty\"" } + type @"".Container struct { ID string "json:\"Id\" yaml:\"Id\""; Created @"time".Time "json:\"Created,omitempty\" yaml:\"Created,omitempty\""; Path string "json:\"Path,omitempty\" yaml:\"Path,omitempty\""; Args []string "json:\"Args,omitempty\" yaml:\"Args,omitempty\""; Config *@"".Config "json:\"Config,omitempty\" yaml:\"Config,omitempty\""; State @"".State "json:\"State,omitempty\" yaml:\"State,omitempty\""; Image string "json:\"Image,omitempty\" yaml:\"Image,omitempty\""; Node *@"".SwarmNode "json:\"Node,omitempty\" yaml:\"Node,omitempty\""; NetworkSettings *@"".NetworkSettings "json:\"NetworkSettings,omitempty\" yaml:\"NetworkSettings,omitempty\""; SysInitPath string "json:\"SysInitPath,omitempty\" yaml:\"SysInitPath,omitempty\""; ResolvConfPath string "json:\"ResolvConfPath,omitempty\" yaml:\"ResolvConfPath,omitempty\""; HostnamePath string "json:\"HostnamePath,omitempty\" yaml:\"HostnamePath,omitempty\""; HostsPath string "json:\"HostsPath,omitempty\" yaml:\"HostsPath,omitempty\""; LogPath string "json:\"LogPath,omitempty\" yaml:\"LogPath,omitempty\""; Name string "json:\"Name,omitempty\" yaml:\"Name,omitempty\""; Driver string "json:\"Driver,omitempty\" yaml:\"Driver,omitempty\""; Volumes map[string]string "json:\"Volumes,omitempty\" yaml:\"Volumes,omitempty\""; VolumesRW map[string]bool "json:\"VolumesRW,omitempty\" yaml:\"VolumesRW,omitempty\""; HostConfig *@"".HostConfig "json:\"HostConfig,omitempty\" yaml:\"HostConfig,omitempty\""; ExecIDs []string "json:\"ExecIDs,omitempty\" yaml:\"ExecIDs,omitempty\""; RestartCount int "json:\"RestartCount,omitempty\" yaml:\"RestartCount,omitempty\""; AppArmorProfile string "json:\"AppArmorProfile,omitempty\" yaml:\"AppArmorProfile,omitempty\"" } + type @"".CreateContainerOptions struct { Name string; Config *@"".Config "qs:\"-\""; HostConfig *@"".HostConfig "qs:\"-\"" } + type @"".TopResult struct { Titles []string; Processes [][]string } + type @"".BlkioStatsEntry struct { Major uint64 "json:\"major,omitempty\" yaml:\"major,omitempty\""; Minor uint64 "json:\"minor,omitempty\" yaml:\"minor,omitempty\""; Op string "json:\"op,omitempty\" yaml:\"op,omitempty\""; Value uint64 "json:\"value,omitempty\" yaml:\"value,omitempty\"" } + type @"".CPUStats struct { CPUUsage struct { PercpuUsage []uint64 "json:\"percpu_usage,omitempty\" yaml:\"percpu_usage,omitempty\""; UsageInUsermode uint64 "json:\"usage_in_usermode,omitempty\" yaml:\"usage_in_usermode,omitempty\""; TotalUsage uint64 "json:\"total_usage,omitempty\" yaml:\"total_usage,omitempty\""; UsageInKernelmode uint64 "json:\"usage_in_kernelmode,omitempty\" yaml:\"usage_in_kernelmode,omitempty\"" } "json:\"cpu_usage,omitempty\" yaml:\"cpu_usage,omitempty\""; SystemCPUUsage uint64 "json:\"system_cpu_usage,omitempty\" yaml:\"system_cpu_usage,omitempty\""; ThrottlingData struct { Periods uint64 "json:\"periods,omitempty\""; ThrottledPeriods uint64 "json:\"throttled_periods,omitempty\""; ThrottledTime uint64 "json:\"throttled_time,omitempty\"" } "json:\"throttling_data,omitempty\" yaml:\"throttling_data,omitempty\"" } + type @"".Stats struct { Read @"time".Time "json:\"read,omitempty\" yaml:\"read,omitempty\""; Network struct { RxDropped uint64 "json:\"rx_dropped,omitempty\" yaml:\"rx_dropped,omitempty\""; RxBytes uint64 "json:\"rx_bytes,omitempty\" yaml:\"rx_bytes,omitempty\""; RxErrors uint64 "json:\"rx_errors,omitempty\" yaml:\"rx_errors,omitempty\""; TxPackets uint64 "json:\"tx_packets,omitempty\" yaml:\"tx_packets,omitempty\""; TxDropped uint64 "json:\"tx_dropped,omitempty\" yaml:\"tx_dropped,omitempty\""; RxPackets uint64 "json:\"rx_packets,omitempty\" yaml:\"rx_packets,omitempty\""; TxErrors uint64 "json:\"tx_errors,omitempty\" yaml:\"tx_errors,omitempty\""; TxBytes uint64 "json:\"tx_bytes,omitempty\" yaml:\"tx_bytes,omitempty\"" } "json:\"network,omitempty\" yaml:\"network,omitempty\""; MemoryStats struct { Stats struct { TotalPgmafault uint64 "json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""; Cache uint64 "json:\"cache,omitempty\" yaml:\"cache,omitempty\""; MappedFile uint64 "json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""; TotalInactiveFile uint64 "json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""; Pgpgout uint64 "json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""; Rss uint64 "json:\"rss,omitempty\" yaml:\"rss,omitempty\""; TotalMappedFile uint64 "json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""; Writeback uint64 "json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""; Unevictable uint64 "json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""; Pgpgin uint64 "json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""; TotalUnevictable uint64 "json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""; Pgmajfault uint64 "json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""; TotalRss uint64 "json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""; TotalRssHuge uint64 "json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""; TotalWriteback uint64 "json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""; TotalInactiveAnon uint64 "json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""; RssHuge uint64 "json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""; HierarchicalMemoryLimit uint64 "json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""; TotalPgfault uint64 "json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""; TotalActiveFile uint64 "json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""; ActiveAnon uint64 "json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""; TotalActiveAnon uint64 "json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""; TotalPgpgout uint64 "json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""; TotalCache uint64 "json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""; InactiveAnon uint64 "json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""; ActiveFile uint64 "json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""; Pgfault uint64 "json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""; InactiveFile uint64 "json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""; TotalPgpgin uint64 "json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\"" } "json:\"stats,omitempty\" yaml:\"stats,omitempty\""; MaxUsage uint64 "json:\"max_usage,omitempty\" yaml:\"max_usage,omitempty\""; Usage uint64 "json:\"usage,omitempty\" yaml:\"usage,omitempty\""; Failcnt uint64 "json:\"failcnt,omitempty\" yaml:\"failcnt,omitempty\""; Limit uint64 "json:\"limit,omitempty\" yaml:\"limit,omitempty\"" } "json:\"memory_stats,omitempty\" yaml:\"memory_stats,omitempty\""; BlkioStats struct { IOServiceBytesRecursive []@"".BlkioStatsEntry "json:\"io_service_bytes_recursive,omitempty\" yaml:\"io_service_bytes_recursive,omitempty\""; IOServicedRecursive []@"".BlkioStatsEntry "json:\"io_serviced_recursive,omitempty\" yaml:\"io_serviced_recursive,omitempty\""; IOQueueRecursive []@"".BlkioStatsEntry "json:\"io_queue_recursive,omitempty\" yaml:\"io_queue_recursive,omitempty\""; IOServiceTimeRecursive []@"".BlkioStatsEntry "json:\"io_service_time_recursive,omitempty\" yaml:\"io_service_time_recursive,omitempty\""; IOWaitTimeRecursive []@"".BlkioStatsEntry "json:\"io_wait_time_recursive,omitempty\" yaml:\"io_wait_time_recursive,omitempty\""; IOMergedRecursive []@"".BlkioStatsEntry "json:\"io_merged_recursive,omitempty\" yaml:\"io_merged_recursive,omitempty\""; IOTimeRecursive []@"".BlkioStatsEntry "json:\"io_time_recursive,omitempty\" yaml:\"io_time_recursive,omitempty\""; SectorsRecursive []@"".BlkioStatsEntry "json:\"sectors_recursive,omitempty\" yaml:\"sectors_recursive,omitempty\"" } "json:\"blkio_stats,omitempty\" yaml:\"blkio_stats,omitempty\""; CPUStats @"".CPUStats "json:\"cpu_stats,omitempty\" yaml:\"cpu_stats,omitempty\""; PreCPUStats @"".CPUStats "json:\"precpu_stats,omitempty\"" } + type @"".StatsOptions struct { ID string; Stats chan<- *@"".Stats; Stream bool; Done <-chan bool; Timeout @"time".Duration } + type @"".Signal int + type @"".KillContainerOptions struct { ID string "qs:\"-\""; Signal @"".Signal } + type @"".RemoveContainerOptions struct { ID string "qs:\"-\""; RemoveVolumes bool "qs:\"v\""; Force bool } + type @"".CopyFromContainerOptions struct { OutputStream @"io".Writer "json:\"-\""; Container string "json:\"-\""; Resource string } + type @"".Image struct { ID string "json:\"Id\" yaml:\"Id\""; Parent string "json:\"Parent,omitempty\" yaml:\"Parent,omitempty\""; Comment string "json:\"Comment,omitempty\" yaml:\"Comment,omitempty\""; Created @"time".Time "json:\"Created,omitempty\" yaml:\"Created,omitempty\""; Container string "json:\"Container,omitempty\" yaml:\"Container,omitempty\""; ContainerConfig @"".Config "json:\"ContainerConfig,omitempty\" yaml:\"ContainerConfig,omitempty\""; DockerVersion string "json:\"DockerVersion,omitempty\" yaml:\"DockerVersion,omitempty\""; Author string "json:\"Author,omitempty\" yaml:\"Author,omitempty\""; Config *@"".Config "json:\"Config,omitempty\" yaml:\"Config,omitempty\""; Architecture string "json:\"Architecture,omitempty\" yaml:\"Architecture,omitempty\""; Size int64 "json:\"Size,omitempty\" yaml:\"Size,omitempty\""; VirtualSize int64 "json:\"VirtualSize,omitempty\" yaml:\"VirtualSize,omitempty\"" } + type @"".CommitContainerOptions struct { Container string; Repository string "qs:\"repo\""; Tag string; Message string "qs:\"m\""; Author string; Run *@"".Config "qs:\"-\"" } + type @"".AttachToContainerOptions struct { Container string "qs:\"-\""; InputStream @"io".Reader "qs:\"-\""; OutputStream @"io".Writer "qs:\"-\""; ErrorStream @"io".Writer "qs:\"-\""; Logs bool; Stream bool; Stdin bool; Stdout bool; Stderr bool; Success chan struct {}; RawTerminal bool "qs:\"-\"" } + type @"".LogsOptions struct { Container string "qs:\"-\""; OutputStream @"io".Writer "qs:\"-\""; ErrorStream @"io".Writer "qs:\"-\""; Follow bool; Stdout bool; Stderr bool; Since int64; Timestamps bool; Tail string; RawTerminal bool "qs:\"-\"" } + type @"".ExportContainerOptions struct { ID string; OutputStream @"io".Writer } + type @"".Exec struct { ID string "json:\"Id,omitempty\" yaml:\"Id,omitempty\"" } + type @"".CreateExecOptions struct { AttachStdin bool "json:\"AttachStdin,omitempty\" yaml:\"AttachStdin,omitempty\""; AttachStdout bool "json:\"AttachStdout,omitempty\" yaml:\"AttachStdout,omitempty\""; AttachStderr bool "json:\"AttachStderr,omitempty\" yaml:\"AttachStderr,omitempty\""; Tty bool "json:\"Tty,omitempty\" yaml:\"Tty,omitempty\""; Cmd []string "json:\"Cmd,omitempty\" yaml:\"Cmd,omitempty\""; Container string "json:\"Container,omitempty\" yaml:\"Container,omitempty\""; User string "json:\"User,omitempty\" yaml:\"User,omitempty\"" } + type @"".StartExecOptions struct { Detach bool "json:\"Detach,omitempty\" yaml:\"Detach,omitempty\""; Tty bool "json:\"Tty,omitempty\" yaml:\"Tty,omitempty\""; InputStream @"io".Reader "qs:\"-\""; OutputStream @"io".Writer "qs:\"-\""; ErrorStream @"io".Writer "qs:\"-\""; RawTerminal bool "qs:\"-\""; Success chan struct {} "json:\"-\"" } + type @"".ExecProcessConfig struct { Privileged bool "json:\"privileged,omitempty\" yaml:\"privileged,omitempty\""; User string "json:\"user,omitempty\" yaml:\"user,omitempty\""; Tty bool "json:\"tty,omitempty\" yaml:\"tty,omitempty\""; EntryPoint string "json:\"entrypoint,omitempty\" yaml:\"entrypoint,omitempty\""; Arguments []string "json:\"arguments,omitempty\" yaml:\"arguments,omitempty\"" } + type @"".ExecInspect struct { ID string "json:\"ID,omitempty\" yaml:\"ID,omitempty\""; Running bool "json:\"Running,omitempty\" yaml:\"Running,omitempty\""; ExitCode int "json:\"ExitCode,omitempty\" yaml:\"ExitCode,omitempty\""; OpenStdin bool "json:\"OpenStdin,omitempty\" yaml:\"OpenStdin,omitempty\""; OpenStderr bool "json:\"OpenStderr,omitempty\" yaml:\"OpenStderr,omitempty\""; OpenStdout bool "json:\"OpenStdout,omitempty\" yaml:\"OpenStdout,omitempty\""; ProcessConfig @"".ExecProcessConfig "json:\"ProcessConfig,omitempty\" yaml:\"ProcessConfig,omitempty\""; Container @"".Container "json:\"Container,omitempty\" yaml:\"Container,omitempty\"" } + type @"".APIImages struct { ID string "json:\"Id\" yaml:\"Id\""; RepoTags []string "json:\"RepoTags,omitempty\" yaml:\"RepoTags,omitempty\""; Created int64 "json:\"Created,omitempty\" yaml:\"Created,omitempty\""; Size int64 "json:\"Size,omitempty\" yaml:\"Size,omitempty\""; VirtualSize int64 "json:\"VirtualSize,omitempty\" yaml:\"VirtualSize,omitempty\""; ParentID string "json:\"ParentId,omitempty\" yaml:\"ParentId,omitempty\""; RepoDigests []string "json:\"RepoDigests,omitempty\" yaml:\"RepoDigests,omitempty\""; Labels map[string]string "json:\"Labels,omitempty\" yaml:\"Labels,omitempty\"" } + type @"".ListImagesOptions struct { All bool; Filters map[string][]string; Digests bool } + type @"".ImageHistory struct { ID string "json:\"Id\" yaml:\"Id\""; Tags []string "json:\"Tags,omitempty\" yaml:\"Tags,omitempty\""; Created int64 "json:\"Created,omitempty\" yaml:\"Created,omitempty\""; CreatedBy string "json:\"CreatedBy,omitempty\" yaml:\"CreatedBy,omitempty\""; Size int64 "json:\"Size,omitempty\" yaml:\"Size,omitempty\"" } + type @"".RemoveImageOptions struct { Force bool "qs:\"force\""; NoPrune bool "qs:\"noprune\"" } + type @"".PushImageOptions struct { Name string; Tag string; Registry string; OutputStream @"io".Writer "qs:\"-\""; RawJSONStream bool "qs:\"-\"" } + type @"".PullImageOptions struct { Repository string "qs:\"fromImage\""; Registry string; Tag string; OutputStream @"io".Writer "qs:\"-\""; RawJSONStream bool "qs:\"-\"" } + type @"".LoadImageOptions struct { InputStream @"io".Reader } + type @"".ExportImageOptions struct { Name string; OutputStream @"io".Writer } + type @"".ExportImagesOptions struct { Names []string; OutputStream @"io".Writer "qs:\"-\"" } + type @"".ImportImageOptions struct { Repository string "qs:\"repo\""; Source string "qs:\"fromSrc\""; Tag string "qs:\"tag\""; InputStream @"io".Reader "qs:\"-\""; OutputStream @"io".Writer "qs:\"-\""; RawJSONStream bool "qs:\"-\"" } + type @"".BuildImageOptions struct { Name string "qs:\"t\""; Dockerfile string "qs:\"dockerfile\""; NoCache bool "qs:\"nocache\""; SuppressOutput bool "qs:\"q\""; Pull bool "qs:\"pull\""; RmTmpContainer bool "qs:\"rm\""; ForceRmTmpContainer bool "qs:\"forcerm\""; Memory int64 "qs:\"memory\""; Memswap int64 "qs:\"memswap\""; CPUShares int64 "qs:\"cpushares\""; CPUSetCPUs string "qs:\"cpusetcpus\""; InputStream @"io".Reader "qs:\"-\""; OutputStream @"io".Writer "qs:\"-\""; RawJSONStream bool "qs:\"-\""; Remote string "qs:\"remote\""; Auth @"".AuthConfiguration "qs:\"-\""; AuthConfigs @"".AuthConfigurations "qs:\"-\""; ContextDir string "qs:\"-\"" } + type @"".TagImageOptions struct { Repo string; Tag string; Force bool } + type @"".APIImageSearch struct { Description string "json:\"description,omitempty\" yaml:\"description,omitempty\""; IsOfficial bool "json:\"is_official,omitempty\" yaml:\"is_official,omitempty\""; IsAutomated bool "json:\"is_automated,omitempty\" yaml:\"is_automated,omitempty\""; Name string "json:\"name,omitempty\" yaml:\"name,omitempty\""; StarCount int "json:\"star_count,omitempty\" yaml:\"star_count,omitempty\"" } + type @"".Env []string + func (@"".env·2 *@"".Env) Decode (@"".src·3 @"io".Reader) (? error) + func (@"".env·2 *@"".Env "esc:0x0") Exists (@"".key·3 string "esc:0x0") (? bool) + func (@"".env·2 *@"".Env "esc:0x0") Get (@"".key·3 string "esc:0x0") (@"".value·1 string) + func (@"".env·2 *@"".Env "esc:0x0") GetBool (@"".key·3 string "esc:0x0") (@"".value·1 bool) + func (@"".env·2 *@"".Env "esc:0x0") GetInt (@"".key·3 string "esc:0x0") (? int) + func (@"".env·2 *@"".Env "esc:0x0") GetInt64 (@"".key·3 string "esc:0x0") (? int64) + func (@"".env·2 *@"".Env "esc:0x0") GetJSON (@"".key·3 string "esc:0x0", @"".iface·4 interface {}) (? error) + func (@"".env·2 *@"".Env "esc:0x0") GetList (@"".key·3 string "esc:0x0") (? []string) + func (@"".env·2 *@"".Env "esc:0x0") Map () (? map[string]string) + func (@"".env·1 *@"".Env) Set (@"".key·2 string "esc:0x0", @"".value·3 string "esc:0x0") { *@"".env·1 = append(*@"".env·1, @"".key·2 + "=" + @"".value·3) } + func (@"".env·1 *@"".Env) SetAuto (@"".key·2 string "esc:0x0", @"".value·3 interface {}) + func (@"".env·1 *@"".Env) SetBool (@"".key·2 string "esc:0x0", @"".value·3 bool) + func (@"".env·1 *@"".Env) SetInt (@"".key·2 string "esc:0x0", @"".value·3 int) + func (@"".env·1 *@"".Env) SetInt64 (@"".key·2 string "esc:0x0", @"".value·3 int64) + func (@"".env·2 *@"".Env) SetJSON (@"".key·3 string "esc:0x0", @"".value·4 interface {}) (? error) + func (@"".env·2 *@"".Env) SetList (@"".key·3 string "esc:0x0", @"".value·4 []string) (? error) + type @"".Endpoint struct { Name string "json:\"name\""; ID string "json:\"id\""; Network string "json:\"network\"" } + type @"".Network struct { Name string "json:\"name\""; ID string "json:\"id\""; Type string "json:\"type\""; Endpoints []*@"".Endpoint "json:\"endpoints\"" } + type @"".CreateNetworkOptions struct { Name string "json:\"name\""; NetworkType string "json:\"network_type\""; Options map[string]interface {} "json:\"options\"" } + type @"".Client struct { SkipServerVersionCheck bool; HTTPClient *@"net/http".Client; TLSConfig *@"crypto/tls".Config; @"".endpoint string; @"".endpointURL *@"net/url".URL; @"".eventMonitor *@"".eventMonitoringState; @"".requestedAPIVersion @"".APIVersion; @"".serverAPIVersion @"".APIVersion; @"".expectedAPIVersion @"".APIVersion } + func (@"".c·2 *@"".Client) AddEventListener (@"".listener·3 chan<- *@"".APIEvents) (? error) + func (@"".c·2 *@"".Client) AttachToContainer (@"".opts·3 @"".AttachToContainerOptions) (? error) + func (@"".c·2 *@"".Client) AuthCheck (@"".conf·3 *@"".AuthConfiguration) (? error) + func (@"".c·2 *@"".Client) BuildImage (@"".opts·3 @"".BuildImageOptions) (? error) + func (@"".c·3 *@"".Client) CommitContainer (@"".opts·4 @"".CommitContainerOptions) (? *@"".Image, ? error) + func (@"".c·3 *@"".Client) ContainerChanges (@"".id·4 string) (? []@"".Change, ? error) + func (@"".c·2 *@"".Client) CopyFromContainer (@"".opts·3 @"".CopyFromContainerOptions) (? error) + func (@"".c·3 *@"".Client) CreateContainer (@"".opts·4 @"".CreateContainerOptions) (? *@"".Container, ? error) + func (@"".c·3 *@"".Client) CreateExec (@"".opts·4 @"".CreateExecOptions) (? *@"".Exec, ? error) + func (@"".c·3 *@"".Client) CreateNetwork (@"".opts·4 @"".CreateNetworkOptions) (? *@"".Network, ? error) + func (@"".c·2 *@"".Client) ExportContainer (@"".opts·3 @"".ExportContainerOptions) (? error) + func (@"".c·2 *@"".Client) ExportImage (@"".opts·3 @"".ExportImageOptions) (? error) + func (@"".c·2 *@"".Client) ExportImages (@"".opts·3 @"".ExportImagesOptions) (? error) + func (@"".c·3 *@"".Client) ImageHistory (@"".name·4 string "esc:0x0") (? []@"".ImageHistory, ? error) + func (@"".c·2 *@"".Client) ImportImage (@"".opts·3 @"".ImportImageOptions) (? error) + func (@"".c·3 *@"".Client) Info () (? *@"".Env, ? error) + func (@"".c·3 *@"".Client) InspectContainer (@"".id·4 string) (? *@"".Container, ? error) + func (@"".c·3 *@"".Client) InspectExec (@"".id·4 string) (? *@"".ExecInspect, ? error) + func (@"".c·3 *@"".Client) InspectImage (@"".name·4 string "esc:0x0") (? *@"".Image, ? error) + func (@"".c·2 *@"".Client) KillContainer (@"".opts·3 @"".KillContainerOptions) (? error) + func (@"".c·3 *@"".Client) ListContainers (@"".opts·4 @"".ListContainersOptions) (? []@"".APIContainers, ? error) + func (@"".c·3 *@"".Client) ListImages (@"".opts·4 @"".ListImagesOptions) (? []@"".APIImages, ? error) + func (@"".c·3 *@"".Client) ListNetworks () (? []@"".Network, ? error) + func (@"".c·2 *@"".Client) LoadImage (@"".opts·3 @"".LoadImageOptions) (? error) + func (@"".c·2 *@"".Client) Logs (@"".opts·3 @"".LogsOptions) (? error) + func (@"".c·3 *@"".Client) NetworkInfo (@"".id·4 string) (? *@"".Network, ? error) + func (@"".c·2 *@"".Client) PauseContainer (@"".id·3 string) (? error) + func (@"".c·2 *@"".Client) Ping () (? error) + func (@"".c·2 *@"".Client) PullImage (@"".opts·3 @"".PullImageOptions, @"".auth·4 @"".AuthConfiguration) (? error) + func (@"".c·2 *@"".Client) PushImage (@"".opts·3 @"".PushImageOptions, @"".auth·4 @"".AuthConfiguration) (? error) + func (@"".c·2 *@"".Client) RemoveContainer (@"".opts·3 @"".RemoveContainerOptions) (? error) + func (@"".c·2 *@"".Client) RemoveEventListener (@"".listener·3 chan *@"".APIEvents "esc:0x0") (? error) + func (@"".c·2 *@"".Client) RemoveImage (@"".name·3 string "esc:0x0") (? error) + func (@"".c·2 *@"".Client) RemoveImageExtended (@"".name·3 string, @"".opts·4 @"".RemoveImageOptions) (? error) + func (@"".c·2 *@"".Client) RenameContainer (@"".opts·3 @"".RenameContainerOptions) (? error) + func (@"".c·2 *@"".Client) ResizeContainerTTY (@"".id·3 string "esc:0x0", @"".height·4 int, @"".width·5 int) (? error) + func (@"".c·2 *@"".Client) ResizeExecTTY (@"".id·3 string, @"".height·4 int, @"".width·5 int) (? error) + func (@"".c·2 *@"".Client) RestartContainer (@"".id·3 string, @"".timeout·4 uint) (? error) + func (@"".c·3 *@"".Client) SearchImages (@"".term·4 string "esc:0x0") (? []@"".APIImageSearch, ? error) + func (@"".c·2 *@"".Client) StartContainer (@"".id·3 string, @"".hostConfig·4 *@"".HostConfig) (? error) + func (@"".c·2 *@"".Client) StartExec (@"".id·3 string, @"".opts·4 @"".StartExecOptions) (? error) + func (@"".c·2 *@"".Client) Stats (@"".opts·3 @"".StatsOptions) (@"".retErr·1 error) + func (@"".c·2 *@"".Client) StopContainer (@"".id·3 string, @"".timeout·4 uint) (? error) + func (@"".c·2 *@"".Client) TagImage (@"".name·3 string "esc:0x0", @"".opts·4 @"".TagImageOptions) (? error) + func (@"".c·3 *@"".Client) TopContainer (@"".id·4 string, @"".psArgs·5 string) (? @"".TopResult, ? error) + func (@"".c·2 *@"".Client) UnpauseContainer (@"".id·3 string) (? error) + func (@"".c·3 *@"".Client) Version () (? *@"".Env, ? error) + func (@"".c·3 *@"".Client) WaitContainer (@"".id·4 string) (? int, ? error) + func (@"".c·2 *@"".Client) @"".checkAPIVersion () (? error) + func (@"".c·2 *@"".Client) @"".createImage (@"".qs·3 string "esc:0x0", @"".headers·4 map[string]string, @"".in·5 @"io".Reader, @"".w·6 @"io".Writer, @"".rawJSONStream·7 bool) (? error) + func (@"".c·4 *@"".Client) @"".do (@"".method·5 string, @"".path·6 string, @"".doOptions·7 @"".doOptions) (? []byte, ? int, ? error) + func (@"".c·2 *@"".Client) @"".eventHijack (@"".startTime·3 int64, @"".eventChan·4 chan *@"".APIEvents, @"".errChan·5 chan error) (? error) + func (@"".c·3 *@"".Client) @"".getServerAPIVersionString () (@"".version·1 string, @"".err·2 error) + func (@"".c·2 *@"".Client) @"".getURL (@"".path·3 string) (? string) + func (@"".c·2 *@"".Client) @"".hijack (@"".method·3 string, @"".path·4 string, @"".hijackOptions·5 @"".hijackOptions) (? error) + func (@"".c·2 *@"".Client) @"".stream (@"".method·3 string, @"".path·4 string, @"".streamOptions·5 @"".streamOptions) (? error) + func @"".NewClient (@"".endpoint·3 string) (? *@"".Client, ? error) + func @"".NewTLSClient (@"".endpoint·3 string, @"".cert·4 string, @"".key·5 string, @"".ca·6 string) (? *@"".Client, ? error) + func @"".NewTLSClientFromBytes (@"".endpoint·3 string, @"".certPEMBlock·4 []byte, @"".keyPEMBlock·5 []byte, @"".caPEMCert·6 []byte) (? *@"".Client, ? error) + func @"".NewVersionedClient (@"".endpoint·3 string, @"".apiVersionString·4 string) (? *@"".Client, ? error) + func @"".NewVersionnedTLSClient (@"".endpoint·3 string, @"".cert·4 string, @"".key·5 string, @"".ca·6 string, @"".apiVersionString·7 string) (? *@"".Client, ? error) + func @"".NewVersionedTLSClient (@"".endpoint·3 string, @"".cert·4 string, @"".key·5 string, @"".ca·6 string, @"".apiVersionString·7 string) (? *@"".Client, ? error) + func @"".NewClientFromEnv () (? *@"".Client, ? error) + func @"".NewVersionedClientFromEnv (@"".apiVersionString·3 string) (? *@"".Client, ? error) + func @"".NewVersionedTLSClientFromBytes (@"".endpoint·3 string, @"".certPEMBlock·4 []byte, @"".keyPEMBlock·5 []byte, @"".caPEMCert·6 []byte, @"".apiVersionString·7 string) (? *@"".Client, ? error) + type @"".Error struct { Status int; Message string } + func (@"".e·2 *@"".Error) Error () (? string) + var @"".ErrContainerAlreadyExists error + func @"".AlwaysRestart () (? @"".RestartPolicy) { return (@"".RestartPolicy{ Name:"always" }) } + func @"".RestartOnFailure (@"".maxRetry·2 int) (? @"".RestartPolicy) { return (@"".RestartPolicy{ Name:"on-failure", MaximumRetryCount:@"".maxRetry·2 }) } + func @"".NeverRestart () (? @"".RestartPolicy) { return (@"".RestartPolicy{ Name:"no" }) } + type @"".NoSuchContainer struct { ID string; Err error } + func (@"".err·2 *@"".NoSuchContainer) Error () (? string) + type @"".ContainerAlreadyRunning struct { ID string } + func (@"".err·2 *@"".ContainerAlreadyRunning "esc:0x0") Error () (? string) { return "Container already running: " + @"".err·2.ID } + type @"".ContainerNotRunning struct { ID string } + func (@"".err·2 *@"".ContainerNotRunning "esc:0x0") Error () (? string) { return "Container not running: " + @"".err·2.ID } + var @"".ErrNoListeners error + var @"".ErrListenerAlreadyExists error + var @"".EOFEvent *@"".APIEvents + type @"".NoSuchExec struct { ID string } + func (@"".err·2 *@"".NoSuchExec "esc:0x0") Error () (? string) { return "No such exec instance: " + @"".err·2.ID } + type @"".ImagePre012 struct { ID string "json:\"id\""; Parent string "json:\"parent,omitempty\""; Comment string "json:\"comment,omitempty\""; Created @"time".Time "json:\"created\""; Container string "json:\"container,omitempty\""; ContainerConfig @"".Config "json:\"container_config,omitempty\""; DockerVersion string "json:\"docker_version,omitempty\""; Author string "json:\"author,omitempty\""; Config *@"".Config "json:\"config,omitempty\""; Architecture string "json:\"architecture,omitempty\""; Size int64 "json:\"size,omitempty\"" } + var @"".ErrNoSuchImage error + var @"".ErrMissingRepo error + var @"".ErrMissingOutputStream error + var @"".ErrMultipleContexts error + var @"".ErrMustSpecifyNames error + func @"".ParseRepositoryTag (@"".repoTag·3 string "esc:0x2") (@"".repository·1 string, @"".tag·2 string) + var @"".ErrNetworkAlreadyExists error + type @"".NoSuchNetwork struct { ID string } + func (@"".err·2 *@"".NoSuchNetwork) Error () (? string) + const @"".SIGABRT @"".Signal = 0x6 + const @"".SIGALRM @"".Signal = 0xE + const @"".SIGBUS @"".Signal = 0x7 + const @"".SIGCHLD @"".Signal = 0x11 + const @"".SIGCLD @"".Signal = 0x11 + const @"".SIGCONT @"".Signal = 0x12 + const @"".SIGFPE @"".Signal = 0x8 + const @"".SIGHUP @"".Signal = 0x1 + const @"".SIGILL @"".Signal = 0x4 + const @"".SIGINT @"".Signal = 0x2 + const @"".SIGIO @"".Signal = 0x1D + const @"".SIGIOT @"".Signal = 0x6 + const @"".SIGKILL @"".Signal = 0x9 + const @"".SIGPIPE @"".Signal = 0xD + const @"".SIGPOLL @"".Signal = 0x1D + const @"".SIGPROF @"".Signal = 0x1B + const @"".SIGPWR @"".Signal = 0x1E + const @"".SIGQUIT @"".Signal = 0x3 + const @"".SIGSEGV @"".Signal = 0xB + const @"".SIGSTKFLT @"".Signal = 0x10 + const @"".SIGSTOP @"".Signal = 0x13 + const @"".SIGSYS @"".Signal = 0x1F + const @"".SIGTERM @"".Signal = 0xF + const @"".SIGTRAP @"".Signal = 0x5 + const @"".SIGTSTP @"".Signal = 0x14 + const @"".SIGTTIN @"".Signal = 0x15 + const @"".SIGTTOU @"".Signal = 0x16 + const @"".SIGUNUSED @"".Signal = 0x1F + const @"".SIGURG @"".Signal = 0x17 + const @"".SIGUSR1 @"".Signal = 0xA + const @"".SIGUSR2 @"".Signal = 0xC + const @"".SIGVTALRM @"".Signal = 0x1A + const @"".SIGWINCH @"".Signal = 0x1C + const @"".SIGXCPU @"".Signal = 0x18 + const @"".SIGXFSZ @"".Signal = 0x19 + func @"".init () + var @"time".months [12]string + var @"time".days [7]string + var @"time".Local *@"time".Location + var @"time".UTC *@"time".Location + var @"bufio".ErrInvalidUnreadRune error + var @"net/http".DefaultTransport @"net/http".RoundTripper + var @"crypto/tls".defaultCurvePreferences []@"crypto/tls".CurveID + import rand "crypto/rand" // indirect + var @"crypto/rand".Reader @"io".Reader + type @"sync".rlocker struct { @"sync".w @"sync".Mutex; @"sync".writerSem uint32; @"sync".readerSem uint32; @"sync".readerCount int32; @"sync".readerWait int32 } + func (@"sync".r·1 *@"sync".rlocker) Lock () + func (@"sync".r·1 *@"sync".rlocker) Unlock () + +$$ +_go_.6 0 0 0 644 1173651 ` +go object darwin amd64 go1.4.2 X:precisestack + +! +go13ldbytes.a"encoding/base64.aencoding/json.aerrors.a +fmt.aio.aos.a path.astrings.abufio.acrypto/tls.acrypto/x509.aio/ioutil.a +net.anet/http.a&net/http/httputil.anet/url.apath/filepath.areflect.aruntime.astrconv.a time.a�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts.a�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir.a�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.a math.a sync.async/atomic.a�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive.a�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils.a�J"".NewAuthConfigurationsFromDockerCfg��eH� %H�D$�H;Aw���H���HDŽ$�HDŽ$H�D$0H�D$8H�D$`H�D$hH�H�,$H��H��H�H��L�L$L�D$H��$�H����H�-H��H���H��H��H��$�H��$�H��$�H�$L�L$pL�L$L�D$xL�D$�H��$�H�$H��$�H�\$H��$�H�\$�H�T$H�D$ H�T$@H�D$HH�H�D$(1�H9��H�\$@H�$H�\$HH�\$�H�D$H�T$H�\$ H�\$hH�D$8H�l$(H�l$0H��H�T$`�AH�H�,$H��H��H�H��L�L$L�D$H��$�H����H�-H��H���H��H��H��$�H��$�H��$�H�$L�L$pL�L$L�D$xL�D$�H��$�H�$H��$�H�\$H��$�H�\$�H�L$H�D$ H�L$PH�D$XH�H�D$(1�H9���H�\$PH�$H�\$XH�\$�H�D$H�L$H�T$ H�T$hH�l$(H��H�L$`t$HDŽ$�H��$�H��$H����H�l$0H�,$H�D$8H�D$�H�T$H�L$H�D$ H��$�H��$�H��$H����H�H�$H�H�\$H�H�\$�H�\$H�\$(�"�����p���H�H�$H�H�\$H�H�\$�H�\$H�\$(���������6 +*0runtime.morestack_noctxt� go.string."HOME"�os.Getenv�""".statictmp_0068�� runtime.duffcopy�4runtime.writebarrierstring�path.Join�4go.itab.*os.File.io.Reader�os.Open� go.string."HOME"�os.Getenv�""".statictmp_0073�� runtime.duffcopy�4runtime.writebarrierstring�path.Join� 4go.itab.*os.File.io.Reader� os.Open� 0"".NewAuthConfigurations� type.*os.File� type.io.Reader� 4go.itab.*os.File.io.Reader�  runtime.typ2Itab� type.*os.File� type.io.Reader� 4go.itab.*os.File.io.Reader�  runtime.typ2Itab0� "".autotmp_0076type.*uint8"".autotmp_0075type.*os.File"".autotmp_0072type.[]string"".autotmp_0071�type.*uint8"".autotmp_0067�type.[]string"".autotmp_0064type.string"".autotmp_0063type.string"".autotmp_0062�type.[2]string"".autotmp_0060�type.string"".autotmp_0059_type.[3]string"".p�type.string"".p�type.string "".err�type.error"".r�type.io.Reader "".~r1type.error "".~r06type.*"".AuthConfigurations&"����F���B\:�R�H $G 77 +2sq+CLq ++C`W>Tgclocals·b8a8407971613b03b21a64dc1e56fba0Tgclocals·2e2a9972ea9ced3a58f9e7510cf4914d�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�0"".NewAuthConfigurations��eH� %H;aw���H��8H�D$XH�D$`H�\$@H�$H�\$HH�\$�H�L$H�D$H�T$ H�T$0H��H�D$(tH�D$PH�D$XH�T$`H��8�H� $�H�T$H�D$H�L$H��tH�D$PH�D$XH�L$`H��8�H�T$PH�D$XH�D$`H��8� + 0runtime.morestack_noctxt�("".parseDockerConfig�"".authConfigsPp "".errtype.error "".~r20type.error "".~r1 6type.*"".AuthConfigurations"".rtype.io.Readerp`op5opo�"~,,  ?�Tgclocals·9edc1f6d8fc7336ae101b48cbf822a45Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�("".parseDockerConfig� � +eH� %H�D$�H;Aw���H���HDŽ$�HDŽ$�H�H�$�H�D$H�D$8H�$H��$�H�\$H��$�H�\$�H�L$8H�yH�QH�AH9��H� H��H)�H��H)�H��t H��H�H��H��$�H��$�H��$�H��$�H��$�H��$�H��$�H��$�H��$�H�H�$�H�L$H�L$@H��$�H�$H��$�H�\$H��$�H�\$H�H�D$`H�D$H�L$hH�L$ �H�L$(H�D$0H�D$XH��H�L$PuyH�H� H�CH�H�$H�\$@H�+H�l$H�L$pH�L$H�D$xH�D$�H�D$ �\$(H�(��t(H��$�HDŽ$�HDŽ$�H����H�H�$�H�L$H�L$HH��$�H�$H��$�H�\$H��$�H�\$H�H�D$`H�D$H�L$hH�L$ �H�D$(H�L$0H��t$HDŽ$�H��$�H��$�H����H�\$HH�+H��$�HDŽ$�HDŽ$�H����� " +*0runtime.morestack_noctxtz"type.bytes.Buffer�"runtime.newobject�0bytes.(*Buffer).ReadFrom�Ttype.map[string]map[string]"".dockerConfig�"runtime.newobject�Vtype.*map[string]map[string]"".dockerConfig�.encoding/json.Unmarshal�"go.string."auths"�Ttype.map[string]map[string]"".dockerConfig�4runtime.mapaccess2_faststr�>type.map[string]"".dockerConfig�"runtime.newobject�@type.*map[string]"".dockerConfig� .encoding/json.Unmarshal� +$runtime.panicsliceP�"".autotmp_0091/type.[]uint8"".autotmp_0090type.error"".autotmp_0089@type.*map[string]"".dockerConfig"".autotmp_0088�type.string"".&confs�@type.*map[string]"".dockerConfig "".&confsWrapper�Vtype.*map[string]map[string]"".dockerConfig "".~r0�type.[]uint8 "".err�type.error"".byteData_type.[]uint8 "".buf�$type.*bytes.Buffer "".~r20type.error "".~r1 >type.map[string]"".dockerConfig"".rtype.io.Reader4"�������/���:�:(�eQ([$0&E-�PY�iTgclocals·e94084972e98c8fdf7f2203a35ca807aTgclocals·74b1ee12d224f81a4fda25605fab855d�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�"".authConfigs��eH� %H��$����H;Aw���H��HDŽ$�HDŽ$�H�H�$H�D$�H�\$H�\$PH�H�$�H�D$H�D$HH�$H�<$�H�\$PH�\$�H�\$HH�\$@H��$�H��$81��H�H�$H�L$H��$8H�\$�H��$81�H9��jH��$@H����H��$�H���H��$8H���qH�.H�l$XH�~H�|$`H��$�H��$��H�5H�4$H��$�H�l$H��H�H��H�l$H�T$ H�L$(H�D$0H�t$8H�t$pH��H�D$ht$HDŽ$�H��$�H��$�H�Ĉ�H��$�H�,$H��$�H�T$H��$�H�L$�H�\$H�,$H��H��H�H�H�H�l$H��H��H�H��L�L$XL�D$`H�T$ H�L$(H�D$0H��$�H��t2HDŽ$�H�H��$�H�H��$�H�Ĉ�L�L$xL��$�H��$�1��H��$�H��$H��H��H�H�H����H��$�H��H��H�H�H��H��$�H��H��$���H��H��$H��H��H�H�L��$(L��$0H�H�$H�\$@H�+H�l$H�\$xH�\$H��$�H�\$�H��$8H�$�H��$81�H9������H�\$@H��$�HDŽ$�HDŽ$�H�Ĉ�� � �������_����%�����2 +00runtime.morestack_noctxt�Htype.map[string]"".AuthConfiguration�runtime.makemap�4type."".AuthConfigurations�"runtime.newobject�.runtime.writebarrierptr�� runtime.duffzero�>type.map[string]"".dockerConfig�&runtime.mapiterinit�� runtime.duffcopy�� runtime.duffcopy�6encoding/base64.StdEncoding�Pencoding/base64.(*Encoding).DecodeString�2runtime.slicebytetostring�go.string.":"�strings.Split� """.AuthParseError� """.AuthParseError� +� runtime.duffzero� Htype.map[string]"".AuthConfiguration� $runtime.mapassign1� &runtime.mapiternext�$runtime.panicindex�$runtime.panicindex@�"".autotmp_0111�(type."".dockerConfig"".autotmp_0110�6type.*"".AuthConfigurations"".autotmp_0109�2type."".AuthConfiguration"".autotmp_0108�type.string"".autotmp_0105�Htype.map.iter[string]"".dockerConfig"".autotmp_0103�Htype.map[string]"".AuthConfiguration"".userpass�type.[]string "".err�type.error"".data�type.[]uint8"".conf�(type."".dockerConfig "".reg�type.string"".c�6type.*"".AuthConfigurations "".~r2 type.error "".~r16type.*"".AuthConfigurations"".confs>type.map[string]"".dockerConfig6%����������+�L�=[ +�B $z2 +�$ -*Q(C��E+Tgclocals·f565a1229afec041643831d3cd6a3b7dTgclocals·1c0f8a36a8ada2462e7c582fc8286897�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�,"".(*Client).AuthCheck� � eH� %H��$h���H;Aw���H��H��$(HDŽ$0HDŽ$81�H9�uXH�H�,$H��H��H�H�H�\$H�H�CH�C�H�L$(H�D$0H��$0H��$8H���H��$�H�H�CH�CH��H�H��$�H��$�H��$�H��$�H��$ H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H�H��$�H�l$(H��H��H�H�H��H�|$@H�t$HH�l$PH�D$XH�L$`H�T$hH��$�H��H�L$xtH��$0H��$8H���H=���H�D$pH��$�H��$�H��$�H��$�H��$�H��$�H��$�1��H��$�H���.H��H��H��$�H��$�H��$�H�H�$H�\$pH�\$�H�D$H�L$H��$�H�$H��$�H�D$H��$�H�L$�H�H�$H��$�H�\$�H�D$H�L$H��$�H��H�$H��$�H�D$H��$�H�L$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$0H��$8H��É�����HDŽ$0HDŽ$8H���" +00runtime.morestack_noctxt�.go.string."conf is nil"�fmt.Errorf�4type.*"".AuthConfiguration� go.string."POST"�"go.string."/auth"�"".(*Client).do�� runtime.duffzero�type.int�runtime.convT2E� 2runtime.writebarrieriface� type.[]uint8� runtime.convT2E� +2runtime.writebarrieriface� +>go.string."auth error (%d): %s"� fmt.Errorf@�"".autotmp_0122"type.interface {}"".autotmp_0121"type.interface {}"".autotmp_0119�&type.[]interface {}"".autotmp_0118�"type."".doOptions"".autotmp_0117type.error"".autotmp_0116otype.[]uint8"".autotmp_0115�type.int"".autotmp_0114?(type.[2]interface {} "".err�type.error"".body�type.[]uint8 "".~r1 type.error"".conf4type.*"".AuthConfiguration"".ctype.*"".Client:%�~��������&��,�EX�  � }��NTgclocals·b29a376724b9675f7c9e576a6dabc1e0Tgclocals·25ee8e11891a6b427c03a740dc761f96�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�&"".(*Change).String��eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$�H��$�1�1�H�kH���lH�H�H�KH�T$8H�T$XH�L$@H�L$`H��$�1��H��$�H���"H��H��H�\$hH�T$pH�L$xH�H�$H�\$XH�\$�H�L$H�D$H�\$hH�$H�L$HH�L$H�D$PH�D$�H�H�$H��$�H�\$H�|$���H�L$H�D$H�\$hH��H�$H�L$HH�L$H�D$PH�D$�H�H�,$H��H��H�H�H�\$hH�\$H�\$pH�\$H�\$xH�\$ �H�L$(H�D$0H��$�H��$�H�ĠÉ%�e���������H��uH�H�H�K����H������H�H�H�K�l��� +*0runtime.morestack_noctxt�go.string."C"�� runtime.duffzero�type.string�runtime.convT2E�2runtime.writebarrieriface�type.string�runtime.convT2E�2runtime.writebarrieriface�"go.string."%s %s"�fmt.Sprintf�go.string."A"�go.string."D"0�"".autotmp_0135"type.interface {}"".autotmp_0134�"type.interface {}"".autotmp_0132o&type.[]interface {}"".autotmp_0129�type.string"".autotmp_0128?(type.[2]interface {}"".kind�type.string "".~r0type.string"".changetype.*"".Change"����S�8B: + �  +��zTgclocals·6d340c3bdac448a6ef1256f331f68dd3Tgclocals·0b4080736ceb8b2da6f0b7e8a876e6b8�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/change.go� "".NewAPIVersion��eH� %H��$8���H;Aw���H��HHDŽ$`HDŽ$hHDŽ$pHDŽ$xHDŽ$�H��$PH�$H��$XH�t$H�5H�l$H��H�H��H��$PH��$X�\$ ���?H��$�H��$�H��$�H�H�CH��$�H����H��H��H��$�H��$�H��$�H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0HDŽ$`HDŽ$hHDŽ$pH��$xH��$�H��HÉ�����H�$H�L$H�H�l$H��H��H�H��H�T$ H�L$(H�D$0H��$H��$ H��$H�H�$H�L$H�L$�H�t$H�T$ H�D$(H��$�H��$�H��$�H�D$hH�D$pH��$H��$H��$ H��$1�H��$H�L$HH��$�H��H�l$HH9��fH��H�L$PH����H� H�kH�D$@H�D$8H�L$xH��$�H�L$XH� $H�l$`H�l$�H��$�H��$�H�l$8H9��@H��H�l$H�+H�D$H�\$ H�\$pH��H�D$h��H��$PH��$�H��$XH��$�H�\$XH��$�H�\$`H��$�H��$(1��H��$(H���UH��H��H��$�H��$�H��$�H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$�H�L$H�D$H��$�H��H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0HDŽ$`HDŽ$hHDŽ$pH��$xH��$�H��HÉ����H�L$PH�D$@H��H��H�l$HH9������H��$`H��$hH��$�H��$pHDŽ$xHDŽ$�H��H�� ��^���0 +00runtime.morestack_noctxt�go.string."."� strings.Contains�type.string�runtime.convT2E�2runtime.writebarrieriface�Lgo.string."Unable to parse version %q"�fmt.Errorf�go.string."."�strings.Split�$type."".APIVersion� "runtime.makeslice� strconv.Atoi�� runtime.duffzero�type.string�runtime.convT2E�2runtime.writebarrieriface�type.string�runtime.convT2E�2runtime.writebarrieriface�xgo.string."Unable to parse version %q: %q is not an integer"�fmt.Errorf�$runtime.panicindexp�4"".autotmp_0161"type.interface {}"".autotmp_0160"type.interface {}"".autotmp_0158&type.[]interface {}"".autotmp_0157�type.string"".autotmp_0156�type.*string"".autotmp_0155type.int"".autotmp_0154type.int"".autotmp_0153�"type.interface {}"".autotmp_0151�&type.[]interface {}"".autotmp_0150type.error"".autotmp_0149�type.string"".autotmp_0148type.string"".autotmp_0147?(type.[2]interface {}"".autotmp_0146type.[]string"".autotmp_0144type.int"".autotmp_0143�type.[]string"".autotmp_0141�type.string"".autotmp_0140�(type.[1]interface {} "".val�type.string"".i�type.int "".err�type.error "".ret�$type."".APIVersion "".arrotype.[]string "".~r2Ptype.error "".~r1 $type."".APIVersion"".inputtype.string4%�������e��� >taO�CIxQ� @ 6��vqA���� Tgclocals·56fad8922133a82d7e9abffb05067a58Tgclocals·979c84cf2ee7fa703a7cd5365c579635�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�("".APIVersion.String��eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$�1�1�H��$�H��$�H��$�H��$�1�H��$�H�l$8H��$�H�l$8H9���H�t$HH�.H�T$@H�T$0H�L$PH�L$pH�D$XH�D$xH�,$�H�L$H�D$H�\$pH�$H�\$xH�\$H�L$`H�L$H�D$hH�D$�H�L$ H�D$(H��$�H��H�l$0H9�~yH�L$PH�D$XH�L$`H� $H�D$hH�D$H�H�l$H��H��H�H��H�L$ H�D$(H�t$HH�T$@H��H��H�l$8H9�����H��$�H��$�H�Ę��� +*0runtime.morestack_noctxt�strconv.Itoa�*runtime.concatstring2�go.string."."�*runtime.concatstring2P�"".autotmp_0175�type.*int"".autotmp_0174�type.int"".autotmp_0173�type.int"".autotmp_0172type.string"".autotmp_0171type.int"".autotmp_0170otype.string"".autotmp_0169Otype.string"".autotmp_0168/$type."".APIVersion"".i�type.int "".str�type.string "".~r00type.string"".version$type."".APIVersion"�����(�:W]B  �6�Tgclocals·c45f1008acf31f9ce337f7dfa1fa0204Tgclocals·37f4150aca71c16b472a5e6f54a4a2bc�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�,"".APIVersion.LessThan��eH� %H;aw���H��8H�\$@H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$H�\$`H�\$ H�\$hH�\$(�H�\$0H��| +�D$pH��8��D$p�� + 0runtime.morestack_noctxt�*"".APIVersion.comparepp "".~r1`type.bool"".other0$type."".APIVersion"".version$type."".APIVersionpTop� �f +U+Tgclocals·d83eab2a3f0aa562c88b153605ebed26Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�>"".APIVersion.LessThanOrEqualTo��eH� %H;aw���H��8H�\$@H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$H�\$`H�\$ H�\$hH�\$(�H�\$0H��~ +�D$pH��8��D$p�� + 0runtime.morestack_noctxt�*"".APIVersion.comparepp "".~r1`type.bool"".other0$type."".APIVersion"".version$type."".APIVersionpTop� �f +U+Tgclocals·d83eab2a3f0aa562c88b153605ebed26Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�2"".APIVersion.GreaterThan��eH� %H;aw���H��8H�\$@H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$H�\$`H�\$ H�\$hH�\$(�H�\$0H�� +�D$pH��8��D$p�� + 0runtime.morestack_noctxt�*"".APIVersion.comparepp "".~r1`type.bool"".other0$type."".APIVersion"".version$type."".APIVersionpTop� �f +U+Tgclocals·d83eab2a3f0aa562c88b153605ebed26Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�D"".APIVersion.GreaterThanOrEqualTo��eH� %H;aw���H��8H�\$@H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$H�\$`H�\$ H�\$hH�\$(�H�\$0H��} +�D$pH��8��D$p�� + 0runtime.morestack_noctxt�*"".APIVersion.comparepp "".~r1`type.bool"".other0$type."".APIVersion"".version$type."".APIVersionpTop� �f +U+Tgclocals·d83eab2a3f0aa562c88b153605ebed26Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�*"".APIVersion.compare��eH� %H;aw���L�L$L�T$ H�|$(H�L$H�\$1�M��L9�}DH�1H��H��H9�|eH9�sYI��H�H9�} +H�D$8�����H9�~ +H�D$8�H��H��L9�|�I9�~ +H�D$8�I9�} +H�D$8�����H�D$8�� �� + 0runtime.morestack_noctxt�$runtime.panicindexp"".autotmp_0186type.int"".autotmp_0185type.int"".autotmp_0184type.int"".autotmp_0183type.int"".autotmp_0182type.int "".~r1`type.int"".other0$type."".APIVersion"".version$type."".APIVersion��D�%   + +  + + + � Tgclocals·d83eab2a3f0aa562c88b153605ebed26Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�"".NewClient��eH� %H;aw���H��8H�D$XH�D$`H�\$@H�$H�\$HH�\$H�\$H�H�C�H�L$ H�D$(H�T$0H��tH�D$PH�D$XH�T$`H��8�H��@�)H�L$PH�D$XH�D$`H��8� + 0runtime.morestack_noctxt�*"".NewVersionedClientPp "".~r20type.error "".~r1 type.*"".Client"".endpointtype.stringpjop%o��,; +! +S]Tgclocals·5dfce38b1d248a3900c6ec750de77702Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�"".NewTLSClient��eH� %H;aw���H��hHDŽ$�HDŽ$�H�\$pH�$H�\$xH�\$H��$�H�\$H��$�H�\$H��$�H�\$ H��$�H�\$(H��$�H�\$0H��$�H�\$8H�\$@H�H�C�H�L$PH�D$XH�T$`H��t!HDŽ$�H��$�H��$�H��h�H��@�)H��$�HDŽ$�HDŽ$�H��h� + 0runtime.morestack_noctxt�0"".NewVersionedTLSClient�� "".~r5�type.error "".~r4�type.*"".Client +"".ca`type.string "".key@type.string"".cert type.string"".endpointtype.string ����.���2�! +4 �yTgclocals·8e51ba8a606dfe7bf8ea610f35b1860aTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�0"".NewTLSClientFromBytes��eH� %H;aw���H��HDŽ$�HDŽ$�H��$�H�$H��$�H�\$H��$�H�\$H��$�H�\$H��$�H�\$ H��$�H�\$(H��$�H�\$0H��$�H�\$8H��$�H�\$@H��$�H�\$HH��$�H�\$PH�\$XH�H�C�H�L$hH�D$pH�T$xH��t$HDŽ$�H��$�H��$�H�Ā�H��@�)H��$�HDŽ$�HDŽ$�H�Ā� + 0runtime.morestack_noctxt�B"".NewVersionedTLSClientFromBytes�� "".~r5�type.error "".~r4�type.*"".Client"".caPEMCert�type.[]uint8"".keyPEMBlockPtype.[]uint8"".certPEMBlock type.[]uint8"".endpointtype.string ����1� +��5�$ +1 �yTgclocals·a3afb5a83dcf14cc57a3d3da3be3a7dfTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�*"".NewVersionedClient� � eH� %H;aw���H��xHDŽ$�HDŽ$�H��$�H�$H��$�H�\$�D$�H�\$H�\$8H�D$ H�L$(H�L$XH��H�D$Pt!HDŽ$�H��$�H��$�H��x�H�D$`H�D$hH�D$pH��$�H�$H��$�H�t$H�5H�l$H��H�H���\$ ��� +H��$�H�$H��$�H�\$�H�\$H�\$`H�\$H�\$hH�\$ H�\$pH�D$(H�L$0H�L$XH��H�D$Pt!HDŽ$�H��$�H��$�H��x�H�H�$�H�\$H�\$HH�H�$�H�L$H��H���P1��H�L$@H� $H�<$�)H�$H�H�\$�H�\$@H�$H�<$��H�$H��$�H�\$H��$�H�\$�H�\$@H�$H�<$��H�$(H�\$8H�\$�H�\$@H�$H�<$tH�$0H�\$HH�\$�H�\$@H�$H�<$tRH�$8H�\$`H�\$H�\$hH�\$H�\$pH�\$�H�\$@H��$�HDŽ$�HDŽ$�H��xÉ%륉%�u����%�E����%�����%�����������h���" + 0runtime.morestack_noctxt� "".parseEndpoint�go.string."."� strings.Contains� "".NewAPIVersion�8type."".eventMonitoringState�"runtime.newobject�type."".Client�"runtime.newobject�� runtime.duffzero�,net/http.DefaultClient�.runtime.writebarrierptr�4runtime.writebarrierstring�.runtime.writebarrierptr� .runtime.writebarrierptr� +2runtime.writebarrierslicep�"".autotmp_0193otype.*"".Client"".autotmp_0192_:type.*"".eventMonitoringState,"".requestedAPIVersion/$type."".APIVersion "".errOtype.error"".u"type.*net/url.URL "".~r3Ptype.error "".~r2@type.*"".Client&"".apiVersionString type.string"".endpointtype.string4���������R�>�2< !?K !�*@.P�,iC`$8�Tgclocals·677e212df4ff2dc5d1bd7207f0cb343fTgclocals·63a71a9d82a0cb5094b44aef6b6fe396�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�2"".NewVersionnedTLSClient��eH� %H;aw���H��hHDŽ$�HDŽ$�H�\$pH�$H�\$xH�\$H��$�H�\$H��$�H�\$H��$�H�\$ H��$�H�\$(H��$�H�\$0H��$�H�\$8H��$�H�\$@H��$�H�\$H�H�T$PH�L$XH�D$`H��$�H��$�H��$�H��h� + 0runtime.morestack_noctxt�0"".NewVersionedTLSClient�� "".~r6�type.error "".~r5�type.*"".Client&"".apiVersionString�type.string +"".ca`type.string "".key@type.string"".cert type.string"".endpointtype.string�����2� �3Tgclocals·c984d5bd78e9da313cca302adec9d408Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�0"".NewVersionedTLSClient� +� +eH� %H�D$�H;Aw���H���HDŽ$8HDŽ$@H��$�H�$H��$�H�\$�H�\$H��$�H�\$H��$�H�\$ H��$�H�D$(H�L$0H��$�H��H��$�t$HDŽ$0H��$8H��$@H����H��$H�$H��$H�\$�H�\$H��$�H�\$H��$�H�\$ H��$�H�D$(H�L$0H��$�H��H��$�t$HDŽ$0H��$8H��$@H����H��$H�$H��$H�\$�H�l$H�T$H�L$ H�D$(H�t$0H��$�H��H��$�t$HDŽ$0H��$8H��$@H����H��$�H�$H��$�H�\$H��$�H�\$H��$�H�\$H��$�H�\$ H��$�H�\$(H��$�H�\$0H��$�H�\$8H��$�H�l$@H��$�H�T$HH��$�H�L$PH��$ H�\$XH��$(H�\$`�H�T$hH�L$pH�D$xH��$0H��$8H��$@H���� +*0runtime.morestack_noctxt�$io/ioutil.ReadFile�$io/ioutil.ReadFile�$io/ioutil.ReadFile� B"".NewVersionedTLSClientFromBytes��"".caPEMCert/type.[]uint8"".keyPEMBlock�type.[]uint8 "".err�type.error"".certPEMBlock_type.[]uint8 "".~r6�type.error "".~r5�type.*"".Client&"".apiVersionString�type.string +"".ca`type.string "".key@type.string"".cert type.string"".endpointtype.string<"�������p�����2�:W$W$?$�S���;Tgclocals·e11b7011fe7d18f281fa367784f98637Tgclocals·f27fde19da2a9a9e0264e00d44cbb36a�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�&"".NewClientFromEnv��eH� %H;aw���H��(H�D$8H�D$@H�$H�H�C�H�L$H�D$H�T$ H��tH�D$0H�D$8H�T$@H��(�H��@�)H�L$0H�D$8H�D$@H��(� + 0runtime.morestack_noctxt�8"".NewVersionedClientFromEnv0P "".~r1type.error "".~r0type.*"".ClientPVOP%O +��,' +% +?aTgclocals·0528ab8f76149a707fd2f0025c2178a3Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�8"".NewVersionedClientFromEnv��eH� %H��$(���H;Aw���H��XHDŽ$xHDŽ$��H� $H�D$H�T$H��$�H��H��$�t$HDŽ$pH��$xH��$�H��X�H���H�H�AH�L$h�Y����H��$�H�$H��$�H�D$H�H�l$H��H��H�H�H�D$ �H�t$(H�l$0H�D$8H��$�H��$H��$�H���7H��$�H��$�H��$�H��$�H��$�H�H�CH��$�H����H��H��H��$H��$H��$H�H�$H��$�H�\$�H�L$H�D$H��$H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$H�\$H��$H�\$H��$H�\$ �H�L$(H�D$0HDŽ$pH��$xH��$�H��XÉ����H��$�H�H�CH��$�H����H��H��H��$H��$H��$H�H�$H��H����H��H�\$�H�T$H�D$H��$H�$H��$�H�T$H��$�H�D$�H�H�,$H��H��H�H�H��$H�\$H��$H�\$H��$H�\$ �H�T$(H�D$0H��$�H��$�H��$8H����H�-H��H���H��H��H��$(H��$0H��$ H�$H�|$hH���wH�wH�|$H�H��H��$ H�$H��$(H�\$H��$0H�\$�H�T$H�D$ H��$�H��$�H��$8H���H�-H��H���H��H��H��$(H��$0H��$ H�$H�|$hH����H�wH�|$H�H��H��$ H�$H��$(H�\$H��$0H�\$�H�T$H�D$ H�T$pH�D$xH��$8H���KH�-H��H���H��H��H��$(H��$0H��$ H�$H�|$hH����H�oH�|$H��H�H��H��$ H�$H��$(H�\$H��$0H�\$�H�L$H�D$ H��$�H�$H��$�H�\$H��$�H�\$H��$�H�\$H�\$pH�\$ H�\$xH�\$(H��$�H�L$0H��$�H�D$8H��$`H�\$@H��$hH�\$H�H�D$PH�T$XH�L$`H��$pH��$xH��$�H��XÉ�����������D����������������0���� ��4���H��$�H�$H��$�H�D$H��$`H�\$H��$hH�\$�H�D$ H�T$(H�L$0H��$pH��$xH��$�H��XÉ�����< +00runtime.morestack_noctxt|"".getDockerEnv�go.string."://"�strings.SplitN�type.string�runtime.convT2E�2runtime.writebarrieriface�hgo.string."could not split %s into two parts by ://"�fmt.Errorf� +type.string� runtime.convT2E� 2runtime.writebarrieriface� ,go.string."https://%s"� fmt.Sprintf� """.statictmp_0222� � runtime.duffcopy�4runtime.writebarrierstring�$path/filepath.Join�""".statictmp_0225�� runtime.duffcopy�4runtime.writebarrierstring�$path/filepath.Join�""".statictmp_0228�� runtime.duffcopy�4runtime.writebarrierstring�$path/filepath.Join�0"".NewVersionedTLSClient�$runtime.panicindex�*"".NewVersionedClientP�B"".autotmp_0229type.*[2]string"".autotmp_0227type.[]string"".autotmp_0226type.*[2]string"".autotmp_0224type.[]string"".autotmp_0221type.[]string"".autotmp_0220"type.interface {}"".autotmp_0219*type.*[1]interface {}"".autotmp_0218&type.[]interface {}"".autotmp_0217�"type.interface {}"".autotmp_0215�&type.[]interface {}"".autotmp_0214type.error"".autotmp_0213type.*"".Client"".autotmp_0212type.error"".autotmp_0210type.string"".autotmp_0209type.[2]string"".autotmp_0208type.string"".autotmp_0207type.[2]string"".autotmp_0206type.string"".autotmp_0205?type.[2]string"".autotmp_0203(type.[1]interface {}"".autotmp_0201�type.string"".autotmp_0200�(type.[1]interface {}"".autotmp_0198otype.[]string +"".ca�type.string "".key�type.string"".cert�type.string"".parts�type.[]string"".dockerHost�type.string "".err�type.error"".dockerEnv�$type.*"".dockerEnv "".~r20type.error "".~r1 type.*"".Client&"".apiVersionStringtype.stringB%�d������������ `�=$\������ gP=��v�v�+�+�+�^AFTgclocals·7df0f47b43308a447b1c5362b1e10571Tgclocals·bae70cbfa95aa7f2f402b02d37b0b239�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�B"".NewVersionedTLSClientFromBytes�+�*eH� %H��$x���H;Aw���H��HDŽ$�HDŽ$�H��$H�$H��$H�\$�D$�H�\$H��$�H�D$ H�L$(H��$0H��H��$(t$HDŽ$xH��$�H��$�H���HDŽ$HHDŽ$PHDŽ$XH��$hH�$H��$pH�t$H�5H�l$H��H�H���\$ ���� H��$hH�$H��$pH�\$�H�\$H��$HH�\$H��$PH�\$ H��$XH�D$(H�L$0H��$0H��H��$(t$HDŽ$xH��$�H��$�H���H��$ ��H��$8��H��$ H�$H��$(H�\$H��$0H�\$H��$8H�\$H��$@H�\$ H��$HH�\$(�H��$xH�l$0H��H���H�D$xH��$�H��$0H��H��$(t$HDŽ$xH��$�H��$�H���H�H�$�H�|$H��$�H���1��H�H�$�H�D$H����H��H��H��$hH��$pH��$xH��$�H���H�H�$H��$`H�D$H��$�H�\$�H��$�H�$H�<$�_H�$H��$`H�\$H��$hH�\$H��$pH�\$�H��$�H��$�H��$P��H��@���H�H�$�H�|$H��H���O1��H��$�H� $H�<$�%H�$hH��$�H�\$�H��$(H��$�H��t,HDŽ$xH��$�H��$0H��$�H���H��$�H�H�$�H�\$H��$�H�H�$�H�|$H��$�H���y1��H�H�$�H�|$H��H��$�H���A1��H�1�H9���H� $H�<$��H��$�H��$8H�D$H��$@H�L$�H��$�H�$H�<$��H�$H��$�H�\$�H��$�H�$H�<$�OH�$H��$�H�\$�H��$�H�$H�<$�H�$H��$H�\$H��$H�\$�H��$�H�$H�<$��H�$(H��$�H�\$�H��$�H�$H�<$��H�$0H��$�H�\$�H��$�H�$H�<$taH�$8H��$HH�\$H��$PH�\$H��$XH�\$�H��$�H��$xHDŽ$�HDŽ$�H��É%떉%�`����%�&����%������%�����%�k����%�!���H�H�$H�H�\$H�H�\$�H��$�H�D$����������������%����������H�H�$H�D$�H�\$H��$�H�H�$H�D$�H�\$H��$�H�H�$�H�D$H��$�H�$H�<$�H��$�H�\$�H��$�H�$H�<$��H�$H��$�H�\$�H��$�H����H�hH�EH�EH�EH��$�H�$H��$PH�\$H��$XH�\$H��$`H�\$��\$ ���H�H�+H��$H�kH��$ HDŽ$�HDŽ$H�H�$�H�D$H��$�H�$H�<$��H��$H�\$H��$ H�\$�H��$�H��$�H�1�H9�t,H��$�HDŽ$xH��$�H��$�H���H�H�$H�H�\$H�H�\$�H�D$륉%�\���H��$�H�$H�<$tH�$@H��$�H�\$��'����%�ۉ�Y����%�$����%������%���������������H�H�+H��$H�kH��$HDŽ$�HDŽ$�H�H�$�H�D$H��$�H�$H�<$��H��$H�\$H��$H�\$�H��$�H��$�H�1�H9�t,H��$�HDŽ$xH��$�H��$�H���H�H�$H�H�\$H�H�\$�H�D$륉%�\��������� +00runtime.morestack_noctxt� "".parseEndpoint�go.string."."� strings.Contains� "".NewAPIVersion�,crypto/tls.X509KeyPair�� runtime.duffcopy� ,type.crypto/tls.Config� "runtime.newobject� +� runtime.duffzero� +go.itab.*bytes.Buffer.io.Reader�1$type.*bytes.Buffer�1type.io.Reader�1>go.itab.*bytes.Buffer.io.Reader�1 runtime.typ2Itab��@"".autotmp_0312type.*uint8"".autotmp_0311�type.*"".Error"".autotmp_0310type.*"".Error"".autotmp_0308�$type.*bytes.Buffer"".autotmp_0307$type.*bytes.Buffer"".autotmp_0306type.*"".Error"".autotmp_0304type.string"".autotmp_0303type.error"".autotmp_0301�type.string"".autotmp_0299$type.*bytes.Buffer"".body/type.[]uint8"".status�type.intbytes.buf·2�type.[]uint8"".body_type.[]uint8"".breader�$type.*bufio.Reader"".dial�type.net.Conn"".address�type.string"".protocol�type.string"".resp�.type.*net/http.Response "".err�type.error "".req�,type.*net/http.Request "".err�type.error "".err�type.error "".buf�type.[]uint8"".params�type.io.Reader "".~r5�type.error "".~r4�type.int "".~r3�type.[]uint8"".doOptionsP"type."".doOptions"".path0type.string"".methodtype.string"".ctype.*"".Client�%�������������]��=���c�����R��������������m Q �NC8 +#.DaN%L^N/R\^O�NJ� 6!A8!(#NTN�B�� +���[# �� W��:��=�1"��i�YyTgclocals·162ebaa0b58a36a0548c1b1cb0cd3669Tgclocals·e420e0cb0a7fa3debafcec3aabd8dd0a�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�&"".(*Client).stream�X�XeH� %H��$����H;Aw���H���H��$�HDŽ$HHDŽ$PH���tH��$�H�4$H�D$H�5L�D$L��H�H��H��$��\$ ���5H��$��HDŽ$�HDŽ$�HDŽ$�H�H�$�H�D$H��$�H�$H�<$��H��$�H�\$H��$�H�\$H��$�H�\$�H��$�H�@H�@ ����H��$�H�1�H9��:H��$�H��$�H��$H��$�H��$H��$�H����H��$�H�,$H��$�H�t$H�5L�D$L��H�H���\$ ���FH��$�H�$H��$�H�\$H��$�H�\$�H�T$H�L$ H��$�H�$H��$�H�t$H��$�H�T$H��$�H�L$H��$H�l$ H��H�H��H�l$0H�L$8H�T$@H��$8H��H��$0tH��$HH��$P��H����H�l$`H�}8H�<$H�H�|$H��H�H�H�H�l$H��H��H�H��H��$�H��ukH��$�H�4$H�D$H�5L�D$L��H�H���\$ ��t8H�t$`H�~8H�<$H�5H�|$H�H�H�H�l$H��H��H�H��H��$H��$�1��H�H�$H�L$H��$�H�\$�H��$�1�H9���H��$�H���iH� H�CH��$�H���IH�3H�SH��$�H��$�H�\$`H�k8H�,$H��$�H�t$H��$�H�T$H��$�H�L$H��$�H�D$ �H��$�H�$�H��$�1�H9��T���H��$�H�k(H����H�UH��$�H�MH��$�H�k(H���|H�]8H��$�H�]@H��$�H��$ uH�H��$ H�H��$(H��$0uH�H��$0H�H��$8H��$�H��� H�$H�L$H�-L�D$L��H��H�H���\$ ����H��$�H�$H��$�H�\$H��$�H�\$H��$�H�\$�H�t$ H��$PH�l$(H��$XH�L$0H�T$8H��$H��H��$tH��$HH��$P��H����H�,$H���5H�^ Sj�YYH���H�H�$H��$PH�\$H��$XH�\$�H�\$H�,$H��H��H�H��H�\$H�\$pH�H�$H��$PH�\$H��$XH�\$�H�\$H�l$H��H��H�H�H�\$`H�$�H�L$H�T$ H��$H��H��$tH��$HH��$P��H����H��$@H�����H�,$�T$H�L$H��$HH�,$��$P�T$H��$XH�L$H��$@H�\$�H�l$ �T$(H�L$0H��$0H�l$��$8�T$H��$@H�L$H��$XH�$H��$PH�[@��H�\$pH�$H�\$`H�\$�H�L$H�T$H�\$ H��$H��H��$�H��$@H��~M1�1�1�H��$HH�l$��$P�T$H��$XH�L$H��$XH�$H��$PH�[@��H��$H��$H�$H�Z ��H�T$H�L$H��$�H�$H��$�H�L$H�H�l$H��H��H�H���\$ ��t,H�H��$HH�H��$P��H����H��$H��$HH��$H��$P��H����H�L$XH���- H�Q@H�iHH��$xH�,$H��$pH��� H�Z Sj�H�T$hYYH���� +H�ZH�����H�ZH�������$����H�z8H�<$H�H�|$H��H�H��H�T$XH�t$H�L$ H���eH��$�H�4$H��$�H�L$H�-L�D$L��H��H�H��H�T$X�\$ �����$����H�=H�<$H��t|H�j@H�|$H��H�H��H�\$H�l$H��H��H�H�H��$ H�,$H��H��H�H��H�T$(H�L$0H��$0H��$HH��$8H��$P��H���É�H�=H�<$H���`H�j@H�|$H��H�H��H�D$H�L$ H��$�H��$�H��$�H��$�H�H�$�H�L$H��H����1��H��$�H� $H�<$��H��$�H�\$H��$�H�\$�H��$�H�\$hH�H�$�H�L$H��$�H�\$hH�$H�H��$`H�D$H��$hH�L$�H��$�H�t$H�T$ H��$(H�-H9�uvH��$ H�4$H�T$H�-H�l$H�-H�l$�H��$(H��$ H��$��\$ ��t&HDŽ$HHDŽ$P��H����H��H��$ tH��$HH��$P��H����H�Y8H����H��$�H�H�CH��$�H����H��H��H��$H��$ H��$(H�H�$H�L$H�D$0�H�D$H�L$H��$H�$H��$`H�D$H��$hH�L$�H��$ H�,$H��H��H�H�H��$H�\$H��$ H�\$H��$(H�\$ �H��$�H�YH�������H��$�H�H�CH��$�H����H��H��H��$H��$ H��$(H�H�$H�L$�H�D$H�L$H��$H�$H��$`H�D$H��$hH�L$�H��$ H�,$H��H��H�H�H��$H�\$H��$ H�\$H��$(H�\$ �������B�����D���H�YH���YH��$`1��H��$`H���1H��H��H��$H��$ H��$(H�H�$H�L$�H�D$H�L$H��$H�$H��$`H�D$H��$hH�L$�H�H�$H��$�H�\$H�D$�H�D$H�L$H��$H��H�$H��$`H�D$H��$hH�L$�H��$ H�,$H��H��H�H�H�H�l$H��H��H�H�H��$H�\$ H��$ H�\$(H��$(H�\$0�H��$�����������H�Y(H�������H�i H��$�H�i(H��$�HDŽ$�HDŽ$�H�H�$�H�\$H�\$xH�\$xH�$H�<$��H��$�H�\$H��$�H�\$�H�\$xH�\$xH� 1�H9�tSH�T$xH��$�H��$�H��$�H��$�H��$�H��$HH��$�H��$P��H����H�H�$H�H�\$H�H�\$�H�L$�{����%�8����%�&���������������$����H�=H�<$H��t|H�j@H�|$H��H�H��H�\$H�l$H��H��H�H�H��$ H�,$H��H��H�H��H�T$(H�L$0H��$0H��$HH��$8H��$P��H���É�H�=H�<$H��tzH�j@H�|$H��H�H��H�\$H�l$ H��H���H��H��H���H��H��$ H�,$H��H��H�H�H��$0H�l$H��H��H�H��H�T$8H�L$@�C�����H�=H�<$H����H�j@H�|$H��H�H��H�\$H�,$H��H��H�H��L�D$L��$�H�|$H��$�H�t$ H��$�H�L$(H�T$0H��$HH��H��$@tH��$HH��$P��H����H�\$XH�kH�l$PL��$H��$H��$H�H�$�H�L$H��$�H�l$PH�)H��$H�$H��$H�\$H��$H�\$�H�\$H�l$H��H��H�H�H��$�H�$H�<$t}H�$�H��$�H��$�H� 1�H9�t&H��$�H��$PH��$H��H����H�H�$H�H�\$H�H�\$�H�L$뫉%�w�����A�����H���É�������������H���É�����H��$�H�kH�,$H�\$`H�\$�H�L$H�T$H�\$ H��$8H��H��$0�`���H�$H�Z ��H�T$H�L$H��$�H�$H��$�H�L$H�H�l$H��H��H�H���\$ ��t,H�H��$HH�H��$P��H����H��$0H��$HH��$8H��$P��H���ÉE�|����E�N�������������H��$��]�������H��$�H�{h�����H��$�H�$�H�L$H�T$H��$H��H��$�]���H��$HH��$P��H����H�H�$H�H�\$H�H�\$�H�D$�����%�*���H�������H��$�H�4$H�D$H�5L�D$L��H�H���\$ ��������j���� +00runtime.morestack_noctxt� go.string."POST"� runtime.eqstring�"type.bytes.Reader�"runtime.newobject�2runtime.writebarrierslice�>go.itab.*bytes.Reader.io.Reader�(go.string."/version"� runtime.eqstring�&"".(*Client).getURL� &net/http.NewRequest� &runtime.deferreturn� +,go.string."User-Agent"� +6go.string."go-dockerclient"� +&net/http.Header.Set�  go.string."POST"�  runtime.eqstring� 0go.string."Content-Type"� ,go.string."plain/text"� &net/http.Header.Set� � runtime.duffzero� ,type.map[string]string� &runtime.mapiterinit�&net/http.Header.Set�&runtime.mapiternext�"io/ioutil.Discard�"io/ioutil.Discard�"io/ioutil.Discard�"io/ioutil.Discard� go.string."unix"� runtime.eqstring�net.Dial�&runtime.deferreturn�"runtime.deferproc�type.io.Reader�runtime.convI2I�bufio.NewReader�type.io.Writer�runtime.convI2I�2net/http.(*Request).Write�&runtime.deferreturn�time.Now�time.Time.Add� +�*net/http.ReadResponse� +� +� encoding/json.(*Decoder).Decode�- io.EOF�. io.EOF�. io.EOF�.runtime.ifaceeq�/&runtime.deferreturn�0&runtime.deferreturn�1type.string�2runtime.convT2E�32runtime.writebarrieriface�4fmt.Fprint�5type.string�5runtime.convT2E�62runtime.writebarrieriface�7fmt.Fprintln�8� runtime.duffzero�9type.string�9runtime.convT2E�:2runtime.writebarrieriface�:type.string�;runtime.convT2E�<2runtime.writebarrieriface�<*go.string."%s %s\x0d"�=fmt.Fprintf�>.type.errors.errorString�>"runtime.newobject�?4runtime.writebarrierstring�@Bgo.itab.*errors.errorString.error�A&runtime.deferreturn�A0type.*errors.errorString�Atype.error�BBgo.itab.*errors.errorString.error�B runtime.typ2Itab�Ctype.io.Reader�Cruntime.convI2I�Dio.Copy�E&runtime.deferreturn�Etype.io.Reader�Fruntime.convI2I�G�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.StdCopy�Gtype.io.Reader�Hruntime.convI2I�H"io/ioutil.ReadAll�J&runtime.deferreturn�Ktype."".Error�K"runtime.newobject�L2runtime.slicebytetostring�L4runtime.writebarrierstring�M.go.itab.*"".Error.error�M&runtime.deferreturn�Ntype.*"".Error�Ntype.error�N.go.itab.*"".Error.error�N runtime.typ2Itab�O&runtime.deferreturn�O&runtime.deferreturn�P*net/http.(*Client).Do�Q +�Qgo.itab.*bytes.Reader.io.Reader�V runtime.typ2Itab�Wgo.string."PUT"�W runtime.eqstring�� �"".autotmp_0360"type.interface {}"".autotmp_0359*type.*[1]interface {}"".autotmp_0358&type.[]interface {}"".autotmp_0357type.*uint8"".autotmp_0356type.error"".autotmp_0355� 0type.*errors.errorString"".autotmp_0354"type.interface {}"".autotmp_0353"type.interface {}"".autotmp_0351&type.[]interface {}"".autotmp_0350"type.interface {}"".autotmp_0348�&type.[]interface {}"".autotmp_0347� 6type.*encoding/json.Decoder"".autotmp_03466type.*encoding/json.Decoder"".autotmp_0345type.io.Reader"".autotmp_0344type.*uint8"".autotmp_0343� type.*"".Error"".autotmp_0342type.*"".Error"".autotmp_0341type.time.Time"".autotmp_0340type.string"".autotmp_0338�$type.*bytes.Reader"".autotmp_0337$type.*bytes.Reader"".autotmp_0336(type.[1]interface {}"".autotmp_03350type.*errors.errorString"".autotmp_0334�(type.[2]interface {}"".autotmp_0333�(type.[1]interface {}"".autotmp_0332type.error"".autotmp_0330type.string"".autotmp_0329type.*"".Error"".autotmp_0328type.bool"".autotmp_0327type.string"".autotmp_0325type.string"".autotmp_0324�type.time.Time"".autotmp_0323�type.time.Time"".autotmp_0322type.error"".autotmp_0320�6type.map.iter[string]string"".autotmp_0318�type.string"".autotmp_0317�type.error"".autotmp_0316$type.*bytes.Reader +"".&m�(type.*"".jsonMessage "".~r0�type.errorerrors.text·2�type.string$encoding/json.r·2�type.io.Reader"".body�type.[]uint8"".status� type.intbytes.b·2�type.[]uint8 "".err�type.error "".dec� 6type.*encoding/json.Decoder "".err�type.error"".body�type.[]uint8"".breader� $type.*bufio.Reader "".err�type.error"".dial�type.net.Conn"".address�type.string"".protocol�type.string"".resp� .type.*net/http.Response "".val�type.string "".key�type.string "".err�type.error "".req� ,type.*net/http.Request "".err�type.error "".~r3�type.error "".streamOptionsP*type."".streamOptions"".path0type.string"".methodtype.string"".ctype.*"".Client�%� �� � �� � �� � �� � -� � : �� � �� � +� � �� � �� � �� � �� � O� � � � �� � -� � �� � ��,��EX�Q �C98�F$&.  Dd%L^�AMZ,.Q"�_.��0&+� � ��(_.�G{�;.HR,.M(# >A �sYH����FO# �W ��xc�^7:O<T?!�$wz +w} �b=b-1.g/^3�t?4@l5165�h�ITgclocals·7e331f181b2554581236d61d5561e53dTgclocals·147deda5d3defe8e1d522f194155c84f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�&"".(*Client).hijack�4�3eH� %H��$����H;Aw���H��H�H�$�L�D$L��$�H��$�L��H���HDŽ$HDŽ$ H��$�H���� H��$�H�,$H��$�H�t$H�5L�D$L��H�H��L��$��\$ ���� HDŽ$�HDŽ$�I�x@�6I�h@H�$H��H��H�H��H�|$H��$�H�t$H��$�H�l$ H��$�H�L$(H�T$0H��$�H��H��$�tH��$H��$ ��H�Ę�H��$hH��$pH��$xH�H�$�H�L$H��H���� +1��H�L$pH� $H�<$�� +H��$hH�\$H��$pH�\$H��$xH�\$�L��$�H�\$pH�\$pH� 1�H9��! +H�\$pH��$�H��$�I�x u.L�$H�$ H�H�\$H�H�\$�L��$�I�x0u&L�$H�$0H�H�\$H�H�\$�H��$�H�$H��$�H�\$H��$�H�\$�H�T$H�L$ H��$�H�$H��$�H�\$H��$HH�T$H��$PH�L$H��$�H�\$ H��$�H�\$(�H�L$0H�\$8H��$�H�\$@H��$H��$�t.H��$�H��$H��$H��$ ��H�Ę�H�L$HH�y8H�<$H�H�|$H��H�H�H�H�l$H��H��H�H��H��$�H�i(H���rH�UH�EH�i(H���TH�]8H��$XH�]@H��$`H����H��$�H�$H��$�H�D$H�-L�D$L��H��H�H��H��$��\$ ����HDŽ$HDŽ$ H�Y1�H9�tYH��$�H����H��$�H�,$H��$�H�t$H�5L�D$L��H�H��H��$��\$ ���pH��$�H�$H��$�H�\$H��$XH�\$H��$`H�\$�H�T$ H��$H�L$(H��$ H�\$0H��$�H�\$8H��$H��$�t.H��$�H��$H��$H��$ ��H�Ę�H�$H�L$H�D$�H�L$H�L$PH� $H� Qj�YYH���~H�\$PH�$H�\$HH�\$�H��$�H�1�H9�tPH�\$HH�H�$H�)H�l$H�\$HH�\$�H�H�$H��$�H�+H�l$H�D$�H�H�$�H�\$H�\$xH�H�$�H�\$H��$�H�\$PH�$�H�T$H�L$H�\$H�\$hH�\$xH�$H��$(H�T$H��$0H�L$�H��$�H�$H�\$hH�\$�H�\$xH�H�kH��$0H�,$H��$(H���7H�Z Sj�YYH���H�H�$�H�\$H��$�H�H�$H�D$�H�L$H��$�H�$H�L$�H�H�$�H�\$H��$�H�H�$H�D$�H�L$H��$�H�$H�L$�H�H�$�H�\$H��$�H�H�$H�D$�H�L$H��$�H�$H�L$�H�H�$�H�L$H�-H�)H�L$`H� $H�<$��H�$H��$�H�\$�H�\$`H�$H�<$��H�$H��$�H�\$�H�\$`H�$H�<$�rH�$H��$�H�\$�H�\$`H�$H�<$�;H�$ H��$�H�\$�H�\$`Sj�YYH�H�$�H�L$H�-H�)H�L$XH� $H�<$��H�$H��$�H�\$�H�\$XH�$H�<$��H�$H�\$xH�\$�H�\$XH�$H�<$�kH�$H��$�H�\$�H�\$XSj�YYH�H�$H��$�H�+H�l$H�D$�HDŽ$8HDŽ$@H�H�$H��$�H�+H�l$H��$8H�\$�H��$8H��$�H��$@H��$�HDŽ$8HDŽ$@H�H�$H��$�H�+H�l$H��$8H�\$�H��$�H��$8H��$�H��$@H��$�H��t&H��$H��$�H��$ ��H�Ę�H��$H��$ ��H�ĘÉ%�����%�U����%�����%�����%�����%�K����%������H�ĘÉ�������H�Ę�H��$�H�$H��$�H�\$H��$XH�\$H��$`H�\$H�iH�l$ �H�T$(H��$H�L$0H��$ H�\$8H��$�H�\$@H��$H��$������H��$�H��$H��$H��$ ��H�Ę�H�H�+H��$�H�kH��$�H�i(H��tH�](H��$XH�]0H��$`�����E�މE�����E����H�H�$H�H�\$H�H�\$�L��$�H�L$�����%�I�����'���H��$��]���9���H��$�H�{h�&���H��$�H�$�L��$�H�L$H�T$H��$H��H��$�����H��$H��$ ��H�Ę�� +00runtime.morestack_noctxtP*type."".hijackOptionsb"runtime.newobject�� runtime.duffcopy�(go.string."/version"� runtime.eqstring�*encoding/json.Marshal�&runtime.deferreturn�"type.bytes.Buffer�"runtime.newobject�� runtime.duffzero�2runtime.writebarrierslice�>go.itab.*bytes.Buffer.io.Reader�"io/ioutil.Discard�"io/ioutil.Discard� 2runtime.writebarrieriface� "io/ioutil.Discard� "io/ioutil.Discard� 2runtime.writebarrieriface� +&"".(*Client).getURL� &net/http.NewRequest� &runtime.deferreturn� 0go.string."Content-Type"�,go.string."plain/text"�&net/http.Header.Set� go.string."unix"� runtime.eqstring� go.string."unix"� runtime.eqstring�net.Dial�&runtime.deferreturn�>net/http/httputil.NewClientConn�Pnet/http/httputil.(*ClientConn).Close·f�"runtime.deferproc�Dnet/http/httputil.(*ClientConn).Do�&type.chan struct {}�"runtime.chansend1�&type.chan struct {}�"runtime.chanrecv1�type.net.Conn�"runtime.newobject�$type.*bufio.Reader�"runtime.newobject�Lnet/http/httputil.(*ClientConn).Hijack�2runtime.writebarrieriface�.runtime.writebarrierptr�"runtime.deferproc�type.chan error�"runtime.newobject�type.chan error� runtime.makechan�.runtime.writebarrierptr�type.chan error�"runtime.newobject�type.chan error� runtime.makechan�.runtime.writebarrierptr�type.chan bool�"runtime.newobject�type.chan bool� runtime.makechan�.runtime.writebarrierptr��type.struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *"".hijackOptions; A3 **bufio.Reader }�"runtime.newobject�"".func·001� .runtime.writebarrierptr�!.runtime.writebarrierptr�!.runtime.writebarrierptr�".runtime.writebarrierptr�"runtime.newproc�"�type.struct { F uintptr; A0 *"".hijackOptions; A1 *net.Conn; A2 *chan error }�#"runtime.newobject�#"".func·002�#.runtime.writebarrierptr�$.runtime.writebarrierptr�%.runtime.writebarrierptr�%runtime.newproc�%type.chan bool�&"runtime.chanrecv1�&type.chan error�'"runtime.chanrecv1�(type.chan error�("runtime.chanrecv1�)&runtime.deferreturn�*&runtime.deferreturn�+&runtime.deferreturn�,&runtime.deferreturn�-"".tlsDial�/&runtime.deferreturn�/go.string."tcp"�0$type.*bytes.Buffer�0type.io.Reader�1>go.itab.*bytes.Buffer.io.Reader�1 runtime.typ2Itab�28"".(*Client).checkAPIVersion�3&runtime.deferreturn��F"".autotmp_0393��type.*struct { F uintptr; A0 *"".hijackOptions; A1 *net.Conn; A2 *chan error }"".autotmp_0392��type.*struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *"".hijackOptions; A3 **bufio.Reader }"".autotmp_0391�$type.*bufio.Reader"".autotmp_0390�type.net.Conn"".autotmp_0388�$type.*bytes.Buffer"".autotmp_0387$type.*bytes.Buffer"".autotmp_0386type.error"".autotmp_0385�type.error"".autotmp_0383type.chan error"".autotmp_0381�type.struct {}"".autotmp_0379�type.string"".autotmp_0378$type.*bytes.Buffer"".&errChanIn� type.*chan error"".&rwc�type.*net.Conn "".&br�&type.**bufio.Reader""".&hijackOptions�,type.*"".hijackOptions"".&errChanOut� type.*chan error"".&exit�type.*chan boolbytes.buf·2_type.[]uint8"".errOut�type.error"".errIn�type.error"".clientconn�Dtype.*net/http/httputil.ClientConn"".dial�type.net.Conn"".addresstype.string"".protocol�type.string "".err�type.error "".req�,type.*net/http.Request "".err�type.error "".buf/type.[]uint8"".params�type.io.Reader "".err�type.error "".~r3�type.error"".path0type.string"".methodtype.string"".ctype.*"".Client�%����������-��������a������������mY P�.&� .C&Td l .#  +')�AQQQ��)em& $0 u. "M(+#�0{�1Yf�O +D��d #�<Q&QQe� L(8+EeK'�h�^ATgclocals·84e82484f467e1dc08e5640e075b9b76Tgclocals·65dfd25068bbba2abebc869f9ef9f7a5�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�&"".(*Client).getURL��eH� %H�D$�H;Aw���H���HDŽ$HDŽ$H��$�H�k(H�,$�H�L$H�D$H�L$xH� $H��$�H�D$H�H�l$H��H��H�H��H�L$ H�D$(H�L$8H�D$@H��$�H�k(H����H�MH�L$xH�EH��$�H��u@H� $H�D$H�-L�D$L��H��H�H���\$ ��tH�D$8H�D$@H��$�H�{8��H�\$8H�\$hH�\$@H�\$pH��$H�\$XH��$H�\$`H��$�1��H��$�H����H��H��H��$�H��$�H��$�H�H�$H�\$hH�\$�H�L$H�D$H��$�H�$H�L$HH�L$H�D$PH�D$�H�H�$H��$�H�\$H�|$��H�D$8�H�L$H�D$H��$�H��H�$H�L$HH�L$H�D$PH�D$�H�H�$H�\$XH�\$�H�L$H�D$H��$�H�� H�$H�L$HH�L$H�D$PH�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$H��$H���É%������l���H�\$8H�\$hH�\$@H�\$pH��$H�\$XH��$H�\$`H��$�1��H��$�H���H��H��H��$�H��$�H��$�H�H�$H�\$hH�\$�H�L$H�D$H��$�H�$H�L$HH�L$H�D$PH�D$�H�H�$H�\$XH�\$�H�L$H�D$H��$�H��H�$H�L$HH�L$H�D$PH�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$H��$H���É������E�2���8 +*0runtime.morestack_noctxt�*net/url.(*URL).String�go.string."/"�"strings.TrimRight� go.string."unix"� runtime.eqstring�� runtime.duffzero�type.string�runtime.convT2E�2runtime.writebarrieriface�$type."".APIVersion�runtime.convT2E�2runtime.writebarrieriface� type.string� runtime.convT2E� +2runtime.writebarrieriface� +(go.string."%s/v%s%s"� fmt.Sprintf� � runtime.duffzero� type.string�runtime.convT2E�2runtime.writebarrieriface�type.string�runtime.convT2E�2runtime.writebarrieriface� go.string."%s%s"�fmt.SprintfP�*"".autotmp_0415"type.interface {}"".autotmp_0414"type.interface {}"".autotmp_0412&type.[]interface {}"".autotmp_0411"type.interface {}"".autotmp_0410"type.interface {}"".autotmp_0409�"type.interface {}"".autotmp_0407�&type.[]interface {}"".autotmp_0406type.string"".autotmp_0405type.string"".autotmp_0404type.string"".autotmp_0403type.string"".autotmp_0402�(type.[2]interface {}"".autotmp_0401type.string"".autotmp_0400�type.string"".autotmp_0399�type.string"".autotmp_0398_(type.[3]interface {}"".autotmp_0396�type.string"".urlStr�type.string "".~r10type.string"".pathtype.string"".ctype.*"".Client("�������� (� +:d_�� (J�����DTgclocals·61dac2719f307a892a4a15123f2e6a2dTgclocals·514c3d378a44440bceb597de19ebfbf7�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�"".queryString��eH� %H��$����H;Aw���H��H��$�HDŽ$�HDŽ$�H��u HDŽ$�HDŽ$�H�Ĩ�H�$H��$�H�\$�H�T$H�L$H�D$ H��$�H�$H��$�H�L$H��$�H�D$�H��$�H��$�H��$�H�\$H���*H�,$H�T$H�L$�H�l$H�T$ H�L$(H��$�H�,$H��$�H�T$H��$�H�L$�H�\$H��t HDŽ$�HDŽ$�H�Ĩ�H�H�$H�D$�H�\$H��$�H�D$xH��$�H�$H��$�H�\$H��$�H�\$�H�D$H�\$xH9��H��$�H�$H��$�H�\$H��$�H�\$�H�T$H�D$ H�\$xH�\$H��$�H�$H��$�H�ZX��H�\$H��$@H��H���H��$@H��$�H��H���H��$�H��tH�\$xH��H�\$x����H��$H�,$H��H��H�H�H�H�l$H��H��H�H��H�L$ H�D$(H��$�H��$�H����H��$�H�,$H��H��H�H��H�L$H�D$H��$�H��$�H��$�H�$H��$�H�\$H��$�H�\$H�\$xH�\$�H�T$ H�L$(H�D$0H��$�H�$H��$�H�\$H��$�H�\$H��$�H�T$H��$�H�L$ H��$�H�D$(������H���[���H� $H�D$H�-L�D$L��H��H�H���\$ ��������$���H��$�H�$�H�L$H�D$H��$�H��$�H�Ĩ������* +00runtime.morestack_noctxt�reflect.ValueOf�$reflect.Value.Kind�$reflect.Value.Elem�$reflect.Value.Kind�0type.map[string][]string�runtime.makemap�,reflect.Value.NumField�$reflect.Value.Type� +�� runtime.duffcopy� � runtime.duffcopy� +go.string."qs"� +*reflect.StructTag.Get� strings.ToLower� &reflect.Value.Field�,"".addQueryStringValue�go.string."-"� runtime.eqstring�*net/url.Values.Encode@� "".autotmp_0433type.string"".autotmp_0431$type.reflect.Value"".autotmp_0430type.string"".autotmp_0428�0type.reflect.StructField"".autotmp_0427�"type.reflect.Type"".autotmp_0426type.int"".autotmp_0425"type.reflect.Kind"".autotmp_0424$type.reflect.Value"".autotmp_0422�$type.reflect.Value "".key�type.string"".field�0type.reflect.StructField"".i�type.int"".items�&type.net/url.Values"".value�$type.reflect.Value "".~r1 type.string"".opts"type.interface {}4%�E���������`� +E %R"6 &G� +C5�<3!6|:::��?Oa3$;Tgclocals·c69849cba6bf70a13b3371331d258b50Tgclocals·928ad969a3698656dfa33d91e2ca9cd1�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�,"".addQueryStringValue�@�@eH� %H��$P���H;Aw���H��0H��$PH�$H��$XH�\$H��$`H�\$�H��$PH��$XH��$`H�D$H����H���]H����H�,$H�T$H�L$��\$����H��$8H��$@H��$HH�H�+H��$�H�kH��$�H��$�H��$�H��$H��$H�H�$H�T$xH�T$H��$�H�L$H��$�H�D$�H�\$ H���8H�H�KH�[H��$H��$ H��$(H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H��$�H��Hk�H�H�$H��$�H�\$H��$�H�\$�H��$�H��$H��$�H��$H��$�H��$H�H�$H�\$xH�\$H��$�H�\$H��$H�\$�H��0É�������H���?H�,$H�T$H�L$�H�\$H���H��$8H�\$hH��$PH�$H��$XH�\$H��$`H�\$�H�\$H�$H�D$ +�H�l$H�T$H��$@H��$HH��$�H��$�H��$�H��$�H��$�H��$�H�H�$H�\$hH�\$H��$�H�L$H��$�H�D$�H�\$ H���5H�H�KH�[H��$H��$ H��$(H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H��$�H��Hk�H�H�$H��$�H�\$H��$�H�\$�H��$�H��$H��$�H��$H��$�H��$H�H�$H�\$hH�\$H��$�H�\$H��$H�\$���������������H�����������H�������H�������H��������t���H���QH�� �TH�,$H�T$H�L$��D$� f.�w�4���H��$8H�\$PH��$PH�$H��$XH�\$H��$`H�\$��D$�$�D$fH�D$����H�D$@�H�l$ H�T$(H��$@H��$HH��$�H��$�H��$�H��$�H��$H��$H�H�$H�\$PH�\$H��$�H�L$H��$�H�D$�H�\$ H���5H�H�KH�[H��$H��$ H��$(H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��H��$�H��$�H��$�H��Hk�H�H�$H��$�H�\$H��$�H�\$�H��$�H��$H��$�H��$H��$�H��$H�H�$H�\$PH�\$H��$�H�\$H��$H�\$�����������H�������H�������H�,$H�T$H�L$�H�\$H�\$@H�\$@H�������1�H�l$@H9������H��$PH�$H��$XH�\$H��$`H�\$H�D$HH�D$�H�T$ H�L$(H�D$0H��$8H�$H��$@H�\$H��$HH�\$H��$�H�T$H��$�H�L$ H��$�H�D$(�H�D$HH���O���H���yH����H�,$H�T$H�L$�H�\$H�D$ H�\$(H����H��$PH�$H��$XH�\$H��$`H�\$�H�L$H�D$ H��$`H� $H��$hH�D$�H�l$H�T$H�L$ H�D$(H�\$0H��$XH��H��$P�S���H��$8H�\$pH��$�H�,$H��$�H�T$H��$�H�L$�H�l$H�T$ H��$@H��$HH��$�H��$�H��$�H��$�H��$0H��$8H�H�$H�\$pH�\$H��$pH�L$H��$xH�D$�H�\$ H���5H�H�KH�[H��$H��$ H��$(H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H��$�H��Hk�H�H�$H��$�H�\$H��$�H�\$�H��$�H��$H��$�H��$H��$�H��$H�H�$H�\$pH�\$H��$�H�\$H��$H�\$��Y����������M���H���C���H�,$H�T$H�L$��\$����H��$PH�$H��$XH�\$H��$`H�\$�H�L$H�D$ H��$`H� $H��$hH�D$�H�l$H�T$H�L$ H�D$(H�\$0H��$HH��H��$@�����H��$8H�\$XH��$�H�,$H��$�H�T$H��$�H�L$�H�l$H�T$ H��$@H��$HH��$�H��$�H��$�H��$�H��$ H��$(H�H�$H�\$XH�\$H��$pH�L$H��$xH�D$�H�\$ H���5H�H�KH�[H��$H��$ H��$(H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H��$�H��Hk�H�H�$H��$�H�\$H��$�H�\$�H��$�H��$H��$�H��$H��$�H��$H�H�$H�\$XH�\$H��$�H�\$H��$H�\$���������������H�������H�������H�,$H�T$H�L$�H�\$H�\$ H���[���H��$8H�\$`H��$PH�$H��$XH�\$H��$`H�\$�H�l$H�T$ H��$@H��$HH��$�H��$�H��$�H��$�H��$�H��$�H�H�$H�\$`H�\$H��$pH�L$H��$xH�D$�H�\$ H���5H�H�KH�[H��$H��$ H��$(H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H��$�H��Hk�H�H�$H��$�H�\$H��$�H�\$�H��$�H��$H��$�H��$H��$�H��$H�H�$H�\$`H�\$H��$�H�\$H��$H�\$��a���������� +00runtime.morestack_noctxt�$reflect.Value.Kind�$reflect.Value.Bool�go.string."1"�&type.net/url.Values�4runtime.mapaccess1_faststr�type.[]string�"runtime.growslice�4runtime.writebarrierstring� &type.net/url.Values� $runtime.mapassign1� +"reflect.Value.Int� "reflect.Value.Int� "strconv.FormatInt� &type.net/url.Values� 4runtime.mapaccess1_faststr�type.[]string�"runtime.growslice�4runtime.writebarrierstring�&type.net/url.Values�$runtime.mapassign1�&reflect.Value.Float�*$f64.0000000000000000�&reflect.Value.Float�&strconv.FormatFloat�&type.net/url.Values�4runtime.mapaccess1_faststr�type.[]string�"runtime.growslice�4runtime.writebarrierstring�&type.net/url.Values�$runtime.mapassign1�"reflect.Value.Len�&reflect.Value.Index�!,"".addQueryStringValue�!*reflect.Value.MapKeys�#.reflect.Value.Interface�#*encoding/json.Marshal�%2runtime.slicebytetostring�&&type.net/url.Values�'4runtime.mapaccess1_faststr�(type.[]string�)"runtime.growslice�*4runtime.writebarrierstring�+&type.net/url.Values�,$runtime.mapassign1�-&reflect.Value.IsNil�-.reflect.Value.Interface�.*encoding/json.Marshal�02runtime.slicebytetostring�1&type.net/url.Values�24runtime.mapaccess1_faststr�3type.[]string�4"runtime.growslice�54runtime.writebarrierstring�6&type.net/url.Values�7$runtime.mapassign1�8(reflect.Value.String�9(reflect.Value.String�:&type.net/url.Values�;4runtime.mapaccess1_faststr�<type.[]string�="runtime.growslice�>4runtime.writebarrierstring�?&type.net/url.Values�@$runtime.mapassign1`��"".autotmp_0509type.uint64"".autotmp_0508type.uint64"".autotmp_0507type.int"".autotmp_0506type.int"".autotmp_0505type.[]string"".autotmp_0504type.string"".autotmp_0503type.uint64"".autotmp_0502type.uint64"".autotmp_0501type.int"".autotmp_0500type.int"".autotmp_0499type.[]string"".autotmp_0498type.string"".autotmp_0497type.uint64"".autotmp_0496type.uint64"".autotmp_0495type.int"".autotmp_0494type.int"".autotmp_0493type.[]string"".autotmp_0492type.uint64"".autotmp_0491type.uint64"".autotmp_0490type.int"".autotmp_0489type.int"".autotmp_0488type.[]string"".autotmp_0487type.uint64"".autotmp_0486type.uint64"".autotmp_0485type.int"".autotmp_0484type.int"".autotmp_0483type.[]string"".autotmp_0478�type.[]string"".autotmp_0476type.int"".autotmp_0475�$type.reflect.Value"".autotmp_0474type.int"".autotmp_0473type.[]string"".autotmp_0472type.[]string"".autotmp_0471type.string"".autotmp_0470type.string"".autotmp_0469"type.interface {}"".autotmp_0468type.int"".autotmp_0466type.[]string"".autotmp_0465type.[]string"".autotmp_0464type.string"".autotmp_0463type.string"".autotmp_0462�"type.interface {}"".autotmp_0461type.bool"".autotmp_0460type.[]string"".autotmp_0459type.[]string"".autotmp_0458type.string"".autotmp_0457type.string"".autotmp_0456�type.string"".autotmp_0455type.string"".autotmp_0454type.[]string"".autotmp_0453type.[]string"".autotmp_0452type.string"".autotmp_0451type.string"".autotmp_0450type.string"".autotmp_0447type.[]string"".autotmp_0446type.[]string"".autotmp_0445type.string"".autotmp_0444type.string"".autotmp_0443type.string"".autotmp_0440_type.[]string"".autotmp_0439/type.[]string"".autotmp_0438�type.string"".autotmp_0437�type.string net/url.value·3�type.stringnet/url.key·2�type.stringnet/url.v·1�&type.net/url.Values net/url.value·3�type.stringnet/url.key·2�type.stringnet/url.v·1�&type.net/url.Values net/url.value·3�type.stringnet/url.key·2�type.stringnet/url.v·1�&type.net/url.Values net/url.value·3�type.stringnet/url.key·2�type.stringnet/url.v·1�&type.net/url.Values net/url.value·3�type.stringnet/url.key·2�type.stringnet/url.v·1�&type.net/url.Values net/url.value·3�type.stringnet/url.key·2�type.stringnet/url.v·1�&type.net/url.Values"".i�type.int"".vLen�type.int "".err�type.error"".b�type.[]uint8 "".err�type.error"".b�type.[]uint8"".v0$type.reflect.Value "".keytype.string"".items&type.net/url.Values%������ �� +%\ +!�DC + +"�   + +  +  +'� +( +� ? +. +,��343" +!��'(': + +'� �K��d.G��djQ��d 3�~�d.�~�d8L ~�dTgclocals·b8c550e5e1ba1f11f1bc237b9d0f0dc8Tgclocals·a157d5303e3a20a2392145ef25e4599b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�"".newError��eH� %H;aw���H��0H�H�$�H�D$H�D$(H�l$8H�(H�\$@H�$H�\$HH�\$H�\$PH�\$�H�\$H�l$H��H��H�H�H�\$(H�$H�<$tH�$�H�\$(H�\$XH��0É%�� + 0runtime.morestack_noctxt:type."".ErrorL"runtime.newobject�2runtime.slicebytetostring�4runtime.writebarrierstringP`"".autotmp_0528type.*"".Error "".~r2@type.*"".Error"".bodytype.[]uint8"".statustype.int`�_`�� �%4WTgclocals·3e69739b44630d52358b28c7a0e238faTgclocals·e1ae6533a9e39048ba0735a2264ce16a�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�""".(*Error).Error��eH� %H;aw���H��HDŽ$�HDŽ$�H�|$`1��H�\$`H���CH��H��H�\$HH�T$PH�L$XH�H�$H��$�H�\$H�|$���H�L$H�D$H�\$HH�$H�L$8H�L$H�D$@H�D$�H�H�$H��$�H�\$H�|$��H�D$�H�L$H�D$H�\$HH��H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H�H�\$HH�\$H�\$PH�\$H�\$XH�\$ �H�L$(H�D$0H��$�H��$�H�ĀÉ%�_����%���������� + 0runtime.morestack_noctxtz� runtime.duffzero�type.int�runtime.convT2E�2runtime.writebarrieriface�type.string�runtime.convT2E�2runtime.writebarrieriface�Tgclocals·13d3af77a5bf02af6db4588efb2ea811Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�"".Port.Proto��eH� %H;aw���H��8H�D$PH�D$XH�\$@H�$H�t$HH�t$H�5H�l$H��H�H��H�T$ H�D$(H�L$0H��uH�H�+H�l$PH�kH�l$XH��8�H��H��vH��H�+H�l$PH�kH�l$XH��8�� + 0runtime.morestack_noctxt�go.string."/"�strings.Split�go.string."tcp"�$runtime.panicindex@p "".~r0 type.string"".ptype."".Portpnop"op��,:7 +RnTgclocals·13d3af77a5bf02af6db4588efb2ea811Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�$"".(*State).String� � eH� %H�D$�H;Aw���H��H��$�HDŽ$�HDŽ$�������^��t&H�H�+H��$�H�kH��$�H�Ĩ��H�$�L$H�D$H�H�T$x��$�H��$�H�T$`H�$�L$h�L$H�D$pH�D$H��$�H���H�o(H�|$H��H�H�H��H�\$0H�\$8H�\$PH�H�CH�\$PH����H��H��H��$�H��$�H��$�H�H�$H�\$8H�\$�H�L$H�D$H��$�H�$H�L$@H�L$H�D$HH�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$�H��$�H�ĨÉ�'���������H�\$PH�H�CH�\$PH����H��H��H��$�H��$�H��$�H�H�$H�t$H�|$��H�D$�H�L$H�D$H��$�H�$H�L$@H�L$H�D$HH�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$�H��$�H�ĨÉ%�W�������� +*0runtime.morestack_noctxt�$go.string."paused"�time.Now�time.UTC�time.Time.Sub�$type.time.Duration�runtime.convT2E�2runtime.writebarrieriface�"go.string."Up %s"�fmt.Sprintf� type.int� runtime.convT2E� +2runtime.writebarrieriface� +&go.string."Exit %d"� fmt.Sprintf0�"".autotmp_0599"type.interface {}"".autotmp_0598*type.*[1]interface {}"".autotmp_0597&type.[]interface {}"".autotmp_0596�"type.interface {}"".autotmp_0594/&type.[]interface {}"".autotmp_0593type.string"".autotmp_0592(type.[1]interface {}"".autotmp_0590�$type.time.Duration"".autotmp_0588�(type.[1]interface {} "".~r0�type.time.Timetime.t·2_type.time.Time "".~r0type.string"".stype.*"".State4"�Z��������� �B  &��}�p�p:Tgclocals·6d340c3bdac448a6ef1256f331f68dd3Tgclocals·f56fbe9e0c86cdefdb59a7a88f742314�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�H"".(*NetworkSettings).PortMappingAPI��eH� %H��$����H;Aw���H���1�H��$0�HDŽ$�HDŽ$�HDŽ$HDŽ$�HDŽ$�HDŽ$�H��$�H�kPH��$�1��H�H�$H�l$H��$�H�\$�H��$�1�H9��4H��$�H����H�;H�sH�SH��$�H����H� H�kH��$�H��$�H��$�H��$�H��$�H��$�H�L$pH� $H�l$xH�l$�H�L$H�D$H��$�H� $H��$�H�D$�H��$�H�\$H�\$HH����H�\$pH�$H�\$xH�\$�H�T$H�L$H��$`1��H�\$HH��$hH��$�H��$pH��$�H��$xH��$�H��$�H��$�H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H�H�$H��H��$�H��Hk�0H�H�\$H��$`H�\$�H��$�H��$�H��$�H��$�H��$�H��$�H��$�H�$�H��$�1�H9������H��$�H��$�H��$�H��$�H��$�H��$H����H��$�H��$�H��$�1�H��$�H�L$XH��$�H�l$XH9��g���H�D$hH���'H��$H��H���H�T$`H��$H��$�H��H���H�\$pH�$H�\$xH�\$�H�L$H�D$H��$�H� $H��$�H�D$�H�t$H�t$@H��$H�,$H��H�H��H�\$H�\$PH�\$pH�$H�\$xH�\$�H�T$H�L$H��$01��H�\$@H��$0H�t$PH��$8H��$�H��$@H��$�H��$HH��$�H��$PH��H�H�H��$�H��$�H��$�H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H�H�$H��H��$�H��Hk�0H�H�\$H��$0H�\$�H��$�H��$�H��$�H��$�H��$�H��$�H�D$hH�T$`H�� H��������������B��������4 +00runtime.morestack_noctxt`� runtime.duffzero�� runtime.duffzero�Btype.map["".Port][]"".PortBinding�&runtime.mapiterinit�"".Port.Port�"".parsePort�"".Port.Proto�� runtime.duffzero�"type.[]"".APIPort� "runtime.growslice� type."".APIPort� +.runtime.writebarrierfat� &runtime.mapiternext�� runtime.duffcopy�� runtime.duffcopy�"".Port.Port�"".parsePort�"".parsePort�"".Port.Proto�� runtime.duffzero�"type.[]"".APIPort�"runtime.growslice�type."".APIPort�.runtime.writebarrierfat@�<"".autotmp_0632type.uint64"".autotmp_0631type.uint64"".autotmp_0630type.int"".autotmp_0629type.int"".autotmp_0628"type.[]"".APIPort"".autotmp_0627�type."".APIPort"".autotmp_0626�&type."".PortBinding"".autotmp_0625�(type.*"".PortBinding"".autotmp_0624type.int"".autotmp_0623type.int"".autotmp_0619�type.int"".autotmp_0618�"type.[]"".APIPort"".autotmp_0617�type."".APIPort"".autotmp_0616�*type.[]"".PortBinding"".autotmp_0615"type.[]"".APIPort"".autotmp_0614type.string"".autotmp_0613type.string"".autotmp_0612*type.[]"".PortBinding"".autotmp_0610type.string"".autotmp_0608�type.string"".autotmp_0607�Ltype.map.iter["".Port][]"".PortBinding"".h�type.int"".p�type.int"".binding�&type."".PortBinding"".p�type.int"".bindings�*type.[]"".PortBinding"".port�type."".Port"".mapping�"type.[]"".APIPort "".~r0"type.[]"".APIPort"".settings0type.*"".NetworkSettings%������ J�X$�R +"� $(8J""�4��\�\A��\`Tgclocals·8ba904616303767b538615d50f9b7d50Tgclocals·61adfc392b1c041e58d54f9101a6f0b7�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�"".parsePort��eH� %H;aw���H��8H�D$XH�D$`H�\$@H�$H�\$HH�\$H�D$ +H�D$�H�T$ H�D$(H�L$0H��tH�D$PH�D$XH�L$`H��8�H�T$PH�D$XH�D$`H��8� + 0runtime.morestack_noctxt�"strconv.ParseUintPp "".~r20type.error "".~r1 type.int"".rawPorttype.stringphopo��,9 +QOTgclocals·5dfce38b1d248a3900c6ec750de77702Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�8"".(*Client).RenameContainer� � eH� %H�D$�H;Aw���H��HDŽ$0HDŽ$8H��$H��$�H��H���H�H�$H��$�H�\$�H�\$H�,$H��H��H�H��H�\$H��$�H�\$H��$�H��$�H�H�CH��$�H����H��H��H��$�H��$�H��$�H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H�L$pH�L$H�D$xH�D$�H�H�,$H��H��H�H�H��$H�l$H��H��H�H�H�H�l$ H��H��H�H��H�\$0H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$�H�H�CH�CH��$H�4$H�5H�l$H��H�H�H��$�H�L$H��$�H�D$ H��$�H�l$(H��H��H�H�H��H�L$`H�D$hH��$0H��$8H��É�_��� +*0runtime.morestack_noctxt�� runtime.duffcopy�go.string."/containers/create?"�*runtime.concatstring2��type.struct { *"".Config; HostConfig *"".HostConfig "json:\"HostConfig,omitempty\" yaml:\"HostConfig,omitempty\"" }�runtime.convT2E� go.string."POST"�"".(*Client).do�""".ErrNoSuchImage�""".ErrNoSuchImage� 8"".ErrContainerAlreadyExists� 8"".ErrContainerAlreadyExists� +"type."".Container� +"runtime.newobject� $type.*"".Container� .encoding/json.Unmarshal� 4runtime.writebarrierstring��"".autotmp_0673o"type."".doOptions"".autotmp_0670��type.struct { *"".Config; HostConfig *"".HostConfig "json:\"HostConfig,omitempty\" yaml:\"HostConfig,omitempty\"" }"".autotmp_0669�type.string"".autotmp_0668?22$c$*: p�h�V�Tgclocals·3f5a7d1842b14039f35be09ef67df5f8Tgclocals·2ca41a02a2a5788f97a4be1897b36700�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go� "".AlwaysRestart��H��H�\$ H�H�CH�CH�$H�H�CH�CH�$H�-H��H��H�H�H�$H�l$ H��H��H�H�H�H����$go.string."always"00"".autotmp_0676/*type."".RestartPolicy "".~r0*type."".RestartPolicy0e/p � PTgclocals·0528ab8f76149a707fd2f0025c2178a3Tgclocals·0528ab8f76149a707fd2f0025c2178a3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�&"".RestartOnFailure��H��H�\$(H�H�CH�CH�$H�H�CH�CH�$H�-H��H��H�H�H�t$ H�t$H�4$H�l$(H��H�H�H�H����,go.string."on-failure"@0"".autotmp_0677/*type."".RestartPolicy "".~r1*type."".RestartPolicy"".maxRetrytype.int0l/� � `Tgclocals·2d8f3a7439ca173dec4205ff264b0edcTgclocals·0528ab8f76149a707fd2f0025c2178a3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�"".NeverRestart��H��H�\$ H�H�CH�CH�$H�H�CH�CH�$H�-H��H��H�H�H�$H�l$ H��H��H�H�H�H����go.string."no"00"".autotmp_0678/*type."".RestartPolicy "".~r0*type."".RestartPolicy0e/p � PTgclocals·0528ab8f76149a707fd2f0025c2178a3Tgclocals·0528ab8f76149a707fd2f0025c2178a3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�6"".(*Client).StartContainer� � eH� %H�D$�H;Aw���H���HDŽ$�HDŽ$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�t$H�5H�l$ H��H�H��L�D$0H�T$8H��$�H��$�H�H�CH�CH�H��$�H��$�H��$�H��$�H����$�H��$�H�4$H�5H�l$H��H�H�L��$�L�D$H��$�H�T$ H��$�H�l$(H��H��H�H�H��H�D$XH�L$`H��$�H�T$hH��$�H=���H�H�$�H�D$H�D$xH�$H�<$��H��$�H�\$H��$�H�\$�H�\$xH�$H�<$��H�$H��$�H�\$H��$�H�\$�H�\$xH�\$xH�1�H9�tH�\$xH��$�H��$�H����H�H�$H�H�\$H�H�\$�H�D$봉%�l����%�-���H=0��H�H�$�H�D$H�D$pH�$H�<$��H��$�H�\$H��$�H�\$�H�\$pH�\$pH�1�H9�tH�\$pH��$�H��$�H����H�H�$H�H�\$H�H�\$�H�D$봉%�q���H��tH��$�H��$�H����HDŽ$�HDŽ$�H����2 +*0runtime.morestack_noctxtz0go.string."/containers/"�$go.string."/start"�*runtime.concatstring3�&type.*"".HostConfig� go.string."POST"�"".(*Client).do�.type."".NoSuchContainer�"runtime.newobject�4runtime.writebarrierstring�2runtime.writebarrieriface�Bgo.itab.*"".NoSuchContainer.error�0type.*"".NoSuchContainer�type.error�Bgo.itab.*"".NoSuchContainer.error�  runtime.typ2Itab� >type."".ContainerAlreadyRunning� "runtime.newobject� +4runtime.writebarrierstring� Rgo.itab.*"".ContainerAlreadyRunning.error� @type.*"".ContainerAlreadyRunning� type.error� Rgo.itab.*"".ContainerAlreadyRunning.error�  runtime.typ2Itab`�"".autotmp_0686type.*uint8"".autotmp_0685�@type.*"".ContainerAlreadyRunning"".autotmp_0683�0type.*"".NoSuchContainer"".autotmp_0682/"type."".doOptions"".autotmp_0681@type.*"".ContainerAlreadyRunning"".autotmp_06800type.*"".NoSuchContainer "".errotype.error"".path�type.string "".~r2@type.error"".hostConfig0&type.*"".HostConfig +"".idtype.string"".ctype.*"".Client:"�������X��� +�2�:Q� � �)$|�88�8�Tgclocals·fc96ae191c2547955912928601e85959Tgclocals·1497b0fbec88b963d1dc5f4cf9421516�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�4"".(*Client).StopContainer��eH� %H��$h���H;Aw���H��HDŽ$@HDŽ$HH��$(H��$�H��$0H��$�H��$8H�\$pH��$�1��H��$�H����H��H��H��$�H��$�H��$�H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H�H�$H�\$pH�\$�H�L$H�D$H��$�H��H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$�H�H�CH�CH��$ H�4$H�5H�l$H��H�H�H��$�H�L$H��$�H�D$ H��$�H�l$(H��H��H�H�H��H�D$XH�L$`H��$�H�T$hH��$�H=���H�H�$�H�L$H��H����1��H��$�H� $H�<$��H��$(H�\$H��$0H�\$�H��$�H��$�H�1�H9�t H��$�H��$HH��$@H���H�H�$H�H�\$H�H�\$�H�D$뱉%�h�����C���H=0��H�H�$�H�D$H�D$xH�$H�<$��H��$(H�\$H��$0H�\$�H�\$xH�\$xH�1�H9�tH�\$xH��$HH��$@H���H�H�$H�H�\$H�H�\$�H�D$봉%�q���H��tH��$@H��$HH���HDŽ$@HDŽ$HH��É�a���< +00runtime.morestack_noctxt�� runtime.duffzero�type.string�runtime.convT2E�2runtime.writebarrieriface�type.uint�runtime.convT2E�2runtime.writebarrieriface�Hgo.string."/containers/%s/stop?t=%d"�fmt.Sprintf� go.string."POST"�"".(*Client).do� .type."".NoSuchContainer� "runtime.newobject� � runtime.duffzero� +4runtime.writebarrierstring� +Bgo.itab.*"".NoSuchContainer.error� 0type.*"".NoSuchContainer� type.error� Bgo.itab.*"".NoSuchContainer.error�  runtime.typ2Itab� 6type."".ContainerNotRunning� "runtime.newobject� 4runtime.writebarrierstring�Jgo.itab.*"".ContainerNotRunning.error�8type.*"".ContainerNotRunning�type.error�Jgo.itab.*"".ContainerNotRunning.error� runtime.typ2Itab`�$"".autotmp_0705type.*uint8"".autotmp_0704�8type.*"".ContainerNotRunning"".autotmp_0702�0type.*"".NoSuchContainer"".autotmp_0701�"type."".doOptions"".autotmp_0700"type.interface {}"".autotmp_0699�"type.interface {}"".autotmp_0697o&type.[]interface {}"".autotmp_06968type.*"".ContainerNotRunning"".autotmp_06950type.*"".NoSuchContainer"".autotmp_0693�type.uint"".autotmp_0692�type.string"".autotmp_0691?(type.[2]interface {} "".err�type.error"".path�type.string "".~r2@type.error"".timeout0type.uint +"".idtype.string"".ctype.*"".Client@%�������X���� �8�=�� � �  *�����8�Tgclocals·42785a4ae44025160cf24924f7d01efbTgclocals·5c42a9dee0c88889a167a0d13b7c2026�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�:"".(*Client).RestartContainer� � eH� %H��$p���H;Aw���H��HDŽ$8HDŽ$@H��$ H��$�H��$(H��$�H��$0H�\$pH��$�1��H��$�H����H��H��H��$�H��$�H��$�H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H�H�$H�\$pH�\$�H�L$H�D$H��$�H��H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$�H�H�CH�CH��$H�4$H�5H�l$H��H�H�H��$�H�L$H��$�H�D$ H��$�H�l$(H��H��H�H�H��H�L$XH�D$`H��$�H�T$hH��$�H�����H�H�$�H�L$H��H����1��H�L$xH� $H�<$��H��$ H�\$H��$(H�\$�H�\$xH�\$xH�1�H9�tH�\$xH��$@H��$8H���H�H�$H�H�\$H�H�\$�H�D$봉%�q�����O���H��tH��$8H��$@H���HDŽ$8HDŽ$@H��É�0���, +00runtime.morestack_noctxt�� runtime.duffzero�type.string�runtime.convT2E�2runtime.writebarrieriface�type.uint�runtime.convT2E�2runtime.writebarrieriface�Ngo.string."/containers/%s/restart?t=%d"�fmt.Sprintf� go.string."POST"�"".(*Client).do� .type."".NoSuchContainer� "runtime.newobject� � runtime.duffzero� +4runtime.writebarrierstring� +Bgo.itab.*"".NoSuchContainer.error� 0type.*"".NoSuchContainer� type.error� Bgo.itab.*"".NoSuchContainer.error�  runtime.typ2Itab`�"".autotmp_0721�0type.*"".NoSuchContainer"".autotmp_0720�"type."".doOptions"".autotmp_0719"type.interface {}"".autotmp_0718�"type.interface {}"".autotmp_0716o&type.[]interface {}"".autotmp_07150type.*"".NoSuchContainer"".autotmp_0713�type.uint"".autotmp_0712�type.string"".autotmp_0711?(type.[2]interface {} "".err�type.error"".path�type.string "".~r2@type.error"".timeout0type.uint +"".idtype.string"".ctype.*"".Client2%����_���� �.�=�� �   �����Tgclocals·1da38d5d89527cd2ab312249704d85d7Tgclocals·5bacfca50b7e6b4494d1c0d96e8b2c7c�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�6"".(*Client).PauseContainer� � eH� %H�D$�H;Aw���H���HDŽ$HDŽ$ H��$H��$�H��$H��$�H��$�H�H�CH��$�H���sH��H��H��$�H��$�H��$�H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$�H�H�CH�CH��$H�4$H�5H�l$H��H�H�H�L$xH�L$H��$�H�D$ H��$�H�l$(H��H��H�H�H��H�L$XH�D$`H��$�H�T$hH��$�H�����H�H�$�H�L$H��H����1��H�L$pH� $H�<$��H��$H�\$H��$H�\$�H�\$pH�\$pH�1�H9�tH�\$pH��$ H��$H����H�H�$H�H�\$H�H�\$�H�D$봉%�q�����O���H��tH��$H��$ H����HDŽ$HDŽ$ H���É����$ +*0runtime.morestack_noctxt�type.string�runtime.convT2E�2runtime.writebarrieriface�@go.string."/containers/%s/pause"�fmt.Sprintf� go.string."POST"�"".(*Client).do�.type."".NoSuchContainer�"runtime.newobject�� runtime.duffzero�4runtime.writebarrierstring� Bgo.itab.*"".NoSuchContainer.error� 0type.*"".NoSuchContainer� +type.error� +Bgo.itab.*"".NoSuchContainer.error� + runtime.typ2ItabP�"".autotmp_0735�0type.*"".NoSuchContainer"".autotmp_0734_"type."".doOptions"".autotmp_0733�"type.interface {}"".autotmp_0731/&type.[]interface {}"".autotmp_07300type.*"".NoSuchContainer"".autotmp_0728�type.string"".autotmp_0727(type.[1]interface {} "".err�type.error"".path�type.string "".~r10type.error +"".idtype.string"".ctype.*"".Client2"����_���� +�.�:�� �  +�v}��Tgclocals·fe0d626f6a1a9cb0d3493cb8c292091bTgclocals·556e2b84f9ef2d507be121d828e30b96�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�:"".(*Client).UnpauseContainer� � eH� %H�D$�H;Aw���H���HDŽ$HDŽ$ H��$H��$�H��$H��$�H��$�H�H�CH��$�H���sH��H��H��$�H��$�H��$�H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$�H�H�CH�CH��$H�4$H�5H�l$H��H�H�H�L$xH�L$H��$�H�D$ H��$�H�l$(H��H��H�H�H��H�L$XH�D$`H��$�H�T$hH��$�H�����H�H�$�H�L$H��H����1��H�L$pH� $H�<$��H��$H�\$H��$H�\$�H�\$pH�\$pH�1�H9�tH�\$pH��$ H��$H����H�H�$H�H�\$H�H�\$�H�D$봉%�q�����O���H��tH��$H��$ H����HDŽ$HDŽ$ H���É����$ +*0runtime.morestack_noctxt�type.string�runtime.convT2E�2runtime.writebarrieriface�Dgo.string."/containers/%s/unpause"�fmt.Sprintf� go.string."POST"�"".(*Client).do�.type."".NoSuchContainer�"runtime.newobject�� runtime.duffzero�4runtime.writebarrierstring� Bgo.itab.*"".NoSuchContainer.error� 0type.*"".NoSuchContainer� +type.error� +Bgo.itab.*"".NoSuchContainer.error� + runtime.typ2ItabP�"".autotmp_0749�0type.*"".NoSuchContainer"".autotmp_0748_"type."".doOptions"".autotmp_0747�"type.interface {}"".autotmp_0745/&type.[]interface {}"".autotmp_07440type.*"".NoSuchContainer"".autotmp_0742�type.string"".autotmp_0741(type.[1]interface {} "".err�type.error"".path�type.string "".~r10type.error +"".idtype.string"".ctype.*"".Client2"����_���� +�.�:�� �  +�v}��Tgclocals·fe0d626f6a1a9cb0d3493cb8c292091bTgclocals·556e2b84f9ef2d507be121d828e30b96�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�2"".(*Client).TopContainer��eH� %H��$(���H;Aw���H��XH��$�1��HDŽ$�HDŽ$�HDŽ$�HDŽ$�H�H�$�H��$�H�\$H�\$xH���H��$xH��$�H��$�H��$�H�H�CH��$�H����H��H��H��$ H��$(H��$0H�H�$H��$�H�\$�H�T$H�D$H��$ H�$H��$�H�T$H��$�H�D$�H�H�,$H��H��H�H�H��$ H�\$H��$(H�\$H��$0H�\$ �H�T$(H�D$0H��$�H��$�H��$hH��$�H��$pH��$�H��$�H��$�H��$�H��$�H��$81��H��$8H����H��H��H��$ H��$(H��$0H�H�$H��$�H�\$�H�T$H�D$H��$ H�$H��$�H�T$H��$�H�D$�H�H�$H��$�H�\$�H�T$H�D$H��$ H��H�$H��$�H�T$H��$�H�D$�H�H�,$H��H��H�H�H��$ H�\$H��$(H�\$H��$0H�\$ �H�T$(H�D$0H��$H�H�CH�CH��$`H�4$H�5H�l$H��H�H�H��$�H�T$H��$�H�D$ H��$H�l$(H��H��H�H�H��L�L$xH�|$@H��$�H�t$HH��$�H�l$PH��$H�T$XH�D$`H��$�L�D$hL��$�H�����H�H�$�H�T$H��H����1��H�T$pH�$H�<$��H��$hH�\$H��$pH�\$�H�\$pH�\$pH�1�H9�t2H�t$xH��$�H���H�\$pH��$�H��$�H��X�H�H�$H�H�\$H�H�\$�H�D$량%�\�����:���H��t+H��$�H��L���H��$�L��$�H��X�H�<$H�t$H�l$H�H��$�H�T$L��$�L�L$ �H�l$xH�D$(H�T$0H��t+H��H��$�H���H��$�H��$�H��X�H��H��$�H���HDŽ$�HDŽ$�H��XÉ�R��������H +00runtime.morestack_noctxt`� runtime.duffzero�"type."".TopResult�"runtime.newobject�type.string�runtime.convT2E�2runtime.writebarrieriface�.go.string."?ps_args=%s"�fmt.Sprintf�� runtime.duffzero�type.string� runtime.convT2E� 2runtime.writebarrieriface� +type.string� +runtime.convT2E� 2runtime.writebarrieriface� @go.string."/containers/%s/top%s"� fmt.Sprintf� go.string."GET"�"".(*Client).do�.type."".NoSuchContainer�"runtime.newobject�� runtime.duffzero�4runtime.writebarrierstring�Bgo.itab.*"".NoSuchContainer.error�� runtime.duffcopy�0type.*"".NoSuchContainer�type.error�Bgo.itab.*"".NoSuchContainer.error� runtime.typ2Itab�� runtime.duffcopy�$type.*"".TopResult�.encoding/json.Unmarshal�� runtime.duffcopy�� runtime.duffcopy��0"".autotmp_0773�0type.*"".NoSuchContainer"".autotmp_0772�"type."".doOptions"".autotmp_0771"type.interface {}"".autotmp_0770"type.interface {}"".autotmp_0768&type.[]interface {}"".autotmp_0767�"type.interface {}"".autotmp_0765o&type.[]interface {}"".autotmp_07620type.*"".NoSuchContainer"".autotmp_0761type.string"".autotmp_0760�type.string"".autotmp_0759type.string"".autotmp_0758?(type.[2]interface {}"".autotmp_0756�type.string"".autotmp_0755�(type.[1]interface {}"".&result�$type.*"".TopResult "".err�type.error"".body�type.[]uint8"".path�type.string"".args�type.string "".~r3�type.error "".~r2P"type."".TopResult"".psArgs0type.string +"".idtype.string"".ctype.*"".Client>%����r��s��2��� P�L" +��� �+C+3 4o�v������Tgclocals·950e6e6b9e7c3fe47672289f0a6f6e8bTgclocals·6a9f496e2cfbe0515ededb4c2d64a743�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�$"".(*Client).Stats��eH� %H��$x���H;Aw���H��H�H�$�H�D$H��$�H��$H�(H�H�$�H�|$H��$�H��$H���HDŽ$HHDŽ$PHDŽ$HHDŽ$PH�H�$�H�\$H��$�H�H�$H�D$�H�D$H��$�H�$H�D$�H�H�$�H�\$H�\$xH�H�$�H�\$H�\$h�H�$H�\$H�\$PH�\$xH�$H�D$�H�\$hH�$H�\$PH�\$�H��$�1��H��$�H�-H�+H�\$HH�$H�<$��H�$H��$�H�\$�H�\$HH�$H�<$��H�$H��$�H�\$�H�L$HH��$HH�iH� $H�<$�xH�$ H�\$xH�\$�H�\$HSj�YYH���>H�H�$�H�L$H�-H�)H�L$@H� $H�<$��H�$H��$�H�\$�H�\$@H�$H�<$��H�$H��$�H�\$�H�\$@H�$H�<$��H�$H�\$hH�\$�H�\$@H�$H�<$�]H�$ H��$�H�\$�H�\$@Sj�YYH�H�$�H�\$H��$�H�H�$H�D$�H�L$H��$�H�$H�L$�H��$�H�+H�,$H� Qj�YYH����H�H�$�H�D$H�-H�(H�D$8H�$H�<$�pH�$H��$�H�\$�H�\$8H�$H�<$�9H�$H�\$xH�\$�H�\$8H�$H�<$�H�$H��$�H�\$�H�\$8Sj�YYH�\$xH�+H�l$XH�1�H9���H�L$XH��$�H��$�H��$�H��$�H�H�$�H�|$H��H���@1��H�L$0H� $H�<$�H��$�H�\$H��$�H�\$�H�\$0H�\$(H�H�$�H�\$H�\$pH�H�$�H�D$H�\$pH�$H�D$�H�L$pH�\$(H�$H�H��$�H�D$H��$�H�L$�H�T$H�L$ H��$�H�-H9���H��H��$�tH��$HH��$P��H���H�\$pH�+H�l$`H�H�$H��$�H�kH�l$H�\$`H�\$�H�H�$�H�D$H�\$pH�$H�D$�H�\$pH�+H�\$(H�$H�H��$�H�D$H��$�H�l$�H�T$H�L$ H��$�H�-H9�����H��$�H�$H�L$H�-H�l$H�-H�l$�H��$�H��$��\$ ��u�����HDŽ$HHDŽ$P��H��É%����������H�H�$H�H�\$H�H�\$�H�D$�@����%������%�����%������H��É%�����%�c����%�,����%�������H��É%�|����%�9����%����� +00runtime.morestack_noctxtPtype.*"".Clientb"runtime.newobject�(type."".StatsOptions�"runtime.newobject�� runtime.duffcopy�type.chan error�"runtime.newobject�type.chan error� runtime.makechan�.runtime.writebarrierptr�&type.*io.PipeReader�"runtime.newobject�&type.*io.PipeWriter�"runtime.newobject�io.Pipe�.runtime.writebarrierptr�.runtime.writebarrierptr�� runtime.duffzero�"".func·003�.runtime.writebarrierptr�.runtime.writebarrierptr�.runtime.writebarrierptr�"runtime.deferproc��type.struct { F uintptr; A0 **"".Client; A1 *"".StatsOptions; A2 **io.PipeWriter; A3 *chan error }�"runtime.newobject� "".func·004� .runtime.writebarrierptr� +.runtime.writebarrierptr� .runtime.writebarrierptr� .runtime.writebarrierptr� runtime.newproc� &type.chan struct {}� "runtime.newobject� &type.chan struct {}�  runtime.makechan� .runtime.writebarrierptr� (runtime.closechan·f� "runtime.deferproc��type.struct { F uintptr; A0 *"".StatsOptions; A1 **io.PipeReader; A2 *chan struct {} }�"runtime.newobject�"".func·005�.runtime.writebarrierptr�.runtime.writebarrierptr�.runtime.writebarrierptr�runtime.newproc�@go.itab.*io.PipeReader.io.Reader�4type.encoding/json.Decoder�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface�type.*"".Stats�"runtime.newobject�type."".Stats�"runtime.newobject�.runtime.writebarrierptr�type.**"".Stats�>encoding/json.(*Decoder).Decode� io.EOF�&runtime.deferreturn�*type.chan<- *"".Stats�"runtime.chansend1�type."".Stats�"runtime.newobject�.runtime.writebarrierptr�type.*"".Stats�>encoding/json.(*Decoder).Decode� io.EOF� io.EOF� io.EOF�runtime.ifaceeq�&runtime.deferreturn�&type.*io.PipeReader�type.io.Reader�@go.itab.*io.PipeReader.io.Reader� runtime.typ2Itab�&runtime.deferreturn�&runtime.deferreturn��2"".autotmp_0803�6type.*encoding/json.Decoder"".autotmp_08026type.*encoding/json.Decoder"".autotmp_0800otype.io.Reader"".autotmp_0799��type.*struct { F uintptr; A0 *"".StatsOptions; A1 **io.PipeReader; A2 *chan struct {} }"".autotmp_0798��type.*struct { F uintptr; A0 **"".Client; A1 *"".StatsOptions; A2 **io.PipeWriter; A3 *chan error }"".autotmp_0797O�type.struct { F uintptr; A0 *"".StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader }"".autotmp_0796��type.*struct { F uintptr; A0 *"".StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader }"".autotmp_0795�&type.*io.PipeWriter"".autotmp_0794�&type.*io.PipeReader"".autotmp_0793type.error"".autotmp_0792type.*"".Stats"".autotmp_0791type.*"".Stats"".autotmp_0790�type.*"".Stats"".autotmp_0786&type.*io.PipeReader"".&stats�type.**"".Stats"".&quit�(type.*chan struct {}"".&readCloser�(type.**io.PipeReader"".&errC� type.*chan error"".&writeCloser�(type.**io.PipeWriter"".&opts�*type.*"".StatsOptions +"".&c� type.**"".Client$encoding/json.r·2�type.io.Reader "".err�type.error"".decoder�6type.*encoding/json.Decoder"".retErrptype.error�%���h�������v��=��$�l� +�Qr"�,�Q)��BV8( � &E$0+2�0(]&E!LlUS+ 8�S V +L�A�<D3pTgclocals·378f3e900c220a5cad1989c9c06023bdTgclocals·1bc79a478470e6209b0ca20d87c54bd3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�4"".(*Client).KillContainer� +� +eH� %H�D$�H;Aw���H���HDŽ$HDŽ$H��$�H��$�H��H��H�H�H�H�H�$H��$�H�\$�H�\$H�,$H��H��H�H��H�L$H�D$H�H�,$H��H��H�H�H��$�H�l$H��H��H�H�H�H�l$ H��H��H�H�H��$�H�L$0H��$�H�D$8�H�L$@H�D$HH��$�H�H�CH�CH��$�H�4$H�5H�l$H��H�H�H�L$xH�L$H��$�H�D$ H��$�H�l$(H��H��H�H�H��H�L$XH�D$`H��$�H�T$hH��$�H�����H�H�$�H�L$H��H����1��H�L$pH� $H�<$��H��$�H�l$H��H��H�H��H�\$pH�\$pH�1�H9�tH�\$pH��$H��$H����H�H�$H�H�\$H�H�\$�H�D$봉%�t�����R���H��tH��$H��$H����HDŽ$HDŽ$H����& +*0runtime.morestack_noctxt�8type."".KillContainerOptions�runtime.convT2E�"".queryString�0go.string."/containers/"�$go.string."/kill?"�*runtime.concatstring4� go.string."POST"�"".(*Client).do�.type."".NoSuchContainer�"runtime.newobject�� runtime.duffzero�4runtime.writebarrierstring�Bgo.itab.*"".NoSuchContainer.error�0type.*"".NoSuchContainer�type.error� Bgo.itab.*"".NoSuchContainer.error�  runtime.typ2Itab`�"".autotmp_0813�0type.*"".NoSuchContainer"".autotmp_0812_"type."".doOptions"".autotmp_08110type.*"".NoSuchContainer"".autotmp_0810type.string"".autotmp_0809/8type."".KillContainerOptions "".err�type.error"".path�type.string "".~r1@type.error"".opts8type."".KillContainerOptions"".ctype.*"".Client,"����_����*� :�� �#n���Tgclocals·bd92ef728a38faac78badef3588d832fTgclocals·1705812f15ec71868ae696027438a358�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�8"".(*Client).RemoveContainer� +� +eH� %H�D$�H;Aw���H���HDŽ$HDŽ$H��$�H��$�H��H��H�H�H�H�H�$H��$�H�\$�H�\$H�,$H��H��H�H��H�L$H�D$H�H�,$H��H��H�H�H��$�H�l$H��H��H�H�H�H�l$ H��H��H�H�H��$�H�L$0H��$�H�D$8�H�L$@H�D$HH��$�H�H�CH�CH��$�H�4$H�5H�l$H��H�H�H�L$xH�L$H��$�H�D$ H��$�H�l$(H��H��H�H�H��H�L$XH�D$`H��$�H�T$hH��$�H�����H�H�$�H�L$H��H����1��H�L$pH� $H�<$��H��$�H�l$H��H��H�H��H�\$pH�\$pH�1�H9�tH�\$pH��$H��$H����H�H�$H�H�\$H�H�\$�H�D$봉%�t�����R���H��tH��$H��$H����HDŽ$HDŽ$H����& +*0runtime.morestack_noctxt�go.string."/containers/%s/copy"�fmt.Sprintf�� runtime.duffcopy� @type."".CopyFromContainerOptions� runtime.convT2E� + go.string."POST"� "".(*Client).do� .type."".NoSuchContainer� "runtime.newobject� � runtime.duffzero�4runtime.writebarrierstring�Bgo.itab.*"".NoSuchContainer.error�0type.*"".NoSuchContainer�type.error�Bgo.itab.*"".NoSuchContainer.error� runtime.typ2Itab�"type.bytes.Buffer�"runtime.newobject�� runtime.duffzero�2runtime.writebarrierslice�>go.itab.*bytes.Buffer.io.Reader�io.Copy�$type.*bytes.Buffer�type.io.Reader�>go.itab.*bytes.Buffer.io.Reader� runtime.typ2Itab��*"".autotmp_0839type.*uint8"".autotmp_0838�$type.*bytes.Buffer"".autotmp_0837$type.*bytes.Buffer"".autotmp_0836type.*uint8"".autotmp_08350type.*"".NoSuchContainer"".autotmp_0834�"type."".doOptions"".autotmp_0833�"type.interface {}"".autotmp_0831�&type.[]interface {}"".autotmp_0829�0type.*"".NoSuchContainer"".autotmp_0828$type.*bytes.Buffer"".autotmp_08270type.*"".NoSuchContainer"".autotmp_0826_@type."".CopyFromContainerOptions"".autotmp_0824�(type.[1]interface {}"".autotmp_08230type.*"".NoSuchContainerbytes.buf·2�type.[]uint8 "".err�type.error"".body�type.[]uint8 "".url�type.string "".~r1ptype.error"".opts@type."".CopyFromContainerOptions"".ctype.*"".ClientB%�������_�����W� +B� =��� ��E:ZI�vqkc I� +YR J4Tgclocals·387212f77114c618c992aca1d7f6e2d3Tgclocals·4e703ba17638508264f032b5f70033cd�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�4"".(*Client).WaitContainer� � eH� %H�D$�H;Aw���H���HDŽ$�HDŽ$H��$�H�H�CH�CH�H�,$H��H��H�H�H��$�H�\$H��$�H�t$H�5H�l$ H��H�H��H�\$0H�l$H��H��H�H�H��$�H�4$H�5H�l$H��H�H�H��$�H�l$(H��H��H�H�H��H�\$@H��$�H�\$HH��$�H�\$PH��$�H�L$XH�D$`H��$�H�T$hH��$�H�����H�H�$�H�L$H��H����1��H�L$pH� $H�<$��H��$�H�\$H��$�H�\$�H�\$pH�\$pH�1�H9�t)HDŽ$�H�\$pH��$H��$�H����H�H�$H�H�\$H�H�\$�H�D$먉%�e�����C���H��t$HDŽ$�H��$�H��$H����H�H�$�H�L$H�L$xH��$�H�$H��$�H�\$H��$�H�\$H�H��$�H�D$H��$�H�L$ �H�D$(H�L$0H��t$HDŽ$�H��$�H��$H����H�\$xH�+H��$�HDŽ$�HDŽ$H����( +*0runtime.morestack_noctxt�0go.string."/containers/"�"go.string."/wait"�*runtime.concatstring3� go.string."POST"�"".(*Client).do�.type."".NoSuchContainer�"runtime.newobject�� runtime.duffzero�4runtime.writebarrierstring�Bgo.itab.*"".NoSuchContainer.error�0type.*"".NoSuchContainer�type.error�Bgo.itab.*"".NoSuchContainer.error� runtime.typ2Itab� type.*struct { StatusCode int }� +.encoding/json.Unmarshal`�"".autotmp_0851�0type.*"".NoSuchContainer"".autotmp_0850/"type."".doOptions"".autotmp_08470type.*"".NoSuchContainer +"".&r�>type.*struct { StatusCode int } "".err�type.error"".body_type.[]uint8 "".~r2@type.error "".~r10type.int +"".idtype.string"".ctype.*"".Client:"����k�����/� +�4� :� �$[$9 �Q��VrTgclocals·eda57d60e805297221010beefc01cf3dTgclocals·ec5d02e01ec699817d1c71b60a3fa4d0�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�8"".(*Client).CommitContainer��eH� %H��$8���H;Aw���H��HHDŽ$�HDŽ$�H��$XH��$�H��H���H�H�$H��$�H�\$�H�\$H�,$H��H��H�H��H�L$H�D$H�H�,$H��H��H�H�H��$�H�L$H��$�H�D$�L�D$ H�T$(H��$�H�H�CH�CH��$�H�H��$�H��$�H��$�H��$�H��$PH�4$H�5H�l$H��H�H�L��$�L�D$H��$�H�T$ H��$�H�l$(H��H��H�H�H��H�\$@H��$�H�\$HH��$�H�\$PH��$�H�L$XH�D$`H��$�H�T$hH��$�H�����H�H�$�H�L$H��H����1��H�L$pH� $H�<$��H��$XH�l$H��H��H�H��H�\$pH�\$pH�1�H9�t)HDŽ$�H�\$pH��$�H��$�H��H�H�H�$H�H�\$H�H�\$�H�D$먉%�h�����F���H��t$HDŽ$�H��$�H��$�H��H�H�H�$�H�L$H�L$xH��$�H�$H��$�H�\$H��$�H�\$H�H��$�H�D$H��$�H�L$ �H�D$(H�L$0H��t$HDŽ$�H��$�H��$�H��H�H�\$xH��$�HDŽ$�HDŽ$�H��H�0 +00runtime.morestack_noctxt�� runtime.duffcopy�"".(*Client).ResizeContainerTTY��eH� %H��$h���H;Aw���H��HDŽ$HHDŽ$PH�H�$H�D$�H�D$H��$�H�D$xH��$8H�$�H�L$H�D$H�H�3H�kH��$�H��$�H��$�H��$�H��$�H��$�H�H�$�H�\$H����H��H��H��$H��$H��$H��$H�$H��$�H�\$H��$�H�\$�H�H�$H�\$xH�\$H��$�H�\$H��$H�\$�H��$�H�\$pH��$@H�$�H�L$H�D$H�H�3H�kH��$�H��$�H��$�H��$�H��$�H��$�H�H�$�H�\$H����H��H��H��$H��$H��$H��$H�$H��$�H�\$H��$�H�\$�H�H�$H�\$pH�\$H��$�H�\$H��$H�\$�H��$�H�$�H�L$H�D$H��$�H�H�CH�CH�H�,$H��H��H�H�H��$(H�\$H��$0H�t$H�5H�l$ H��H�H�H��$�H�L$0H��$�H�D$8�H�\$@H�l$H��H��H�H�H��$ H�4$H�5H�l$H��H�H�H��$�H�l$(H��H��H�H�H��H�L$`H�D$hH��$HH��$PH��É�f�����U���0 +00runtime.morestack_noctxt�&type.net/url.Values�runtime.makemap�strconv.Itoa�go.string."h"�type.[1]string�"runtime.newobject�4runtime.writebarrierstring�&type.net/url.Values�$runtime.mapassign1�strconv.Itoa�go.string."w"�type.[1]string�"runtime.newobject�4runtime.writebarrierstring�&type.net/url.Values� $runtime.mapassign1� *net/url.Values.Encode� +0go.string."/containers/"� (go.string."/resize?"� *runtime.concatstring4�  go.string."POST"� "".(*Client).dop�*"".autotmp_0896_"type."".doOptions"".autotmp_0895type.*[1]string"".autotmp_0893type.string"".autotmp_0892type.[]string"".autotmp_0891type.string"".autotmp_0890type.string"".autotmp_0889/type.[]string"".autotmp_0888�type.string"".autotmp_0887type.string net/url.value·3�type.stringnet/url.key·2�type.stringnet/url.v·1�&type.net/url.Values net/url.value·3�type.stringnet/url.key·2�type.stringnet/url.v·1�&type.net/url.Values"".params�&type.net/url.Values "".~r3Ptype.error"".width@type.int"".height0type.int +"".idtype.string"".ctype.*"".Client%�����*�=��� 8Q#X`4X`4�Q:Tgclocals·78ce512784b85b97418b7726f81bf730Tgclocals·564b0cd8045e1e3a560aecfc019285da�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�8"".(*Client).ExportContainer� � eH� %H��$h���H;Aw���H��HDŽ$HHDŽ$PH��$0H����H�H�$�H�L$H��H����1��H��$�H� $H�<$��H��$(H�l$H��H��H�H��H��$�H��$�H�1�H9�t H��$�H��$PH��$HH���H�H�$H�H�\$H�H�\$�H�D$뱉%�k�����F���H��$�H�H�CH��$�H���pH��H��H��$�H��$�H��$�H�H�$H��$(H�\$�H�T$H�D$H��$�H�$H��$�H�T$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �L�D$(H�T$0H��$�1��H��@��$�H��$8H��$�H��H�H�H��$ H�4$H�5H�l$H��H�H�L��$�L�D$H��$�H�T$ H��$�H�l$(H��H����H�L$pH�D$xH��$HH��$PH��É����( +00runtime.morestack_noctxt�.type."".NoSuchContainer�"runtime.newobject�� runtime.duffzero�4runtime.writebarrierstring�Bgo.itab.*"".NoSuchContainer.error�0type.*"".NoSuchContainer�type.error�Bgo.itab.*"".NoSuchContainer.error� runtime.typ2Itab�type.string�runtime.convT2E�2runtime.writebarrieriface�Bgo.string."/containers/%s/export"�fmt.Sprintf�� runtime.duffzero� go.string."GET"� +� runtime.duffcopy� +&"".(*Client).streamp�"".autotmp_0912�*type."".streamOptions"".autotmp_0911�"type.interface {}"".autotmp_0909�&type.[]interface {}"".autotmp_0907�0type.*"".NoSuchContainer"".autotmp_0904�(type.[1]interface {}"".autotmp_09030type.*"".NoSuchContainer "".url�type.string "".~r1Ptype.error"".optsgo.string."No such container: "�*runtime.concatstring20� "".~r0type.string "".err0type.*"".NoSuchContainer�R�D��15S +RnTgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�F"".(*ContainerAlreadyRunning).Error��eH� %H;aw���H��0H�D$@H�D$HH�H�,$H��H��H�H�H�|$8H��t-H�/H�|$H��H�H��H�\$ H�\$@H�\$(H�\$HH��0É�� + 0runtime.morestack_noctxt^Ngo.string."Container already running: "�*runtime.concatstring20` "".~r0type.string "".err@type.*"".ContainerAlreadyRunning`^_`� �,T +[%Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�>"".(*ContainerNotRunning).Error��eH� %H;aw���H��0H�D$@H�D$HH�H�,$H��H��H�H�H�|$8H��t-H�/H�|$H��H�H��H�\$ H�\$@H�\$(H�\$HH��0É�� + 0runtime.morestack_noctxt^Fgo.string."Container not running: "�*runtime.concatstring20` "".~r0type.string "".err8type.*"".ContainerNotRunning`^_`� �,T +[%Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�"".(*Env).Get��eH� %H;aw���H��8H�D$XH�D$`H�\$@H�$�H�D$H�T$HH�L$PH�H�$H�D$H�T$(H�T$H�L$0H�L$�H�\$ H��tH� H�kH�L$XH�l$`H��8É�� + + 0runtime.morestack_noctxtl"".(*Env).Map�,type.map[string]string�4runtime.mapaccess1_faststrPp +"".autotmp_0921type.string"".autotmp_0920type.string"".value0type.string "".keytype.string "".envtype.*"".Envpxop � +(,t +5kTgclocals·14c45952157723c8762210d9c661bf29Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go� "".(*Env).Exists��eH� %H;aw���H��@H�\$HH�$�H�D$H�T$PH�L$XH�H�$H�D$H�T$0H�T$H�L$8H�L$�H�L$ �\$(H��t �\$`H��@É�� + + 0runtime.morestack_noctxtH"".(*Env).Mapt,type.map[string]string�4runtime.mapaccess2_faststr@�"".autotmp_0923type.string "".~r10type.bool "".keytype.string "".envtype.*"".Env�^��4V  +#]Tgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go�""".(*Env).GetBool��eH� %H;aw���H��`H�\$hH�$H�\$pH�\$H�\$xH�\$�H�L$H�D$ H�L$PH� $H�D$XH�D$H�H�l$H��H��H�H��H�L$ H�D$(H�L$@H� $H�D$HH�D$�H�L$H�D$H�L$0H��t@H��uGH�t$0H�4$H�D$8H�D$H�5L�D$L��H�H��H�D$8�\$ ��t Ƅ$�H��`�H��u:H�t$0H�4$H�D$8H�D$H�5L�D$L��H�H��H�D$8�\$ ��u�H��u>H�t$0H�4$H�D$8H�D$H�5L�D$L��H�H��H�D$8�\$ ���o���H��u9H�t$0H�4$H�D$8H�D$H�5L�D$L��H�H���\$ ���0���Ƅ$�H��`� + 0runtime.morestack_noctxtp"".(*Env).Get�go.string." \t"�strings.Trim�strings.ToLower�go.string."0"� runtime.eqstring�go.string."no"� runtime.eqstring�"go.string."false"� runtime.eqstring� go.string."none"� runtime.eqstring@� "".autotmp_0926?type.string"".autotmp_0925type.string"".s_type.string"".value0type.bool "".keytype.string "".envtype.*"".Env"������ �D�F �7��(Tgclocals·9ff42bf311af152488d11f0f78c8d5ceTgclocals·23c4785fa8abd7e258acfe91c9f325f3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go�""".(*Env).SetBool��eH� %H�D$�H;Aw���H���H��$�H��$�H��$���$���H�T$HH��H��H�H� H�CH�t$pH�4$H�T$xH�T$H�H�l$H��H��H�H�H�L$`H�L$ H�D$hH�D$(�H�\$0H��$�H�\$8H��$�H�\$HH���1H�H�KH�[H��$�H��$�H��$�H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H��$�H��Hk�H�H�$H��$�H�\$H��$�H�\$�H��$�H��$�H��$�H�\$HH�$H�<$t4H��$�H�T$H��$�H�L$H��$�H�D$�H���É%�É�����H�T$@H��H��H�H� H�CH��$�H�4$H��$�H�T$H�H�l$H��H��H�H�H�L$PH�L$ H�D$XH�D$(�H�\$0H��$�H�\$8H��$�H�\$@H���.H�H�KH�[H��$�H��$�H��$�H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��H��$�H��$�H��$�H��Hk�H�H�$H��$�H�\$H��$�H�\$�H��$�H��$�H��$�H�\$@H�$H�<$t1H��$�H�T$H��$�H�L$H��$�H�D$��9����%�Ɖ����� +*0runtime.morestack_noctxt�go.string."1"�go.string."="�*runtime.concatstring3�type."".Env�"runtime.growslice�4runtime.writebarrierstring�2runtime.writebarrierslice�go.string."0"�go.string."="� *runtime.concatstring3� type."".Env� "runtime.growslice� 4runtime.writebarrierstring�2runtime.writebarrierslice@�*"".autotmp_0943type.uint64"".autotmp_0942type.uint64"".autotmp_0941type.int"".autotmp_0940type.int"".autotmp_0939type."".Env"".autotmp_0938type."".Env"".autotmp_0937type.string"".autotmp_0932_type."".Env"".autotmp_0931/type."".Env"".autotmp_0930type.string"".autotmp_0929type."".Env"".autotmp_0928type."".Env"".value�type.string "".key�type.string "".env�type.*"".Env"".value�type.string "".key�type.string "".env�type.*"".Env"".value0type.bool "".keytype.string "".envtype.*"".Env"������&V:��,��]Ty�]T 'Tgclocals·f774b632f7ff7d029527413a83030842Tgclocals·2d894b3b66dff3ff7aaa2a78013804f9�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go� "".(*Env).GetInt��eH� %H;aw���H�� H�\$(H�$H�\$0H�\$H�\$8H�\$�H�\$H�\$@H�� � + 0runtime.morestack_noctxtp$"".(*Env).GetInt64@@ "".~r10type.int "".keytype.string "".envtype.*"".Env@0?P +l6 +7Tgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go� "".(*Env).SetInt��eH� %H�D$�H;Aw���H��H��$�H�\$@H��$�H�$�H�L$H�D$H��$�H��$�H�|$XH�<$H�T$`H�T$H�H�|$H��H�H�H�L$HH�L$ H�D$PH�D$(�H�\$0H�\$hH�\$8H�\$pH�\$@H���"H�H�KH�[H��$�H��$�H��$�H��H)�H��}LH�H�$H�T$xH�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H�T$xH��Hk�H�H�$H�\$hH�\$H�\$pH�\$�H�T$xH��$�H��$�H�\$@H�$H�<$t4H��$�H�T$H��$�H�L$H��$�H�D$�H�ĨÉ%�É����� +*0runtime.morestack_noctxtxstrconv.Itoa�go.string."="�*runtime.concatstring3�type."".Env�"runtime.growslice�4runtime.writebarrierstring�2runtime.writebarrierslice@�"".autotmp_0955_type."".Env"".autotmp_0954/type."".Env"".autotmp_0953type.string"".autotmp_0952type."".Env"".autotmp_0951type.string"".value�type.string "".key�type.string "".env�type.*"".Env"".value0type.int "".keytype.string "".envtype.*"".Env"�����v"�;�TQ(Tgclocals·2cda55eacf8f3a391cf15caecdfeef06Tgclocals·6fac742cdcfec8bff38f6662e683bbda�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go�$"".(*Env).GetInt64��eH� %H;aw���H��XH�\$`H�$H�\$hH�\$H�\$pH�\$�H�L$H�D$ H�L$HH� $H�D$PH�D$H�H�l$H��H��H�H��H�L$ H�D$(H�L$8H� $H�D$@H�D$H�D$ +H�D$@�H�L$ H�D$(H�\$0H��tH�D$x����H��X�H�L$xH��X� + 0runtime.morestack_noctxtp"".(*Env).Get�go.string." \t"�strings.Trim� strconv.ParseInt@� +"".autotmp_0963type.string"".s?type.string "".~r10type.int64 "".keytype.string "".envtype.*"".Env ���� � ��d9 7�Tgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go�$"".(*Env).SetInt64��eH� %H�D$�H;Aw���H��H��$�H�\$@H��$�H�$H�D$ +�H�L$H�D$H��$�H��$�H�|$XH�<$H�T$`H�T$H�H�|$H��H�H�H�L$HH�L$ H�D$PH�D$(�H�\$0H�\$hH�\$8H�\$pH�\$@H���"H�H�KH�[H��$�H��$�H��$�H��H)�H��}LH�H�$H�T$xH�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H�T$xH��Hk�H�H�$H�\$hH�\$H�\$pH�\$�H�T$xH��$�H��$�H�\$@H�$H�<$t4H��$�H�T$H��$�H�L$H��$�H�D$�H�ĨÉ%�É����� +*0runtime.morestack_noctxt�"strconv.FormatInt�go.string."="�*runtime.concatstring3�type."".Env�"runtime.growslice�4runtime.writebarrierstring�2runtime.writebarrierslice@�"".autotmp_0969_type."".Env"".autotmp_0968/type."".Env"".autotmp_0967type.string"".autotmp_0966type."".Env"".autotmp_0965type.string"".value�type.string "".key�type.string "".env�type.*"".Env"".value0type.int64 "".keytype.string "".envtype.*"".Env"������"�D�TQTgclocals·2cda55eacf8f3a391cf15caecdfeef06Tgclocals·6fac742cdcfec8bff38f6662e683bbda�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go�""".(*Env).GetJSON��eH� %H;aw���H��HH�D$xHDŽ$�H�\$PH�$H�\$XH�\$H�\$`H�\$�H�L$H�D$ H��uH�D$xHDŽ$�H��H�H�L$8H� $H�D$@H�D$�H�\$H�,$H��H��H�H�H�H�\$hH�\$H�\$pH�\$ �H�L$(H�D$0H�L$xH��$�H��H� + + 0runtime.morestack_noctxt�"".(*Env).Get�2runtime.stringtoslicebyte�.encoding/json.Unmarshalp� +"".svaltype.string "".~r2Ptype.error"".iface0"type.interface {} "".keytype.string "".envtype.*"".Env�`��a���/,e L�Tgclocals·528c559c9193f2a671691be2686ab724Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go�""".(*Env).SetJSON� +� +eH� %H�D$�H;Aw���H���HDŽ$HDŽ$H��$�H�$H��$�H�\$�H�t$H�l$H�T$ H�D$(H�L$0H�L$pH��H�D$htH��$H��$H����H��$�H�\$@H��$�H�4$H��$�H�l$H��$�H�T$�H�L$H�D$ H��$�H��$�H�L$xH��$�H�|$XH�<$H�T$`H�T$H�H�|$H��H�H�H�L$HH�L$ H�D$PH�D$(�H�\$0H�\$xH�\$8H��$�H�\$@H���FH�H�KH�[H��$�H��$�H��$�H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H��$�H��Hk�H�H�$H�\$xH�\$H��$�H�\$�H��$�H��$�H��$�H�\$@H�$H�<$tLH��$�H�T$H��$�H�L$H��$�H�D$�HDŽ$HDŽ$H���É%뫉���� +*0runtime.morestack_noctxt�*encoding/json.Marshal�2runtime.slicebytetostring�go.string."="�*runtime.concatstring3�type."".Env�"runtime.growslice�4runtime.writebarrierstring� 2runtime.writebarrierslicep�"".autotmp_0983_type."".Env"".autotmp_0982/type."".Env"".autotmp_0981type.string"".autotmp_0980�type.string"".autotmp_0979type."".Env"".value�type.string "".key�type.string "".env�type.*"".Env "".err�type.error"".sval�type.[]uint8 "".~r2Ptype.error"".value0"type.interface {} "".keytype.string "".envtype.*"".Env&"�v������"�:< � Sy�ZT;Tgclocals·bb06efbb6a26e0f286c10766fad350d7Tgclocals·299a4d24490b926d38628658bb77eeb1�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go�""".(*Env).GetList� +� eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$�HDŽ$�H��$�H�$H��$�H�\$H��$�H�\$�H�L$H�D$ H�L$PH�D$XH��u,HDŽ$�HDŽ$�HDŽ$�H�İ�H�H�$�H�D$H�D$HH�D$@H�\$PH�$H�\$XH�\$�H�\$H�,$H��H��H�H�H�H�L$@H�H�D$pH�D$H�L$xH�L$ �H�L$(H�D$0H�D$hH��H�L$`�H�\$HH�H�KH�[H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H��$�H��Hk�H�H�$H�\$PH�\$H�\$XH�\$�H��$�H��$�H��$�H�\$HH�$H��$�H�T$H��$�H�L$H��$�H�D$�H�\$HH�+H��$�H�kH��$�H�kH��$�H�İ� +*0runtime.morestack_noctxt�"".(*Env).Get�type.[]string�"runtime.newobject�2runtime.stringtoslicebyte�type.*[]string�.encoding/json.Unmarshal�type.[]string�"runtime.growslice�4runtime.writebarrierstring� 2runtime.writebarrierslice`�"".autotmp_0995_type.[]string"".autotmp_0994/type.[]string"".autotmp_0992�type.*[]string +"".&l�type.*[]string "".err�type.error"".sval�type.string "".~r10type.[]string "".keytype.string "".envtype.*"".Env""������ �&�F: ,z�: lV':�M?Tgclocals·f09ff24693e6d72e9e2f82319a6e45a0Tgclocals·80f0398afc092a879ad303c2fec80b66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go�""".(*Env).SetList��eH� %H;aw���H��PHDŽ$�HDŽ$�H�\$pH�\$8H�\$xH�\$@H��$�H�\$HH�H�$H�\$8H�\$�H�\$H�l$H��H���H��H��H���H��H�\$XH�$H�\$`H�\$H�\$hH�\$�H�L$(H�D$0H��$�H��$�H��P� + + 0runtime.morestack_noctxt�type.[]string�runtime.convT2E�""".(*Env).SetJSON�� +"".autotmp_1005/type.[]string "".~r2`type.error"".value0type.[]string "".keytype.string "".envtype.*"".Env�����2� +hxTgclocals·ff7af1025fb7deae6ebf3487eab30c33Tgclocals·61e2515c69061b8fed0e66ece719f936�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go�"".(*Env).Set��eH� %H;aw���H��H��$�H�$H��$�H�t$H�5H�l$H��H�H�H��$�H�\$ H��$�H�\$(�H�\$0H�\$@H�\$8H�\$HH��$�H���H�H�KH�[H�T$hH�L$pH�\$xH��H)�H��}FH�H�$H�T$PH�T$H�L$XH�L$H�D$`H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$XH�D$`H��H�T$PH��Hk�H�H�$H�\$@H�\$H�\$HH�\$�H�T$PH�L$XH�D$`H��$�H�$H�<$t+H�T$hH�T$H�L$pH�L$H�D$xH�D$�H�ĀÉ%�̉����� + 0runtime.morestack_noctxtrgo.string."="�*runtime.concatstring3�type."".Env�"runtime.growslice�4runtime.writebarrierstring�2runtime.writebarriersliceP�"".autotmp_1010_type."".Env"".autotmp_1009/type."".Env"".autotmp_1008type.string"".autotmp_1007type."".Env"".value0type.string "".keytype.string "".envtype.*"".Env�������c�NE'Tgclocals·46b690808f7e1a8626f300054e53774fTgclocals·f9166171185d1f1926264897a0c959c1�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go� "".(*Env).Decode� +� +eH� %H�D$�H;Aw���H���HDŽ$�HDŽ$H�H�$�H�\$H�\$0H�H�$H�D$�H�D$H�\$0H�$H�D$�H��$�H�\$HH��$�H�\$PH�H�$�H�L$H��H����1��H�L$(H� $H�<$��H�\$HH�\$H�\$PH�\$�H�L$(H�D$0H� $H��H�H�D$xH�D$H��$�H�L$�H�D$H�L$ H�L$pH��H�D$htH��$�H��$H����H�\$0H�+H��$�1��H�H�$H�l$H��$�H�\$�H��$�1�H9���H��$�H����H� H�{H��$�H����H�3H�kH�L$xH��$�H��$�H�$H�t$XH�t$H�l$`H�l$H�L$8H�L$H�|$@H�|$ �H��$�H�$�H��$�1�H9��d���HDŽ$�HDŽ$H���É�h�����H����%�_�����=���$ +*0runtime.morestack_noctxtz8type.map[string]interface {}�"runtime.newobject�8type.map[string]interface {}�runtime.makemap�.runtime.writebarrierptr�4type.encoding/json.Decoder�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface�:type.*map[string]interface {}�>encoding/json.(*Decoder).Decode�� runtime.duffzero�8type.map[string]interface {}�&runtime.mapiterinit�""".(*Env).SetAuto�&runtime.mapiternextP�"".autotmp_1025"type.interface {}"".autotmp_1024�6type.*encoding/json.Decoder"".autotmp_10236type.*encoding/json.Decoder"".autotmp_1022�Btype.map.iter[string]interface {}"".autotmp_10218type.map[string]interface {} +"".&m�:type.*map[string]interface {}$encoding/json.r·2�type.io.Reader"".v�"type.interface {}"".k�type.string "".err�type.error "".~r10type.error "".srctype.io.Reader "".envtype.*"".Env("�������-�0�:K��9$  E#BF4k�Tgclocals·784852ecd61fa458e8af6c57e3ee02b8Tgclocals·8f12e5afe7e149987419843938d69919�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go�""".(*Env).SetAuto��eH� %H��$(���H;Aw���H��XH�H�$H��$xH�\$H��$�H�\$��D$�\$ ��t=H��$`H�$H��$hH�\$H��$pH�\$�H,�H�\$�H��X�H�H�$H��$xH�\$H��$�H�\$�H�L$H��$�H�t$ H��$��\$(����H��$`H�\$@H��$hH��$pH��H��$�H�<$H��$�H�T$H�H�|$H��H�H�H�L$XH�L$ H�D$`H�D$(�H�\$0H��$�H�\$8H��$�H�\$@H���.H�H�KH�[H��$@H��$HH��$PH��H)�H��}OH�H�$H��$(H�T$H��$0H�L$H��$8H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$0H��$8H��H��$(H��Hk�H�H�$H��$�H�\$H��$�H�\$�H��$(H��$0H��$8H�\$@H�$H�<$t1H��$@H�T$H��$HH�L$H��$PH�D$�������%�Ɖ�����H��$xH�$H��$�H�\$�H��$`H�l$H��$�H�T$H��$H�L$ H��$H�D$(H�\$0H��$�H��H��$���H�t$HH�,$H�T$H�L$�H�L$H�D$ H��$hH��$pH��$�H��$�H��$�H�<$H��$�H�T$H�H�|$H��H�H�H�L$hH�L$ H�D$pH�D$(�H�\$0H��$�H�\$8H��$�H�\$HH���.H�H�KH�[H��$(H��$0H��$8H��H)�H��}OH�H�$H��$@H�T$H��$HH�L$H��$PH�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��H��$HH��$PH��$@H��Hk�H�H�$H��$�H�\$H��$�H�\$�H��$@H��$HH��$PH�\$HH�$H�<$t1H��$(H�T$H��$0H�L$H��$8H�D$������%�Ɖ�����H�t$PH��$�H�H�CH��$�H���JH��H��H��$H��$ H��$H�$H��$xH�\$H��$�H�\$�H�H�,$H��H��H�H�H��$H�\$H��$H�\$H��$ H�\$ �H�L$(H�D$0H��$hH��$pH��$�H�<$H��$�H�T$H�H�|$H��H�H�H�L$xH�L$ H��$�H�D$(�H�\$0H��$�H�\$8H��$�H�\$PH���.H�H�KH�[H��$(H��$0H��$8H��H)�H��}OH�H�$H��$@H�T$H��$HH�L$H��$PH�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��H��$HH��$PH��$@H��Hk�H�H�$H��$�H�\$H��$�H�\$�H��$@H��$HH��$PH�\$PH�$H�<$t1H��$(H�T$H��$0H�L$H��$8H�D$��%����%�Ɖ����������< +00runtime.morestack_noctxtPtype.float64�$runtime.assertE2T2�$"".(*Env).SetInt64�type.string�$runtime.assertE2T2�go.string."="�*runtime.concatstring3�type."".Env�"runtime.growslice� 4runtime.writebarrierstring� +2runtime.writebarrierslice� *encoding/json.Marshal� 2runtime.slicebytetostring�go.string."="�*runtime.concatstring3�type."".Env�"runtime.growslice�4runtime.writebarrierstring�2runtime.writebarrierslice�2runtime.writebarrieriface�go.string."%v"�fmt.Sprintf�go.string."="�*runtime.concatstring3�type."".Env�"runtime.growslice�4runtime.writebarrierstring�2runtime.writebarriersliceP�N"".autotmp_1056type.uint64"".autotmp_1055type.uint64"".autotmp_1054type.int"".autotmp_1053type.int"".autotmp_1052type."".Env"".autotmp_1051type."".Env"".autotmp_1050type.string"".autotmp_1048�&type.[]interface {}"".autotmp_1047type.uint64"".autotmp_1046type.uint64"".autotmp_1045type.int"".autotmp_1044type.int"".autotmp_1043type."".Env"".autotmp_1042type."".Env"".autotmp_1041type.string"".autotmp_1040type.string"".autotmp_1035_type."".Env"".autotmp_1034/type."".Env"".autotmp_1033�type.string"".autotmp_1032type."".Env"".autotmp_1031type.string"".autotmp_1030�(type.[1]interface {}"".autotmp_1029type."".Env"".autotmp_1028type."".Env"".value�type.string "".key�type.string "".env�type.*"".Env"".value�type.string "".key�type.string "".env�type.*"".Env"".value�type.string "".key�type.string "".env�type.*"".Env "".err�type.error "".val�type.[]uint8"".sval�type.string"".value0"type.interface {} "".keytype.string "".envtype.*"".Env%�v��� �F�%:5 R�  q��!NJ��]T��]T �Ab�]T+Tgclocals·f2bff8318847e30874c64d3cd9d3a459Tgclocals·7c09a673592d13ccf4305e509b0c4fdf�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go�"".(*Env).Map��eH� %H�D$�H;Aw���H��H��$�H�kH��uHDŽ$�H�İ�H�H�$H�D$�H�\$H�\$PH��$�H���;H�H�CH�kH��$�1�H��$�H�D$@H��$�H��H�l$@H9���H�D$XH����H�H�xH�L$HH�T$pH�|$xH�T$`H�$H�|$hH�|$H�H�|$H��H�H�H�D$ �H�L$(H�D$0H�T$8H��$�H�H�$H�\$PH�\$H��vgH�L$H��H��$�H��H��$�vBH��H�\$�H�D$XH�L$HH��H��H�l$@H9��*���H�\$PH��$�H�İ�� � ���������� +*0runtime.morestack_noctxt�,type.map[string]string�runtime.makemap�go.string."="�strings.SplitN�,type.map[string]string�$runtime.mapassign1�$runtime.panicindex�$runtime.panicindex �"".autotmp_1076type.string"".autotmp_1075�type.*string"".autotmp_1074�type.int"".autotmp_1073type.int"".autotmp_1071/type."".Env"".autotmp_1069�type.int"".parts_type.[]string +"".kv�type.string"".m�,type.map[string]string "".~r0,type.map[string]string "".envtype.*"".Env&"�%������0�"#rKG\��Tgclocals·2148c3737b2bb476685a1100a2e8343eTgclocals·f3e8856499aee240134cb47f88c6cd55�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go�:"".(*Client).AddEventListener��eH� %H;aw���H��0H�D$HH�D$PH�D$ H�D$(H�\$8H�k0H�,$�H�T$8�\$����H�j0H�,$H�T$�H�T$8H�L$H�D$H�D$(H��H�L$ tH�L$HH�D$PH��0�H�j0H�,$H�\$@H�\$�H�D$H�L$H��tH�D$HH�L$PH��0�H�D$HH�D$PH��0�� + + 0runtime.morestack_noctxt�H"".(*eventMonitoringState).isEnabled�`"".(*eventMonitoringState).enableEventMonitoring�L"".(*eventMonitoringState).addListener@` +"".autotmp_1079type.error "".errtype.error "".~r1 type.error"".listener2type.chan<- *"".APIEvents"".ctype.*"".Client$`�_`5_`_`�0�,%& ! K�Tgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go�@"".(*Client).RemoveEventListener��eH� %H;aw���H��0H�D$HH�D$PH�\$8H�k0H�,$H�\$@H�\$�H�T$8H�D$H�L$H�L$(H��H�D$ tH�D$HH�L$PH��0�H�j0H�]XH��u$H�j0H�,$�H�D$HH�D$PH��0��� + 0runtime.morestack_noctxt�R"".(*eventMonitoringState).removeListener�b"".(*eventMonitoringState).disableEventMonitoring@` "".errtype.error "".~r1 type.error"".listener.type.chan *"".APIEvents"".ctype.*"".Client`[_`1_`� �,0   +CmTgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go�L"".(*eventMonitoringState).addListener� � eH� %H;aw���H��pHDŽ$�HDŽ$�H�\$xH�$H�<$��H�\$xH�$H�<$��H� Qj�YYH����H��$�H�$H�\$xH�\$H�|$��H�D$P��\$��t)H�H��$�H�H��$���H��p�H�\$xH�$H�<$�8H�$H�D$�H�\$xH���H�sPH�SXH�K`H�t$XH�T$`H�L$hH��H)�H��}FH�H�$H�t$@H�t$H�T$HH�T$H�L$PH�L$H�D$ �H�t$(H�T$0H�L$8H��H��H�l$HH�L$PH�t$@H��H�$H��$�H�\$�H�l$@H�T$HH�L$PH�\$xH�$H�<$tKH�$PH�l$XH�l$H�T$`H�T$H�L$hH�L$�HDŽ$�HDŽ$���H��pÉ%묉������%�����%�^�����H��pÉ%� ����%����� + 0runtime.morestack_noctxt�(sync.(*RWMutex).Lock�2sync.(*RWMutex).Unlock·f�"runtime.deferproc�""".listenerExists�6"".ErrListenerAlreadyExists�6"".ErrListenerAlreadyExists�&runtime.deferreturn�*sync.(*WaitGroup).Add�6type.[]chan<- *"".APIEvents�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice�&runtime.deferreturn�&runtime.deferreturn@� "".autotmp_1085_6type.[]chan<- *"".APIEvents"".autotmp_1084/6type.[]chan<- *"".APIEvents"".autotmp_10836type.[]chan<- *"".APIEvents "".~r1 type.error"".listener2type.chan<- *"".APIEvents"".eventState:type.*"".eventMonitoringStateB�Mh�����2���:�2.7)'�#   $F[--�GVTgclocals·4ab27d0e7d4f80bb5765ef5f61de5fe5Tgclocals·551282070bdf4bca9f3b8ada2a8f2d2a�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go�R"".(*eventMonitoringState).removeListener� � eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$�H��$�H�$H�<$���H��$�H�$H�<$�\H� Qj�YYH���4H��$�H�$H��$�H�\$H�|$�H�D$P��\$����H��$�1�1�E1�H����H�sPH�CXH�k`H��$�1�H��$�H�D$@H��$�H�l$@H9���H�t$XL�H�|$HL�L$PH��$�I9���H�T$`H�L$hL�D$pL��L��H)�H��}QH�H�$H�T$xH�T$H��$�H�L$H��$�H�D$H�D$ �L�L$PH�T$(H�L$0H�D$8H��H��H��$�H��$�H�T$xH��H�$L�L$�H�|$HH�t$XH�T$xH��$�L��$�H��H��H�l$@H9�� ���H��$�H�$H�<$t}H�$PH�T$`H�T$H�L$hH�L$L�D$pL�D$�H��$�H�$H�<$t9H�$H�D$�����HDŽ$�HDŽ$���H�ĨÉ%뾉%�w�����1���뼉%�������H�ĨÉ%�����%�p��� +*0runtime.morestack_noctxt�(sync.(*RWMutex).Lock�2sync.(*RWMutex).Unlock·f�"runtime.deferproc�""".listenerExists�6type.[]chan<- *"".APIEvents�"runtime.growslice�.runtime.writebarrierptr� 2runtime.writebarrierslice� *sync.(*WaitGroup).Add� +&runtime.deferreturn� +&runtime.deferreturn@�"".autotmp_1100_6type.[]chan<- *"".APIEvents"".autotmp_1098�4type.*chan<- *"".APIEvents"".autotmp_1097�type.int"".autotmp_1096�type.int"".autotmp_1094/6type.[]chan<- *"".APIEvents"".l�2type.chan<- *"".APIEvents"".newListeners�6type.[]chan<- *"".APIEvents "".~r1 type.error"".listener2type.chan<- *"".APIEvents"".eventState:type.*"".eventMonitoringState6"�S���7��'�V�:1>P� +;&&    Qa�AoCmTgclocals·00180cfd7eeeff04c22905d29bdac052Tgclocals·158185e77a15ce9170c1aa92e62cd73e�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go�R"".(*eventMonitoringState).closeListeners��eH� %H;aw���H��@H�\$HH����H�SPH�CXH�k`H�l$81�H�D$0H�D$H�T$(H��H�l$H9�}TH�D$ H�(H�L$H�,$�H�\$HH�$H�<$t^H�$H�D$�����H�D$ H�L$H��H��H�l$H9�|�H�\$HH��t!H�kPH�EH�EH�EH��@É�ۉ%뙉�=��� + 0runtime.morestack_noctxt�"runtime.closechan�*sync.(*WaitGroup).Add� +"".autotmp_1111?4type.*chan<- *"".APIEvents"".autotmp_1110_type.int"".autotmp_1109Otype.int"".autotmp_1108/6type.[]chan<- *"".APIEvents"".eventState:type.*"".eventMonitoringState����,�K #'   i�Tgclocals·ac5bea9c8a91f5fb1d31bdacc5067b57Tgclocals·29f0050a5ee7c2b9348a75428171d7de�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go�""".listenerExists��H�|$H�\$H��t2H�H�sH�k1�H9�}H�H9�u�D$�H��H��H9�|��D$É��0 "".~r2 type.bool"".list8type.*[]chan<- *"".APIEvents"".a2type.chan<- *"".APIEventsPP �  + Tgclocals·d3486bc7ce1948dc22d7ad1c0be2887aTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go�`"".(*eventMonitoringState).enableEventMonitoring��eH� %H;aw���H��H�D$0H�D$8H�\$ H�$H�<$���H�\$ H�$H�<$�aH� Qj�H�T$0YYH���7�Z0����H��@�j0H�H�$�H�L$H�H�\$ H�$H�<$��H�$8H�L$�H�H�$H�D$d�H�L$H�\$ H�$H�<$��H�$@H�L$�H�H�$H�D$�H�L$H�\$ H�$H�<$tPH�$HH�L$�H�\$ H�$H�\$(H�\$H� Qj�YYH�D$0H�D$8��H��É%막%�^����%������H��É%�����%�n���$ + 0runtime.morestack_noctxt�(sync.(*RWMutex).Lock�2sync.(*RWMutex).Unlock·f�"runtime.deferproc�type.int64�"runtime.newobject�.runtime.writebarrierptr�.type.chan *"".APIEvents� runtime.makechan�.runtime.writebarrierptr�type.chan error� runtime.makechan�.runtime.writebarrierptr�V"".(*eventMonitoringState).monitorEvents·f�runtime.newproc�&runtime.deferreturn�&runtime.deferreturn@0 "".~r1 type.error"".ctype.*"".Client"".eventState:type.*"".eventMonitoringState<0G �/0+/0&�D�,3  #A=$   @[�]Tgclocals·fa051c55663fc115869f36c85a0645b9Tgclocals·0115f8d53b75c1696444f08ad03251d9�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go�b"".(*eventMonitoringState).disableEventMonitoring��eH� %H;aw���H��H�D$H�D$ H�\$H�$H�<$���H�\$H�$H�<$��H� Qj�YYH����H�\$H�$�H�\$H�$H�<$tZH�$�H�L$�Y0��t%1�@�i0H�i@H�,$�H�\$H�kHH�,$�H�D$H�D$ ��H��É%띐�H��É%�N����%�)��� + 0runtime.morestack_noctxt�(sync.(*RWMutex).Lock�2sync.(*RWMutex).Unlock·f�"runtime.deferproc�R"".(*eventMonitoringState).closeListeners�,sync.(*WaitGroup).Wait�"runtime.closechan�"runtime.closechan�&runtime.deferreturn�&runtime.deferreturn0 "".~r0type.error"".eventState:type.*"".eventMonitoringState,G�!�8�,.   @<d@Tgclocals·a9282ac20787dc3025c0916068a42263Tgclocals·0115f8d53b75c1696444f08ad03251d9�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go�P"".(*eventMonitoringState).monitorEvents��eH� %H��$(���H;Aw���H��XH�H�$�H�D$H�D$`H��$`H�(H�H�$�H�D$H�D$hH��$hH�(H�D$pH�D$xH�\$`H�+H�,$��\$��tH�$������H�\$`H�+H�,$H�\$hH�+H�l$�H�T$H�L$H�L$xH��H�T$ptH�\$`H�+H�,$���H��X�H�\$`H�+H�,$��\$����H�$���H�L$`H�\$H�\$0H�H�k@H�l$XH�D$PH�H�kHH�l$HHDŽ$�HDŽ$�H��$�1��H��$�H�$H�D$��D$�H��$�H�,$H�l$XH�l$H�l$PH�l$H�l$/H�l$�H�t$`�\$ ����H�T$P�\$/��u��H��X�H�-H9�uH�.H�,$���H��X�H�.H�,$H�T$8H�T$�H�\$`H�+H�,$H�\$8H�\$H� Qj�YY����H��$�H�,$H�l$HH�l$H��$�H�l$��\$���2H��$�H��$�H�L$xH�-H9�uYH�T$pH�$H�L$H�-H�l$H�-H�l$�H�T$p�\$ ��tH�\$`H�+H�,$���H��X�H��H�T$p�����H�H�$�H�L$H�-H�)H�L$@H� $H�<$tqH�$H�\$`H�\$�H�\$@H�$H�<$tDH�$H�\$hH�\$�H�\$@Sj�YYH��u��H��XÐ�H��XÉ%볉%�H��$�H�,$H�l$0H�l$H�D$��\$�������H��$�H�$� ��H��X�R +00runtime.morestack_noctxtP:type.*"".eventMonitoringStateb"runtime.newobject�type.*"".Client�"runtime.newobject�L"".(*eventMonitoringState).noListeners�time.Sleep�V"".(*eventMonitoringState).connectWithRetry�b"".(*eventMonitoringState).disableEventMonitoring�&runtime.deferreturn�H"".(*eventMonitoringState).isEnabled�time.After�� runtime.duffzero�"runtime.newselect�&runtime.selectrecv2�&runtime.deferreturn�"".EOFEvent�b"".(*eventMonitoringState).disableEventMonitoring�&runtime.deferreturn�R"".(*eventMonitoringState).updateLastSeen� N"".(*eventMonitoringState).sendEvent·f� runtime.newproc� +$runtime.selectrecv� +""".ErrNoListeners� """.ErrNoListeners� """.ErrNoListeners� runtime.ifaceeq� b"".(*eventMonitoringState).disableEventMonitoring� &runtime.deferreturn� �type.struct { F uintptr; A0 **"".eventMonitoringState; A1 **"".Client }� "runtime.newobject� "".func·006� .runtime.writebarrierptr�.runtime.writebarrierptr�"runtime.deferproc�&runtime.deferreturn�&runtime.deferreturn�$runtime.selectrecv� runtime.selectgo�&runtime.deferreturn �"".autotmp_1130��type.*struct { F uintptr; A0 **"".eventMonitoringState; A1 **"".Client }"".autotmp_1129��type.struct { tcase uint16; ncase uint16; pollorder *uint8; lockorder *uint8; scase [3]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }; lockorderarr [3]*uint8; pollorderarr [3]uint16 }"".autotmp_1128�type.error"".autotmp_1127�type.chan error"".autotmp_1126�type.bool"".autotmp_1125�$type.*"".APIEvents"".autotmp_1124�.type.chan *"".APIEvents"".autotmp_1122type.bool +"".&c� type.**"".Client"".&eventState�net/http/httputil.NewClientConn�go.string."GET"�&net/http.NewRequest�2runtime.writebarrieriface�Dnet/http/httputil.(*ClientConn).Do�2runtime.writebarrieriface��type.struct { F uintptr; A0 *error; A1 **"".Client; A2 *chan *"".APIEvents; A3 *chan error }�"runtime.newobject�"".func·007�.runtime.writebarrierptr�.runtime.writebarrierptr�.runtime.writebarrierptr�.runtime.writebarrierptr�runtime.newproc�Bgo.itab.*crypto/tls.Conn.net.Conn�crypto/tls.Dial�2runtime.writebarrieriface�*type.*crypto/tls.Conn�type.net.Conn�Bgo.itab.*crypto/tls.Conn.net.Conn� runtime.typ2Itab�go.string."tcp"`�2"".autotmp_1160��type.*struct { F uintptr; A0 *error; A1 **"".Client; A2 *chan *"".APIEvents; A3 *chan error }"".autotmp_1159type.error"".autotmp_1158type.error"".autotmp_1157type.error"".autotmp_1156�type.*uint8"".autotmp_1154�type.error"".autotmp_1153�"type.interface {}"".autotmp_1151/&type.[]interface {}"".autotmp_1149�type.string"".autotmp_1148�type.int64"".autotmp_1147�(type.[1]interface {}"".autotmp_1146otype.string"".&errChan� type.*chan error"".&eventChan�0type.*chan *"".APIEvents +"".&c� type.**"".Client"".&err�type.*error "".res�.type.*net/http.Response "".req�,type.*net/http.Request"".conn�Dtype.*net/http/httputil.ClientConn"".dial�type.net.Conn"".addressOtype.string"".protocol�type.string "".uri�type.string "".~r3@type.error"".startTimetype.int64V%������������������ +� 1L�'1�_4� -K�7- h0(%�vB��O\O?E?w+(' !�O-�Tgclocals·8c02cd934f4d00aa05beba150d4d3e04Tgclocals·c1f40b05e3ffba0283c820006999a7cf�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go�."".(*Client).CreateExec��eH� %H��$8���H;Aw���H��HHDŽ$�HDŽ$�H��$�H�H�CH��$�H����H��H��H��$�H��$�H��$H�H�$H��$XH�\$H�D$ �H�T$H�D$H��$�H�$H��$�H�T$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$H�\$ �H�T$(H�D$0H��$�H��$�H��$XH��$H��H���H��$�H�H�CH�CH�H�$H��$H�\$�H�\$H��$�H��H��H�H�H��$PH�4$H�5H�l$H��H�H�H��$�H�\$H��$�H�t$ H��$�H�l$(H��H�H�H��H�\$@H��$�H�\$HH��$�H�\$PH��$�H�L$XH�D$`H��$�H�T$hH��$�H�����H�H�$�H�L$H��H����1��H�L$pH� $H�<$��H��$xH�l$H��H��H�H��H�\$pH�\$pH�1�H9�t)HDŽ$�H�\$pH��$�H��$�H��H�H�H�$H�H�\$H�H�\$�H�D$먉%�h�����F���H��t$HDŽ$�H��$�H��$�H��H�H�H�$�H�D$H�D$xH��$�H�$H��$�H�\$H��$�H�\$H� H��$�H�L$H��$�H�D$ �H�D$(H�L$0H��t$HDŽ$�H��$�H��$�H��H�H�\$xH��$�HDŽ$�HDŽ$�H��HÉ�>���2 +00runtime.morestack_noctxt�type.string�runtime.convT2E�2runtime.writebarrieriface�>go.string."/containers/%s/exec"�fmt.Sprintf�� runtime.duffcopy�2type."".CreateExecOptions�runtime.convT2E� go.string."POST"�"".(*Client).do� .type."".NoSuchContainer� "runtime.newobject� � runtime.duffzero� +4runtime.writebarrierstring� +Bgo.itab.*"".NoSuchContainer.error� 0type.*"".NoSuchContainer� type.error� Bgo.itab.*"".NoSuchContainer.error�  runtime.typ2Itab� type."".Exec� "runtime.newobject�type.*"".Exec�.encoding/json.Unmarshal��"".autotmp_1175�0type.*"".NoSuchContainer"".autotmp_1174�"type."".doOptions"".autotmp_1173�"type.interface {}"".autotmp_1171�&type.[]interface {}"".autotmp_11680type.*"".NoSuchContainer"".autotmp_11672type."".CreateExecOptions"".autotmp_1165�(type.[1]interface {}"".&exec�type.*"".Exec "".err�type.error"".body�type.[]uint8"".path�type.string "".~r2�type.error "".~r1�type.*"".Exec"".opts2type."".CreateExecOptions"".ctype.*"".Client@%����k�����,���>�=�� �$[$-*�vqkcI +�VuTgclocals·5a21f577b603cab6ea76228d95a69547Tgclocals·b0ce568b8ee350283c34690ddf2c6892�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/exec.go�,"".(*Client).StartExec��eH� %H��$����H;Aw���H��H��$�HDŽ$HDŽ$H����H�H�$�H�D$H��$�H�$H�<$��H��$�H�\$H��$�H�\$�H��$�H��$�H�1�H9�t H��$�H��$H��$H�Ĩ�H�H�$H�H�\$H�H�\$�H�D$뱉%�h���H��$�H��$�H��$�H��$�H�H�CH��$�H��� H��H��H��$�H��$H��$H�H�$H��$�H�\$�H�T$H�D$H��$�H�$H��$�H�T$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$H�\$H��$H�\$ �H�T$(H�D$0H��$�H��$���$�����H��$�H��$H��H���H��$�H�H�CH�CH�H�$H��$H�\$�H�\$H��$�H��H��H�H�H��$�H�4$H�5H�l$H��H�H�H��$�H�\$H��$�H�t$ H��$�H�l$(H��H�H�H��H�L$XH�D$`H��$�H�T$hH��$�H�����H�H�$�H�D$H��$�H�$H�<$��H��$�H�\$H��$�H�\$�H��$�H��$�H�1�H9�t H��$�H��$H��$H�Ĩ�H�H�$H�H�\$H�H�\$�H�D$뱉%�h���H��tH��$H��$H�Ĩ�HDŽ$HDŽ$H�Ĩ�H��$�H��$H��H���H�H��$XH��H���H��$H��$X��$@��$`H��$�H��$hH��H�H�H��$�H��$xH��H��H�H�H��$�H��$�H��H��H�H�H�H�$H��$H�\$�H�\$H��$�H��H��H�H�H��$�H�4$H�5H�l$H��H�H�H��$�H�\$H��$�H�t$ H��$XH�l$(H����H�L$xH��$�H��$H��$H�ĨÉ�����H +00runtime.morestack_noctxt�$type."".NoSuchExec�"runtime.newobject�4runtime.writebarrierstring�8go.itab.*"".NoSuchExec.error�&type.*"".NoSuchExec�type.error�8go.itab.*"".NoSuchExec.error� runtime.typ2Itab�type.string�runtime.convT2E�2runtime.writebarrieriface�4go.string."/exec/%s/start"�fmt.Sprintf� � runtime.duffcopy� 0type."".StartExecOptions� +runtime.convT2E� + go.string."POST"� "".(*Client).do� $type."".NoSuchExec� "runtime.newobject� 4runtime.writebarrierstring�8go.itab.*"".NoSuchExec.error�&type.*"".NoSuchExec�type.error�8go.itab.*"".NoSuchExec.error� runtime.typ2Itab�� runtime.duffcopy�""".statictmp_1200�� runtime.duffcopy�0type."".StartExecOptions�runtime.convT2E� go.string."POST"�� runtime.duffcopy�&"".(*Client).hijack��&"".autotmp_1199�*type."".hijackOptions"".autotmp_1198type.*uint8"".autotmp_1197&type.*"".NoSuchExec"".autotmp_1196�"type."".doOptions"".autotmp_1195�"type.interface {}"".autotmp_1193�&type.[]interface {}"".autotmp_1191�&type.*"".NoSuchExec"".autotmp_11890type."".StartExecOptions"".autotmp_1188&type.*"".NoSuchExec"".autotmp_1187�0type."".StartExecOptions"".autotmp_1185�type.string"".autotmp_1184�(type.[1]interface {}"".autotmp_1183&type.*"".NoSuchExec "".err�type.error"".path�type.string "".~r2�type.error"".opts00type."".StartExecOptions +"".idtype.string"".ctype.*"".ClientN%�������X������� � R�E +��� �  �  x' 4Z;�v�k< ;� +j6Tgclocals·1bba016e9a05211bda029a0d75dbaa68Tgclocals·49a517ef1dc6f4c5ce9b594512df2da4�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/exec.go�4"".(*Client).ResizeExecTTY��eH� %H��$���H;Aw���H��pHDŽ$�HDŽ$�H�H�$H�D$�H�D$H��$�H�D$pH��$�H�$�H�L$H�D$H�H�3H�kH��$�H��$�H��$�H��$�H��$�H��$H�H�$�H�\$H����H��H��H��$8H��$@H��$HH��$8H�$H��$�H�\$H��$�H�\$�H�H�$H�\$pH�\$H��$�H�\$H��$8H�\$�H��$�H�\$xH��$�H�$�H�L$H�D$H�H�3H�kH��$�H��$�H��$�H��$�H��$�H��$H�H�$�H�\$H����H��H��H��$8H��$@H��$HH��$8H�$H��$�H�\$H��$�H�\$�H�H�$H�\$xH�\$H��$�H�\$H��$8H�\$�H��$�H��$�H��$�H��$H��$�H�$�H�\$H��$�H�\$H��$�H��$P1��H��$PH����H��H��H��$ H��$(H��$0H�H�$H��$�H�\$�H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$�H�L$H�D$H��$ H��H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$ H�\$H��$(H�\$H��$0H�\$ �H�L$(H�D$0H��$H�H�CH�CH��$xH�4$H�5H�l$H��H�H�H��$�H�L$H��$�H�D$ H��$H�l$(H��H��H�H�H��H�L$`H�D$hH��$�H��$�H��pÉ�H�����P�����?���< +00runtime.morestack_noctxt�&type.net/url.Values�runtime.makemap�strconv.Itoa�go.string."h"�type.[1]string�"runtime.newobject�4runtime.writebarrierstring�&type.net/url.Values�$runtime.mapassign1�strconv.Itoa�go.string."w"�type.[1]string�"runtime.newobject�4runtime.writebarrierstring�&type.net/url.Values� $runtime.mapassign1� +*net/url.Values.Encode� � runtime.duffzero� type.string� runtime.convT2E� 2runtime.writebarrieriface� type.string� runtime.convT2E�2runtime.writebarrieriface�Q#X`4X`41~��KTgclocals·a32b6721babe8d92a9fbbdd02846be55Tgclocals·2087372b16bcbe3ecc60cbe0fcb1f127�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/exec.go�0"".(*Client).InspectExec��eH� %H��$h���H;Aw���H��HDŽ$@HDŽ$HH��$(H��$�H��$0H��$�H��$�H�H�CH��$�H���AH��H��H��$H��$H��$H�H�$H��$�H�\$�H�L$H�D$H��$H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$H�\$H��$H�\$H��$H�\$ �H�L$(H�D$0H��$�H�H�CH�CH��$ H�4$H�5H�l$H��H�H�H��$�H�L$H��$�H�D$ H��$�H�l$(H��H��H�H�H��H�\$@H��$�H�\$HH��$�H�\$PH��$�H�L$XH�D$`H��$�H�T$hH��$�H�����H�H�$�H�D$H�D$pH�$H�<$��H��$(H�\$H��$0H�\$�H�\$pH�\$pH�1�H9�t)HDŽ$8H�\$pH��$HH��$@H���H�H�$H�H�\$H�H�\$�H�D$먉%�e���H��t$HDŽ$8H��$@H��$HH���H�H�$�H�D$H�D$xH��$�H�$H��$�H�\$H��$�H�\$H� H��$�H�L$H��$�H�D$ �H�D$(H�L$0H��t$HDŽ$8H��$@H��$HH���H�\$xH��$8HDŽ$@HDŽ$HH��É����* +00runtime.morestack_noctxt�type.string�runtime.convT2E�2runtime.writebarrieriface�2go.string."/exec/%s/json"�fmt.Sprintf�go.string."GET"�"".(*Client).do�$type."".NoSuchExec�"runtime.newobject� 4runtime.writebarrierstring� 8go.itab.*"".NoSuchExec.error� +&type.*"".NoSuchExec� +type.error� +8go.itab.*"".NoSuchExec.error� + runtime.typ2Itab� &type."".ExecInspect� "runtime.newobject� (type.*"".ExecInspect� .encoding/json.Unmarshal`�"".autotmp_1243�&type.*"".NoSuchExec"".autotmp_1242_"type."".doOptions"".autotmp_1241�"type.interface {}"".autotmp_1239/&type.[]interface {}"".autotmp_1236&type.*"".NoSuchExec"".autotmp_1234�type.string"".autotmp_1233�(type.[1]interface {}"".&exec�(type.*"".ExecInspect "".err�type.error"".body�type.[]uint8"".path�type.string "".~r2@type.error "".~r10(type.*"".ExecInspect +"".idtype.string"".ctype.*"".Client@%����d�����,�� �>�=�� �$[$- &�v���VoTgclocals·f09ff24693e6d72e9e2f82319a6e45a0Tgclocals·384a8bdfde3344f7f6d876df7f86cb62�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/exec.go�,"".(*NoSuchExec).Error��eH� %H;aw���H��0H�D$@H�D$HH�H�,$H��H��H�H�H�|$8H��t-H�/H�|$H��H�H��H�\$ H�\$@H�\$(H�\$HH��0É�� + 0runtime.morestack_noctxt^Fgo.string."No such exec instance: "�*runtime.concatstring20` "".~r0type.string "".err&type.*"".NoSuchExec`^_`� �,T +[%Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/exec.go�."".(*Client).ListImages� � eH� %H�D$�H;Aw���H��HDŽ$(HDŽ$0HDŽ$8HDŽ$@HDŽ$H��$��$�H��$H��$���$ ��$�H�H�$H��$�H�\$�H�\$H�,$H��H��H�H��H�L$H�D$H�H�,$H��H��H�H�H��$�H�L$H��$�H�D$�H�L$ H�D$(H��$�H�H�CH�CH��$H�4$H�5H�l$H��H�H�H�L$xH�L$H��$�H�D$ H��$�H�l$(H��H��H�H�H��H�\$@H��$�H�\$HH��$�H�\$PH��$�H�D$`H�L$hH��$�H��H��$�ty  i�Tgclocals·40aca4155fa4487b7677bf99ab045398Tgclocals·af8ab8e4cb4a9346224baf13f9cce340�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image.go�,"".(*Client).LoadImage��eH� %H�D$�H;Aw���H���HDŽ$�HDŽ$�H��$�1��H��@��$�H��$�H��$�H��H�H�H��$�H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H�H��$�H�l$(H��H����H�L$pH�D$xH��$�H��$�H���� +*0runtime.morestack_noctxt�� runtime.duffzero� go.string."POST"�0go.string."/images/load"�� runtime.duffcopy�&"".(*Client).streamP�"".autotmp_1320�*type."".streamOptions "".~r10type.error"".opts0type."".LoadImageOptions"".ctype.*"".Client"�����:5� �4Tgclocals·57e1009a600f832f844e0e3c49ba5a89Tgclocals·aa2b73cba71b69bc124f64f356bee8e7�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image.go�0"".(*Client).ExportImage��eH� %H��$p���H;Aw���H��HDŽ$@HDŽ$HH��$�H�H�CH��$�H���pH��H��H��$�H��$�H��$�H�H�$H��$ H�\$�H�T$H�D$H��$�H�$H��$�H�T$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �L�D$(H�T$0H��$�1��H��@��$�H��$0H��$�H��H�H�H��$H�4$H�5H�l$H��H�H�L��$�L�D$H��$�H�T$ H��$�H�l$(H��H����H�L$pH�D$xH��$@H��$HH��É���� +00runtime.morestack_noctxt�type.string�runtime.convT2E�2runtime.writebarrieriface�4go.string."/images/%s/get"�fmt.Sprintf�� runtime.duffzero�go.string."GET"�� runtime.duffcopy�&"".(*Client).streamp�"".autotmp_1327�*type."".streamOptions"".autotmp_1326�"type.interface {}"".autotmp_1324�&type.[]interface {}"".autotmp_1322�type.string"".autotmp_1321�(type.[1]interface {} "".~r1Ptype.error"".opts4type."".ExportImageOptions"".ctype.*"".Client%���� +��=�x +�v�Tgclocals·740354061e4e9c9d9a50f05557f21f54Tgclocals·fabba7188ed1b3b8bac23e1a07c66457�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image.go�2"".(*Client).ExportImages��eH� %H�D$�H;Aw���H���H�H�$�H�D$H��$H��H���HDŽ$(HDŽ$0H�8�&H�hH���H��$�H��H�H��$�H�$H��$�H�T$�L�D$H�T$H��$�1��H����$�H��$�H�oH��$�H��H�H�H�H�,$H��H��H�H�L��$�L�D$H��$�H�T$�H�\$ H�l$H��H��H�H�H��$�H�4$H�5H�l$H��H�H�H��$�H�l$(H��H����H�L$pH�D$xH��$(H��$0H����H�H��$(H�H��$0H���� +*0runtime.morestack_noctxtJ6type."".ExportImagesOptions\"runtime.newobject�� runtime.duffcopy�8type.*"".ExportImagesOptions�"".queryString�� runtime.duffzero�0go.string."/images/get?"�*runtime.concatstring2�go.string."GET"�� runtime.duffcopy�&"".(*Client).stream�,"".ErrMustSpecifyNames�,"".ErrMustSpecifyNames�� +"".autotmp_1335�*type."".streamOptions"".autotmp_1333�type.string"".&opts�8type.*"".ExportImagesOptions "".~r1`type.error"".ctype.*"".Client "����%� ��b�w .-xvPUTgclocals·0629ba7d00f7a57ad6e2352df47e7bb3Tgclocals·d83f68254ae1224f9001a252749abef2�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image.go�0"".(*Client).ImportImage��eH� %H�D$�H;Aw���H���H�H�$�H�D$H��$�H��H���HDŽ$HHDŽ$PH�XH��u&H�H��$HH�H��$PH����H�D$hH�PH�HH����H��$�H�$H��$�H�L$H�-L�D$L��H��H�H��H�D$h�\$ ���]H�PH��$�H�HH��$�H����H�$H�L$H�-L�D$L��H��H�H��H�D$h�\$ ����H��H�H��$�H�$H��$�H�L$�H�D$hH�T$H�L$H��$�H�<$H��$�H�T$H��$�H�L$H�D$H�h0H�|$ H��H�H�H�h@H�\$0H��H��H�H��hP@�l$@�H�L$HH�D$PH��$HH��$PH����H�hH�$H��H��H�H��H�D$h�\$������H�hH�$H��H��H�H��H�L$H�D$H�T$ H�T$xH��H�D$ptH��$HH��$PH����H�L$`H�1�H9���H�L$`H��$�H�$H��$�H�L$�H�T$H�L$H�D$ H�\$(H�\$pH�\$0H�\$xH��$�H��$�H��$�H��$�H��$�H��$�H�H�$�H�L$H��H����1��H�L$XH� $H�<$��H��$�H�\$H��$�H�\$H��$�H�\$�H�\$XH�\$XH�1�H9�tQH�\$hH�$H�$0H�L$XH��$�H�D$H��$�H�L$�H�D$hH�hH�H��H��H�H��p���H�H�$H�H�\$H�H�\$�H�D$뀉%�0��������H�H�$H�H�\$H�H�\$�H�D$�D���H�h0H�EH�E����B +*0runtime.morestack_noctxtJ4type."".ImportImageOptions\"runtime.newobject�� runtime.duffcopy�""".ErrNoSuchImage�""".ErrNoSuchImage�go.string."-"� runtime.eqstring�go.string."-"� runtime.eqstring�6type.*"".ImportImageOptions�"".queryString�0"".(*Client).createImage�"".isURL�os.Open� +4go.itab.*os.File.io.Reader� +"io/ioutil.ReadAll� "type.bytes.Buffer� "runtime.newobject� � runtime.duffzero� 2runtime.writebarrierslice� >go.itab.*bytes.Buffer.io.Reader�2runtime.writebarrieriface�go.string."-"�$type.*bytes.Buffer�type.io.Reader�>go.itab.*bytes.Buffer.io.Reader� runtime.typ2Itab�type.*os.File�type.io.Reader�4go.itab.*os.File.io.Reader� runtime.typ2Itab��"".autotmp_1349type.*uint8"".autotmp_1348�$type.*bytes.Buffer"".autotmp_1347$type.*bytes.Buffer"".autotmp_1345type.string"".autotmp_1344type.string"".autotmp_1342type.string"".autotmp_1340$type.*bytes.Buffer"".autotmp_1339�type.*os.File"".&opts�6type.*"".ImportImageOptionsbytes.buf·2_type.[]uint8"".b/type.[]uint8 "".err�type.error "".~r1�type.error"".ctype.*"".Client4"�o�����w����P�b +&^Y�*+ ]�B2 0-��9�YJGB2Tgclocals·10971996d6a01a6d477c3318892d070fTgclocals·9b3781349ecf8ea1253d7ba626d001b4�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image.go�."".(*Client).BuildImage��eH� %H��$����H;Aw���H��H�H�$�H�T$H��$�H��H���HDŽ$�HDŽ$�H�z`u&H�H��$�H�H��$�H�Đ�H��$(1��H��$(H����H��H��H��$H��$H��$ H�H�$H��$�H�T$H�D$��H�L$H�D$H��$H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$H�D$��H�L$H�D$H��$H��H�$H��$�H�L$H��$�H�D$�H��$H�$H��$H�\$H��$ H�\$�H��$�H�\$H��$�H�L$ H�T$(H��$�H��H��$�tH��$�H��$�H�Đ�H���H��t.H�XH��u$H�$H�hxH�\$H��H��H�H��H��$�H�xP��H���H���tH���H��u&H�H��$�H�H��$�H�Đ�H���H��� H�xPt&H�H��$�H�H��$�H�Đ�HDŽ$�HDŽ$�H���H�$H��H��H�H�H�hH�\$H��H��H�H��H�L$ H�D$(H�\$0H��$�H�\$8H��$�H�H�$H��$�H�L$H��$�H�D$�H�\$H�l$H��H��H�H�H��$�H�$H�$P�H��$�H��$�H��t H��$�H��$�H��$�H�Đ�H�H��$�H�$H��$�H�D$�H�\$H��$H�\$H��$H��$�H�H�CH��$�H����H��H��H��$H��$H��$ H�H�$H��$H�\$�H�T$H�D$H��$H�$H��$�H�T$H��$�H�D$�H�H�,$H��H��H�H�H��$H�\$H��$H�\$H��$ H�\$ �H��$�L�D$(H�T$0H�H��$HH��H����Xp��$IH��$�H��$PH�hPH��$XH��H�H�H�h`H��$hH��H��H�H�H��$�H�4$H�5H�l$H��H�H�L��$�L�D$H��$�H�T$ H��$HH�l$(H��H����H�L$pH�D$xH��$�H��$�H�ĐÉ�X���H�H�+H��$H�kH��$H�H�+H��$�H�kH��$�H�H�$H��$�H�\$H��$H�\$H��$�H�\$�H��$��?�����V���P +00runtime.morestack_noctxtP2type."".BuildImageOptionsb"runtime.newobject�� runtime.duffcopy�2"".ErrMissingOutputStream�2"".ErrMissingOutputStream�� runtime.duffzero�2type."".AuthConfiguration�runtime.convT2E�2runtime.writebarrieriface�4type."".AuthConfigurations�runtime.convT2E�2runtime.writebarrieriface�$"".headersWithAuth�4runtime.writebarrierstring� """.ErrMissingRepo� +""".ErrMissingRepo� +,"".ErrMultipleContexts� +,"".ErrMultipleContexts� $"".createTarStream� type.io.Reader� runtime.convI2I�2runtime.writebarrieriface�4type.*"".BuildImageOptions�"".queryString�type.string�runtime.convT2E�2runtime.writebarrieriface�*go.string."/build?%s"�fmt.Sprintf�""".statictmp_1373�� runtime.duffcopy� go.string."POST"�� runtime.duffcopy�&"".(*Client).stream�0go.string."Content-Type"�6go.string."application/tar"�,type.map[string]string�$runtime.mapassign1��&"".autotmp_1372�*type."".streamOptions"".autotmp_1371"type.interface {}"".autotmp_1369&type.[]interface {}"".autotmp_1368�$type.io.ReadCloser"".autotmp_1367"type.interface {}"".autotmp_1366�"type.interface {}"".autotmp_1364�&type.[]interface {}"".autotmp_1362�type.string"".autotmp_1361type.string"".autotmp_1359�(type.[1]interface {}"".autotmp_1358�type.string"".autotmp_1357�type.string"".autotmp_1356�(type.[2]interface {}"".&opts�4type.*"".BuildImageOptions "".err�type.error "".err�type.error"".headers�,type.map[string]string "".~r1�type.error"".ctype.*"".Client^%�l��������=���������� f�e&�$ &&� � x { +>0������v��Tgclocals·689d5e2b826a4f8fae61c828d739d7d9Tgclocals·f991e5818d95c260e9075daec3edcda1�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image.go�*"".(*Client).TagImage� � eH� %H�D$�H;Aw���H���H�H�$�H�D$H��$H��H���HDŽ$(HDŽ$0H��$�H��u&H�H��$(H�H��$0H����H��H�H�D$pH�$H�L$xH�L$�H�\$H��$�H�\$H��$�H��$�H�H�CH��$�H����H��H��H��$�H��$�H��$�H�H�$H��$�H�\$�H�D$H�L$H��$�H�$H�D$pH�D$H�L$xH�L$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�t$H�5H�l$ H��H�H��H�\$0H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$�H�H�CH�CH��$�H�4$H�5H�l$H��H�H�H��$�H�L$H��$�H�D$ H��$�H�l$(H��H��H�H�H��H�T$XH�L$`H�D$hH���u&H�H��$(H�H��$0H����H��$(H��$0H���É�+���( +*0runtime.morestack_noctxtJ.type."".TagImageOptions\"runtime.newobject�� runtime.duffcopy�""".ErrNoSuchImage�""".ErrNoSuchImage�0type.*"".TagImageOptions�"".queryString�type.string�runtime.convT2E�2runtime.writebarrieriface�(go.string."/images/"�&go.string."/tag?%s"�*runtime.concatstring3�fmt.Sprintf� go.string."POST"� "".(*Client).do� +""".ErrNoSuchImage� +""".ErrNoSuchImage��"".autotmp_1389_"type."".doOptions"".autotmp_1388"type.interface {}"".autotmp_1386/&type.[]interface {}"".autotmp_1385�type.string"".autotmp_1384�type.string"".autotmp_1382(type.[1]interface {} "".~r2�type.error"".nametype.string"".ctype.*"".Client2"�s��������4�b&�B� & -���Tgclocals·23803564b4b262dab15001f621fd3b37Tgclocals·c6e5a101f01f70a879acdb3760944b0d�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image.go�"".isURL��eH� %H;aw���H��PH�\$XH�$H�\$`H�\$�H�T$H�D$H�\$ H�\$8H��H�D$0t +�D$hH��P�H�T$(H����H� +H�BH����H�L$@H� $H�D$HH�D$H�-L�D$L��H��H�H��H�T$(�\$ ��tkH��<uKH��tVH� +H�BH��uBH�L$@H� $H�D$HH�D$H�-L�D$L��H��H�H���\$ ��t +�D$hH��P��D$h���1�똉�<��� + 0runtime.morestack_noctxt\net/url.Parse� go.string."http"� runtime.eqstring�"go.string."https"� runtime.eqstring0� "".autotmp_1396type.string"".autotmp_1395type.string "".err?type.error"".pO"type.*net/url.URL "".~r1 type.bool"".utype.string&�@�������,  +�-w[1Tgclocals·d7e8a62d22b1cde6d92b17a55c33fe8fTgclocals·2b892b6166a29da84b4f26d3316f1499�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image.go�$"".headersWithAuth��eH� %H��$���H;Aw���H��hHDŽ$�HDŽ$�H�H�$H�D$�H�\$H�\$HH��$pH��$xH��$�H��$`1�H��$XH�D$8H��$PH��H�l$8H9���H�D$XH����H�H�hH�T$@H��$�H��$H��$�H� $H��$�H�l$��L$�L$4���hQ��H�H�$H��$�H�\$H��$�H�\$��L$4�\$����H�H�$�H�D$H�D$hH�D$`H�1�H9��@H�L$`H��$�H�D$xH��$�H��$�H�H�$�H�L$H��H����1��H�L$PH� $H�<$��H�\$xH�\$H��$�H�\$�H�\$PH�$H��$�H�\$H��$H�\$�H�L$H�D$ H��$�H��H��$�t$HDŽ$�H��$�H��$�H��h�H�H�+H��$�H�kH��$�H�L$hH�yH�QH�AH9��H� H��H)�H��H)�H��t H��H�H��H��$8H��$@H��$HH�H�$H��$ H�L$H��$(H�t$H��$0H�T$�H�\$ H��$�H�\$(H��$�H�H�$H�\$HH�\$H��$�H�\$H��$�H�\$�H�D$XH�T$@H��H��H�l$8H9��=���H�\$HH��$�HDŽ$�HDŽ$�H��h�� �%�&��������H�H�$H�H�\$H�H�\$�H�D$�������W�[�\���H�H�$H��$�H�\$H��$�H�\$��\$���$���H�H�$�H�D$H�D$pH�D$`H�1�H9���H�L$`H��$�H��$�H��$�H��$�H�H�$�H�L$H��H����1��H�L$PH� $H�<$��H��$�H�\$H��$�H�\$�H�\$PH�$H��$�H�\$H��$H�\$�H�L$H�D$ H��$�H��H��$�t$HDŽ$�H��$�H��$�H��h�H�H�+H��$�H�kH��$�H�D$pH�xH�PH�HH9���H�H��H)�H��H)�H��t H��H�H��H��$8H��$@H��$HH�H�$H��$H�D$H��$H�t$H��$H�T$�H�\$ H��$�H�\$(H��$�H�H�$H�\$HH�\$H��$�H�\$H��$�H�\$������ �%�j�����H���H�H�$H�H�\$H�H�\$�H�D$�������&���Z +00runtime.morestack_noctxt�,type.map[string]string�runtime.makemap�$runtime.efacethash�2type."".AuthConfiguration�&runtime.assertE2TOK�"type.bytes.Buffer�"runtime.newobject�>go.itab.*bytes.Buffer.io.Writer�4type.encoding/json.Encoder�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface�>encoding/json.(*Encoder).Encode� 6go.string."X-Registry-Auth"� 6encoding/base64.URLEncoding� Tencoding/base64.(*Encoding).EncodeToString� ,type.map[string]string� $runtime.mapassign1�$runtime.panicslice�$type.*bytes.Buffer�type.io.Writer�>go.itab.*bytes.Buffer.io.Writer� runtime.typ2Itab�4type."".AuthConfigurations�&runtime.assertE2TOK�"type.bytes.Buffer�"runtime.newobject�>go.itab.*bytes.Buffer.io.Writer�4type.encoding/json.Encoder�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface�>encoding/json.(*Encoder).Encode�:go.string."X-Registry-Config"�6encoding/base64.URLEncoding�Tencoding/base64.(*Encoding).EncodeToString�,type.map[string]string�$runtime.mapassign1�$runtime.panicslice�$type.*bytes.Buffer�type.io.Writer�>go.itab.*bytes.Buffer.io.Writer� runtime.typ2Itab`�N"".autotmp_1430type.uint64"".autotmp_1429type.uint64"".autotmp_1428type.uint64"".autotmp_1427type.[]uint8"".autotmp_14266type.*encoding/json.Encoder"".autotmp_14256type.*encoding/json.Encoder"".autotmp_1424type.*uint8"".autotmp_1423type.io.Writer"".autotmp_1419_type.[]uint8"".autotmp_1418�6type.*encoding/json.Encoder"".autotmp_14176type.*encoding/json.Encoder"".autotmp_1415�type.io.Writer"".autotmp_1414�type.uint32"".autotmp_1412"type.interface {}"".autotmp_1411�"type.interface {}"".autotmp_1410�$type.*interface {}"".autotmp_1409�type.int"".autotmp_1408�type.int"".autotmp_1407type.string"".autotmp_1406type.string"".autotmp_1405type.error"".autotmp_1404$type.*bytes.Buffer"".autotmp_1403�type.string"".autotmp_1402�type.string"".autotmp_1400�$type.*bytes.Buffer"".autotmp_1399/&type.[]interface {}"".&buf�$type.*bytes.Buffer"".&buf�$type.*bytes.Buffer "".~r0�type.[]uint8$encoding/json.w·2�type.io.Writer "".~r0�type.[]uint8$encoding/json.w·2�type.io.Writer "".err�type.error "".err�type.error"".auth�"type.interface {}"".headers�,type.map[string]string "".~r2@type.error "".~r10,type.map[string]string"".auths&type.[]interface {}8%������������ `�=#c~�$� "-ED�$�ETQ�`SI(�NQ=@tL(� N +=%Tgclocals·afcded8c13354e18af605d7f21ec25feTgclocals·7a03355e34b75c37acf5eff7bf674ad4�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image.go�2"".(*Client).SearchImages� +� eH� %H�D$�H;Aw���H���HDŽ$�HDŽ$�HDŽ$�HDŽ$HDŽ$H��$�H�H�CH�CH�H�,$H��H��H�H�H��$�H�\$H��$�H�\$�H�\$ H�l$H��H��H�H�H��$�H�4$H�5H�l$H��H�H�H��$�H�l$(H��H��H�H�H��H�\$@H��$�H�\$HH��$�H�\$PH��$�H�D$`H�L$hH��$�H��H�D$xtgo.itab.*bytes.Reader.io.Reader� "".(*Env).Decode� $type.*bytes.Reader� type.io.Reader� +>go.itab.*bytes.Reader.io.Reader� + runtime.typ2Itab@�"".autotmp_1448�$type.*bytes.Reader"".autotmp_1447$type.*bytes.Reader"".autotmp_1446/"type."".doOptions"".autotmp_1444$type.*bytes.Reader"".&env�type.*"".Envbytes.b·2�type.[]uint8 "".err�type.error"".body_type.[]uint8 "".~r1 type.error "".~r0type.*"".Env"".ctype.*"".Client4"�������,��@�* :�$�$-@"�{JEY�Tgclocals·9cf15d8275d9c299f023024ca604cf90Tgclocals·41bb44495be0a59dc118277b1d9139f9�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/misc.go�""".(*Client).Info� +� +eH� %H�D$�H;Aw���H���HDŽ$HDŽ$H��$�H�H�CH�CH��$�H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H�H��$�H�l$(H��H��H�H�H��H�\$@H��$�H�\$HH��$�H�\$PH��$�H�D$`H�L$hH��$�H��H��$�t$HDŽ$�H��$H��$H����H�H�$�H�\$H�\$xH��$�H��$�H��$�H��$�H��$�H��$�H�H�$�H�D$H�D$pH�$H�<$�H��$�H�\$H��$�H�\$H��$�H�\$�H�D$pH�@H�@ ����H�D$pH�1�H9���H�\$xH�$H�L$pH��$�H�D$H��$�H�L$�H�D$H�L$ H��t$HDŽ$�H��$H��$H����H�\$xH��$�HDŽ$HDŽ$H����H�H�$H�H�\$H�H�\$�H�D$�@����%����� +*0runtime.morestack_noctxt�go.string."GET"�"go.string."/info"�"".(*Client).do�type."".Env�"runtime.newobject�"type.bytes.Reader�"runtime.newobject�2runtime.writebarrierslice�>go.itab.*bytes.Reader.io.Reader� "".(*Env).Decode� $type.*bytes.Reader� type.io.Reader� +>go.itab.*bytes.Reader.io.Reader� + runtime.typ2Itab@�"".autotmp_1456�$type.*bytes.Reader"".autotmp_1455$type.*bytes.Reader"".autotmp_1454/"type."".doOptions"".autotmp_1452$type.*bytes.Reader"".&info�type.*"".Envbytes.b·2�type.[]uint8 "".err�type.error"".body_type.[]uint8 "".~r1 type.error "".~r0type.*"".Env"".ctype.*"".Client4"�������,��@�.>:�$�$-@"�{JEY�Tgclocals·9cf15d8275d9c299f023024ca604cf90Tgclocals·41bb44495be0a59dc118277b1d9139f9�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/misc.go�*"".ParseRepositoryTag��eH� %H;aw���H��@H�D$hH�D$pH�D$XH�D$`H�\$HH�$H�t$PH�t$H�5H�l$H��H�H��H�t$HH�L$PH�D$ H��}!H�t$XH�L$`H�D$hH�D$pH��@�H��H�D$(H��H��H9���H��H)�H��tH�H��H�D$0H�$H�T$8H�T$H�H�l$H��H��H�H��H�T$HH�L$P�\$ ��u4H�D$(H9�r#H�T$XH�D$`H�\$0H�\$hH�\$8H�\$pH��@�� H�T$XH�L$`H�D$hH�D$pH��@��  + 0runtime.morestack_noctxt�go.string.":"�"strings.LastIndex�go.string."/"� strings.Contains�$runtime.panicslice�$runtime.panicslice`�"".autotmp_1467type.uint64"".autotmp_1466type.uint64"".autotmp_1463type.uint64"".autotmp_1462type.int "".tagtype.string"".n/type.int "".tag@type.string"".repository type.string"".repoTagtype.string.�����'��"f>:!k4!d�F9Tgclocals·ca1ebfc68aaed1d083688775167e5178Tgclocals·8d600a433c6aaa81a4fe446d95c5546b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/misc.go�2"".(*Client).ListNetworks� � eH� %H�D$�H;Aw���H���HDŽ$�HDŽ$�HDŽ$�HDŽ$�HDŽ$�H��$�H�H�CH�CH��$�H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H�H��$�H�l$(H��H��H�H�H��H�\$@H��$�H�\$HH��$�H�\$PH��$�H�D$`H�L$hH��$�H��H�D$xtgo.itab.*"".NoSuchNetwork.error�,type.*"".NoSuchNetwork�type.error�>go.itab.*"".NoSuchNetwork.error� runtime.typ2Itab�type."".Network�"runtime.newobject�  type.*"".Network� +.encoding/json.Unmarshal`�"".autotmp_1482�,type.*"".NoSuchNetwork"".autotmp_1481/"type."".doOptions"".autotmp_1478,type.*"".NoSuchNetwork"".&network� type.*"".Network "".err�type.error"".body_type.[]uint8"".path�type.string "".~r2@type.error "".~r10 type.*"".Network +"".idtype.string"".ctype.*"".Client:"����d�����,� �2p:>� �$a$7i��VpTgclocals·be34fa03b4e4d696adaf8f647f7704fdTgclocals·0c8fa0fcc4836d09a64d3d20b95663fe�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/network.go�4"".(*Client).CreateNetwork� � eH� %H�D$�H;Aw���H���HDŽ$8HDŽ$@H��$H��$�H��H���H��$�H�H�CH�CH�H�$H��$�H�\$�H�\$H��$�H��H��H�H�H��$H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H�H��$�H�l$(H��H��H�H�H��H�\$@H��$�H�\$HH��$�H�\$PH��$�H�L$XH�D$`H�T$hH��$�H���u2HDŽ$0H�H��$8H�H��$@H����H��H��$�t$HDŽ$0H��$8H��$@H����H�H�$�H�\$H�\$xH�H�$�H�L$H�L$pH��$�H�$H��$�H�\$H��$�H�\$H�H��$�H�D$H��$�H�L$ �H�L$(H�D$0H��$�H��H��$�t$HDŽ$0H��$8H��$@H����H�t$xH�4$H��$H�l$H��H�H��H�\$xH�$H�$H�|$pH�/H�|$H��H�H��H�t$xH�4$H�$ H��$H�l$H��H�H��H�\$xH��$0HDŽ$8HDŽ$@H����& +*0runtime.morestack_noctxt�� runtime.duffcopy�8type."".CreateNetworkOptions�runtime.convT2E� go.string."POST"�*go.string."/networks"�"".(*Client).do�4"".ErrNetworkAlreadyExists�4"".ErrNetworkAlreadyExists�type."".Network�"runtime.newobject�@type."".createNetworkResponse·1�"runtime.newobject�Btype.*"".createNetworkResponse·1�.encoding/json.Unmarshal� +4runtime.writebarrierstring� +4runtime.writebarrierstring� 4runtime.writebarrierstring��"".autotmp_1490"type."".doOptions"".autotmp_1487O8type."".CreateNetworkOptions"".&network� type.*"".Network"".&resp�Btype.*"".createNetworkResponse·1 "".err�type.error"".body�type.[]uint8 "".~r2ptype.error "".~r1` type.*"".Network"".opts8type."".CreateNetworkOptions"".ctype.*"".Client<"����1������� +�T�: +S > 2$c$"''6$�j�V�bTgclocals·81381a8f40f0e35a38db28a8bb50de11Tgclocals·35aa8cef5f531c9de8f76600ceb85b27�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/network.go�2"".(*NoSuchNetwork).Error��eH� %H;aw���H��pHDŽ$�HDŽ$�H�\$HH�H�CH�\$HH����H��H��H�\$XH�T$`H�L$hH�H�$H�\$xH�\$H�|$���H�L$H�D$H�\$XH�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H�H�\$XH�\$H�\$`H�\$H�\$hH�\$ �H�L$(H�D$0H��$�H��$�H��pÉ%�l�����'��� + 0runtime.morestack_noctxt�type.string�runtime.convT2E�2runtime.writebarrieriface�>go.string."No such network: %s"�fmt.Sprintf0� +"".autotmp_1497o"type.interface {}"".autotmp_1495/&type.[]interface {}"".autotmp_1493O(type.[1]interface {} "".~r0type.string "".err,type.*"".NoSuchNetwork������2��d9Tgclocals·6d340c3bdac448a6ef1256f331f68dd3Tgclocals·403a8d79fd24b295e8557f6970497aa3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/network.go�$"".createTarStream��eH� %H��$����H;Aw���H��HDŽ$�HDŽ$�HDŽ$�HDŽ$�H��$�H�$H��$�H�\$�H�\$H��$H�\$H��$H�\$ H��$H�D$(H�L$0H��$�H��H��$�t0HDŽ$�HDŽ$�H��$�H��$�H�Ġ�H�H�$�H�D$H��� +H�-H��H��H�H�H��H��H��$�H��$�H��$�H��$`H����H�5H���H��H��H��$�H��$�H��$�H��H�$H��$�H�\$H��$�H�\$�H��$�H��$�H��$�H��$X1�H��$PH�D$@H��$HH�l$@H9�}PH�t$XH���H�H�nH�|$HH��$�H�L$`H��$�H�l$hH����H��H��H�l$@H9�|�H��$�H�$H��$�H�\$H��$H�\$H��$H�\$H��$H�\$ �H�D$(H�L$0H��$�H��H��$�t0HDŽ$�HDŽ$�H��$�H��$�H�Ġ�H�H�$�H�L$H��H���1��H�L$PH� $H�<$��H�$H��$H�\$H��$H�\$H��$H�\$�H�\$PH�$H�<$��H��$�H�\$H��$�H�\$H��$�H�\$�H�D$PH�@0H��@�h8H��$�H�$H��$�H�\$H�D$�H�L$H�D$ H�l$(H�T$0H��$�H��$�H��$�H��$�H�ĠÉ%�Q����%����������H� $H�l$H��$H�\$H��$H�\$H��$H�\$ �H�|$HH�t$X�\$(H��H�L$0H�\$8H�\$xH��H�L$p��H�\$`H��$�H�\$hH��$�H��$�1��H��$�H���DH��H��H��$0H��$8H��$@H�H�$H��$�H�\$�H�L$H�D$H��$0H�$H��$�H�L$H��$�H�D$�H�\$pH�$H�\$xH�\$�H�L$H�D$H��$0H��H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$0H�\$H��$8H�\$H��$@H�\$ �H�L$(H�D$0HDŽ$�HDŽ$�H��$�H��$�H�ĠÉ����<�#���H��$�H��$�H��$�H��H)�H��}OH�H�$H��$H�T$H��$ H�L$H��$(H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��H��$ H��$(H��$H��Hk�H�H�$H�\$`H�\$H�\$hH�\$�H�|$HH�t$XH��$H��$ H��$(H��$�H��$�H��$��)�����������?���������8 +00runtime.morestack_noctxt�("".parseDockerignore�type.[1]string�"runtime.newobject�""".statictmp_1509�""".statictmp_1511�� runtime.duffcopy�4runtime.writebarrierstring� +6"".validateContextDirectory� �type.github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive.TarOptions� "runtime.newobject� � runtime.duffzero� 2runtime.writebarrierslice� 2runtime.writebarrierslice��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive.TarWithOptions��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils.Matches�� runtime.duffzero�type.string�runtime.convT2E�2runtime.writebarrieriface�runtime.convI2E�2runtime.writebarrieriface�jgo.string."cannot match .dockerfile: '%s', error: %s"�fmt.Errorf�type.[]string�"runtime.growslice�4runtime.writebarrierstring��8"".autotmp_1527��type.*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive.TarOptions"".autotmp_1524type.int"".autotmp_1523type.int"".autotmp_1522�type.[]string"".autotmp_1521"type.interface {}"".autotmp_1520�"type.interface {}"".autotmp_1518�&type.[]interface {}"".autotmp_1517�type.string"".autotmp_1516�type.*string"".autotmp_1515type.int"".autotmp_1514type.int"".autotmp_1513type.[2]string"".autotmp_1508type.error"".autotmp_1506type.error"".autotmp_1503�type.string"".autotmp_1502?(type.[2]interface {}"".autotmp_1501�type.[]string "".err�type.error "".err�type.error"".includeFile�type.string("".forceIncludeFiles�type.[]string"".includes�type.[]string "".err�type.error"".excludes�type.[]string "".~r3`type.error "".~r2@$type.io.ReadCloser""".dockerfilePath type.string"".srcPathtype.stringF%���������������`*UW0Vnqe0 �_V��  Hn����@;��Mz�W_Tgclocals·d01647b6fcc19f6b40c264ab6c580992Tgclocals·4df3d887804869ca0d16462c47a4175f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tar.go�6"".validateContextDirectory��eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$�H�\$hH����H�-H��H���H��H��H�T$@H�L$HH�\$8H�$H��$�H�\$H��$�H�\$�H�\$8H�$H�\$@H�\$H�\$HH�\$�H�T$H�L$ H�\$PH�-H�+H��$�H�kH��$�H�kH�T$(H�$H�L$0H�L$H�\$�H�L$H�D$ H��$�H��$�H�ĈÉ���� +*0runtime.morestack_noctxt�""".statictmp_1545�� runtime.duffcopy�4runtime.writebarrierstring�$path/filepath.Join�"".func·008�$path/filepath.Walkp�"".autotmp_1544�type.[]string"".autotmp_1542oftype.struct { F uintptr; A0 *string; A1 *[]string }"".autotmp_1541�type.string"".autotmp_1540?type.[2]string "".~r2Ptype.error"".excludes type.[]string"".srcPathtype.string"������:�JfI�"N:Tgclocals·e3c75ef39e8363f5b00a257bd2be7adbTgclocals·7546955fbaa0a8a5c520077bd4d47105�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tar.go�("".parseDockerignore� � eH� %H�D$�H;Aw���H���HDŽ$HDŽ$HDŽ$ HDŽ$(HDŽ$0HDŽ$�HDŽ$�HDŽ$�H��$�H����H�-H��H���H��H��H��$�H��$�H��$�H�$H��$H�\$H��$H�\$�H��$�H�$H��$�H�\$H��$�H�\$�H�L$H�D$ H�L$hH� $H�D$pH�D$�H�\$H�\$xH�\$H��$�H�\$ H��$�H�D$(H�L$0H�L$@H��H�D$8�FH�$H�L$��\$���*H�\$XH�H�CH�\$XH���H��H��H��$�H��$�H��$�H�\$8H�$H�\$@H�\$�H�L$H�D$H��$�H�$H�L$HH�L$H�D$PH�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$�H��$H��$�H��$H��$�H��$ H��$(H��$0H���É�����H�\$xH�$H��$�H�\$H��$�H�\$�H�\$H�,$H��H��H�H�H�H�l$H��H��H�H��H�T$ H�L$(H�D$0H��$H��$H��$ HDŽ$(HDŽ$0H���É�,��� +*0runtime.morestack_noctxt�""".statictmp_1558�� runtime.duffcopy�4runtime.writebarrierstring�path.Join�$io/ioutil.ReadFile�os.IsNotExist�runtime.convI2E�2runtime.writebarrieriface�Zgo.string."error reading .dockerignore: '%s'"� fmt.Errorf� 2runtime.slicebytetostring� go.string."\n"� strings.Splitp�"".autotmp_1562�"type.interface {}"".autotmp_1560�&type.[]interface {}"".autotmp_1557otype.[]string"".autotmp_1556type.[]string"".autotmp_1554�(type.[1]interface {}"".autotmp_1552�type.string"".autotmp_1551?type.[2]string "".err�type.error"".ignore�type.[]uint8"".excludes�type.[]string "".~r2Ptype.error "".~r1 type.[]string"".roottype.string("��������(�^$�+�e8 $�+rop� �Tgclocals·2f519926ed4d9241bccfb3ede7c3f0baTgclocals·738657360054077c9c40a6546341a136�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tar.go�:"".(*tlsClientCon).CloseWrite��eH� %H;aw���H��@H�D$PH�D$XH�H�$H�|$HH��tmH�oH�|$H��H�H��H�T$H�L$ �\$(��t-H�L$8H� $H�T$0H�Z ��H�L$H�D$H�L$PH�D$XH��@�H�D$PH�D$XH��@É� + + 0runtime.morestack_noctxt^Jtype.interface { CloseWrite() error }�$runtime.assertI2I2� +0� "".cwcJtype.interface { CloseWrite() error } "".~r0type.error"".c*type.*"".tlsClientCon �}���0, ++- +RnTgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tls.go�("".tlsDialWithDialer��eH� %H��$p���H;Aw���H��H��$HDŽ$HHDŽ$PHDŽ$XHDŽ$`H�+H�l$HH�K�C��$�H�kH��$�H��$�H��������H��<u|�H�$�L$H�D$H��$H����H�oH�<$H��H�H�H�H��$�H�T$��$�L$ H��$H�D$(�H�L$HH�D$0H���KH9��BH�H�$�H�\$H�\$xH�\$HH����H�H�$H�D$�H�D$H�\$xH�$H�D$�H�H�$�H�D$H�-H�(H�D$pH�$H�<$��H�$H�\$xH�\$�H�\$HH�$H�\$pH�\$�H��$H�$H��$ H�\$H��$(H�\$H��$0H�\$H��$8H�\$ �H�\$(H��$�H�\$0H��$�H�D$8H�L$@H��$�H��H��$�t0HDŽ$HHDŽ$PH��$XH��$`H���H��$0H�$H��$8H�t$H�5H�l$H��H�H��H��$8H�D$ H���uH��H9��wH��$0H��$�H��$�H��$@H�[hH����H�H�$�H�D$H�H�$H��$�H�D$H��$@H�\$H�|$���H��$�H�$H�$`H��$�H�\$H��$�H�\$�H��$�H��$@H�H�$�H�\$H��$�H��$�H��$�H��$�H��$�H��$@H�\$PH�H�$�H�L$H��H���?1��H�L$hH� $H�<$�H��$�H�\$H��$�H�\$�H�\$hH�$H�<$��H�$8H�\$PH�\$�H�D$hH��@�hH��$�H�$H�D$�H�\$HH����H��$�H�+H�,$�H�L$H�D$H��$�H��H��$�tZH��$�H�$H��$�H�[ ��HDŽ$HHDŽ$PH��$�H��$XH��$�H��$`H���H�H�$�H�D$H�D$XH�$H�<$��H��$�H�+H�l$�H�\$XH�$H�<$��H�$H��$�H�\$H��$�H�\$�H�\$XH�\$XH�1�H9�t5H�\$XH��$PH��$HHDŽ$XHDŽ$`H���H�H�$H�H�\$H�H�\$�H�D$뜉%�T����%����H�H�$�H�D$H�-H�(H�D$`H�$H�<$��H�$H�\$xH�\$�H�\$`H�$H�<$��H�$H��$�H�\$�H�\$`Sj�YYHDŽ$�HDŽ$�H�H�$H�\$xH�+H�l$H��$�H�\$�H��$�H��$�H��$������%�o����%�;����%�����%�����������%������ �%�K���H�D$H������U���1��&���d +00runtime.morestack_noctxt�time.Now�time.Time.Sub�type.chan error�"runtime.newobject�type.chan error� runtime.makechan�.runtime.writebarrierptr�Rtype.struct { F uintptr; A0 *chan error }�"runtime.newobject�"".func·009�.runtime.writebarrierptr�time.AfterFunc�$net.(*Dialer).Dial� +go.string.":"� +"strings.LastIndex� ,type.crypto/tls.Config� "runtime.newobject� ,type.crypto/tls.Config� .runtime.writebarrierfat� 4runtime.writebarrierstring�*type.*crypto/tls.Conn�"runtime.newobject�(type.crypto/tls.Conn�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface�.runtime.writebarrierptr�.runtime.writebarrierptr�8crypto/tls.(*Conn).Handshake� +�(type."".tlsClientCon�"runtime.newobject�.runtime.writebarrierptr�2runtime.writebarrieriface�Bgo.itab.*"".tlsClientCon.net.Conn�*type.*"".tlsClientCon�type.net.Conn�Bgo.itab.*"".tlsClientCon.net.Conn� runtime.typ2Itab�~type.struct { F uintptr; A0 *chan error; A1 **crypto/tls.Conn }�"runtime.newobject�"".func·010�.runtime.writebarrierptr�.runtime.writebarrierptr�runtime.newproc�type.chan error�"runtime.chanrecv1�$runtime.panicslice��2"".autotmp_1585�*type.*"".tlsClientCon"".autotmp_1584��type.*struct { F uintptr; A0 *chan error; A1 **crypto/tls.Conn }"".autotmp_1583�*type.*crypto/tls.Conn"".autotmp_1582*type.*crypto/tls.Conn"".autotmp_1579�Ttype.*struct { F uintptr; A0 *chan error }"".autotmp_1577*type.*"".tlsClientCon"".autotmp_1576type.error"".autotmp_1574type.int"".autotmp_1570/type.time.Time"".&conn�,type.**crypto/tls.Conn"".&errChannel� type.*chan error +"".&c�.type.*crypto/tls.Config(crypto/tls.config·3�.type.*crypto/tls.Config$crypto/tls.conn·2�type.net.Conntime.t·2_type.time.Time"".hostname�type.string "".err�type.error"".rawConn�type.net.Conn"".timeout�$type.time.Duration "".~r5�type.error "".~r4`type.net.Conn"".configP.type.*crypto/tls.Config"".addr0type.string"".networktype.string"".dialer type.*net.DialerJ%��������������F]@i +1_q0>!K0�&@ +��Z      j��_�];0 JL(&#=M.8� 78De0Tgclocals·0389232f9bf0423206204d8b27e58130Tgclocals·a6f85fd4ba75b8cdab35ab28e50023d9�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tls.go�"".tlsDial��eH� %H;aw���H��PHDŽ$�HDŽ$�HDŽ$�HDŽ$�H�H�$�H�\$H�$H�\$XH�\$H�\$`H�\$H�\$hH�\$H�\$pH�\$ H�\$xH�\$(�H�l$0H�T$8H�L$@H�D$HH��$�H��$�H��$�H��$�H��P� + + 0runtime.morestack_noctxt�type.net.Dialer�"runtime.newobject�("".tlsDialWithDialer�� + "".~r4ptype.error "".~r3Ptype.net.Conn"".config@.type.*crypto/tls.Config"".addr type.string"".networktype.string�����J� U�Tgclocals·d85453ba2fc2b16513844b65495ea6c3Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tls.go�"".func·001� � eH� %H�D$�H;Aw���H��H�JH�ZH�\$XH�ZH�\$PH�Z H�\$`H�)H�,$H� Qj�YYH����H�\$XH�+H�,$H� Qj�L�D$`H�T$pYYH����H�D$hH�D$pA�X����H�*H�l$HH�1�H9���I�h H�$H��H��H�H�H�L$HH�D$xH�D$H��$�H�L$�H�T$(H�L$0H�T$hH��$�H�L$pH��$�H�H�$H�\$XH�+H�l$H��$�H�\$���H�Ę�H�H�$H�H�\$H�H�\$�L�D$PH�D$�:���H�*H�l$HH� 1�H9�tUI�h H�$H��H��H�H�I�h0H�\$H��H��H�H�H�T$HH�L$xH�L$ H��$�H�T$(�H�T$8H�L$@� ���H�H�$H�H�\$H�H�\$�L�D$PH�L$�t�����H�ĘÐ�H�Ę�. +*"runtime.morestack�(runtime.closechan·f�"runtime.deferproc�(runtime.closechan·f�"runtime.deferproc�>go.itab.*bufio.Reader.io.Reader�io.Copy�type.chan error�"runtime.chansend1�&runtime.deferreturn�$type.*bufio.Reader�type.io.Reader�>go.itab.*bufio.Reader.io.Reader� runtime.typ2Itab�>go.itab.*bufio.Reader.io.Reader��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.StdCopy�$type.*bufio.Reader�type.io.Reader�>go.itab.*bufio.Reader.io.Reader� runtime.typ2Itab�&runtime.deferreturn� &runtime.deferreturn�"".autotmp_1597type.*uint8"".autotmp_1595type.error"".autotmp_1594$type.*bufio.Reader"".autotmp_1593�$type.*bufio.Reader "".&bro&type.**bufio.Reader""".&hijackOptions�,type.*"".hijackOptions"".&errChanOut type.*chan error "".err_type.errorN"�.������ ��<� A!0WD 7f7 (R�N1k7Tgclocals·7c13896baab3273e10662a9a37b348ceTgclocals·51d2fd2674ba9ccfd7abd80151d2e032�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�"".func·002��eH� %H;aw���H��H�jH�JH�L$8H�BH�D$HH�l$@H�}��H�H�$H�l$H��H��H�H��H�\$H�,$H��H��H�H�H�|$@H�oH�|$H��H�H��H�L$(H�D$0H�L$PH�L$pH�D$XH�D$xH�H�$H�\$HH�+H�l$H�\$pH�\$�H�H�$H�t$8H�l$H��H�H��H�L$H�D$ H�D$hH�$H�L$`H�Y ��H�Ā�H�D$pH�D$xH�H�$H�(H�l$H�\$pH�\$�� + "runtime.morestack�type.io.Writer�runtime.convI2I�io.Copy�type.chan error�"runtime.chansend1�Jtype.interface { CloseWrite() error }�"runtime.assertI2I� +�type.chan error�"runtime.chansend1� "".autotmp_1603type.error"".autotmp_1602type.error"".&errChanIno type.*chan error"".&rwc�type.*net.Conn""".&hijackOptions,type.*"".hijackOptions "".err_type.error����E�$� 3V; ? 4]-E!ZTgclocals·0372b889336bbdf612862c172920463dTgclocals·2b592d649ecec7c5b5fac74b8e09bee8�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�"".func·003��eH� %H;aw���H��hH�BH�ZH�\$0H�ZH�\$ H�Z H�\$(H�hH�,$�H�D$XH�D$`H�-H�,$H�l$XH�l$H�l$0L�EL�D$�H�T$ �\$��t3H�D$XH�L$`H�L$@H��H�D$8tH�:uH�$H�D$H�L$�H�\$(H�+H�,$�H�T$ H�D$H�L$H��t#H�:uH�$H�D$HH�D$H�L$PH�L$�H��h� + "runtime.morestack�"runtime.closechan�type.chan error�(runtime.selectnbrecv�2runtime.writebarrieriface�,io.(*PipeReader).Close�2runtime.writebarrieriface� "".autotmp_1605type.error"".&readCloser(type.**io.PipeReader"".&retErr�type.*error"".&errCo type.*chan error "".err?type.error "".err_type.error����$� +9 X ,A:B8 +Tgclocals·7c13896baab3273e10662a9a37b348ceTgclocals·50e42ec547586bf00be346cef54257da�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�"".func·004��eH� %H��$����H;Aw���H��H�ZH��$�H�rH�ZH��$�H�Z H��$�H��$(1��H��$(H���2H��H��H��$H��$H��$ H�H�$H��$�H�t$�H�T$H�D$H��$H�$H��$�H�T$H��$�H�D$�H�H�$H��$�H�\$H�D$�H�T$H�D$H��$H��H�$H��$�H�T$H��$�H�D$�H�H�,$H��H��H�H�H��$H�\$H��$H�\$H��$ H�\$ �H�\$(H��$H�\$0H��$H��$�H�+H��$�H��$H1��H����$IH����$JH�1�H9���H��$�H��$�H��$hH��$�H��$pH��$�H�](H��$�H��$�H�>H�<$H�5H�|$H�H�H��$H�\$H��$H�t$ H��$HH�l$(H����H�D$pH�L$xH��$�H��H��$���H�H�$H�D$H�L$�H�L$�\$ ����H�H�����H�H�$�H�L$H��H���h1��H��$�H� $H�<$�>H��$�H�/H�|$H��H�H��H��$�H��$�H�1�H9���H��$�H��$�H��$�H��$�H�+H�,$�H��$�H�D$H�L$H��$�H��H��$�tH��u H��H��$�H��$�H��$�H��$�H��$�H�H�$H��$�H�+H�l$H��$�H�\$�H��$�H�+H�,$�H�Đ�H�H�$H�H�\$H�H�\$�H�D$�����%���������H�H�$H�H�\$H�H�\$�H�D$�M���������F +0"runtime.morestack�� runtime.duffzero�type.string�runtime.convT2E�2runtime.writebarrieriface�type.bool�runtime.convT2E�2runtime.writebarrieriface�Tgo.string."/containers/%s/stats?stream=%v"�fmt.Sprintf�� runtime.duffzero�@go.itab.*io.PipeWriter.io.Writer�go.string."GET"� � runtime.duffcopy� &"".(*Client).stream� +type.*"".Error� +$runtime.assertI2T2� .type."".NoSuchContainer� "runtime.newobject� � runtime.duffzero� 4runtime.writebarrierstring� Bgo.itab.*"".NoSuchContainer.error�,io.(*PipeWriter).Close�type.chan error�"runtime.chansend1�"runtime.closechan�0type.*"".NoSuchContainer�type.error�Bgo.itab.*"".NoSuchContainer.error� runtime.typ2Itab�&type.*io.PipeWriter�type.io.Writer�@go.itab.*io.PipeWriter.io.Writer� runtime.typ2Itab�$"".autotmp_1621type.*uint8"".autotmp_1620�0type.*"".NoSuchContainer"".autotmp_1618�*type."".streamOptions"".autotmp_1617"type.interface {}"".autotmp_1616�"type.interface {}"".autotmp_1614�&type.[]interface {}"".autotmp_1613�type.error"".autotmp_1612type.error"".autotmp_16110type.*"".NoSuchContainer"".autotmp_1609�&type.*io.PipeWriter"".autotmp_1608�type.string"".autotmp_1607�(type.[2]interface {}"".&errC� type.*chan error"".&writeCloser�(type.**io.PipeWriter"".&opts�*type.*"".StatsOptions +"".&c� type.**"".Client"".closeErr�type.error "".err�type.error%������ \� M�+< +]  $ �B ME -:���>3LN�0ETgclocals·fb05dbbfacbbe47b8b1eb4226ce34430Tgclocals·df3c8560fdbead80e4ddce1ccdbd1147�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�"".func·005��eH� %H�D$�H;Aw���H���H�BH�ZH�\$(H�ZH�\$0H�h H�l$ H�|$81��H�\$8H�$H�D$��D$�H�l$8H�,$H�l$ H�l$H�D$��\$��tH�\$(H�+H�,$�H����H�l$8H�,$H�l$0L�EL�D$H�D$��\$��tH����H�\$8H�$�  +*"runtime.morestack�� runtime.duffzero�"runtime.newselect�$runtime.selectrecv�,io.(*PipeReader).Close�$runtime.selectrecv� runtime.selectgo�"".autotmp_1629��type.struct { tcase uint16; ncase uint16; pollorder *uint8; lockorder *uint8; scase [2]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }; lockorderarr [2]*uint8; pollorderarr [2]uint16 }"".autotmp_1628� type.<-chan bool"".&quit�(type.*chan struct {}"".&readCloser�(type.**io.PipeReader&"����6���(� 8 ++/g!-0Tgclocals·73423680ca5f2d7df4fe760a82d507fbTgclocals·f34a2133376bc2e71ee31cc35164f3d3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�"".func·006��eH� %H;aw���H��H�JH�BH�)H�,$H�(H�l$H� Qj�YYH��� + "runtime.morestackhV"".(*eventMonitoringState).monitorEvents·fxruntime.newproc   +P�P +;Tgclocals·3280bececceccd33cb74587feedb1f9fTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go�"".func·007��eH� %H�D$�H;Aw���H���H�ZH�\$XH�ZH�\$`H�ZH�\$@H�Z H�\$PH��$�H�$H� Qj�YYH���kH��$�H���RH�S@H�kHH��$�H�,$H��$�H���%H�Z Sj�YYH���H�H�$H��$�H����H�o@H�|$H��H�H��H�L$H�D$ H��$�H�L$hH��$�H�D$pH�H�$�H�L$H��H���z1��H�L$0H� $H�<$�SH�\$hH�\$H�\$pH�\$�H�\$0H�\$(H�H�$�H�L$H�L$HH�\$(H�$H�H�D$xH�D$H��$�H�L$�H�L$H�D$ H�\$XH�$H��$�H�L$H��$�H�D$�H�D$XH�8�3H�H�-H9���L�$L��H��H�H�H�-H�l$H�-H�l$�H�D$X�\$ ��tXH�\$`H�H�k0H�,$��\$��t7H�H�$H�\$@H�+H�l$H�H�\$���H������H�H�-H9�u>L�$L��H��H�H�H�-H�l$H�-H�l$�H�D$X�\$ ���[���H�(H��$�H�hH��$�H�H�$H�\$PH�+H�l$H��$�H�\$�H�l$HH�]0H���3���H�\$`H�H�k0H�,$��\$��u��H����H�\$HH�\$8H�H�$H�\$@H�+H�l$H�\$8H�\$�������%����������������H���É������������H����J +*"runtime.morestack�Pnet/http/httputil.(*ClientConn).Close·f�"runtime.deferproc�"runtime.deferproc�type.io.Reader�runtime.convI2I�4type.encoding/json.Decoder�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface�"type."".APIEvents�"runtime.newobject�$type.*"".APIEvents�>encoding/json.(*Decoder).Decode�2runtime.writebarrieriface� io.EOF� io.EOF� io.EOF�runtime.ifaceeq� H"".(*eventMonitoringState).isEnabled� .type.chan *"".APIEvents� "".EOFEvent� +"runtime.chansend1� +&runtime.deferreturn� +&io.ErrUnexpectedEOF� +&io.ErrUnexpectedEOF� &io.ErrUnexpectedEOF� runtime.ifaceeq� type.chan error� "runtime.chansend1� H"".(*eventMonitoringState).isEnabled� &runtime.deferreturn� .type.chan *"".APIEvents�"runtime.chansend1�&runtime.deferreturn�&runtime.deferreturn � "".autotmp_1638�6type.*encoding/json.Decoder"".autotmp_16376type.*encoding/json.Decoder"".autotmp_1636_type.io.Reader"".autotmp_1635�$type.*"".APIEvents"".autotmp_1634type.bool"".autotmp_1633?type.error"".autotmp_1631type.error"".&event�$type.*"".APIEvents$encoding/json.r·2�type.io.Reader"".&errChan� type.*chan error"".&eventChan�0type.*chan *"".APIEvents +"".&c� type.**"".Client"".&err�type.*error"".decoder�6type.*encoding/json.Decoder"".connDtype.*net/http/httputil.ClientConn "".res.type.*net/http.ResponseZ"�8F������]��� �V�F&O�rM)!MA1!>\�F6�3 J +T( :$5Tgclocals·8e6ff68ca952ded665cfa894236f9944Tgclocals·48a37d9114fa45f0336e02e754d41f88�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go�"".func·008��eH� %H�D$�H;Aw���H���H�rH�ZH�\$HHDŽ$HDŽ$H�,$H��H�H�H��$�H�\$H��$�H�\$�H�l$ H�T$(H�D$0H�L$8H�L$xH��H�D$ptH��$H��$H����H�l$PH�,$H�T$XH�T$H�t$HH�l$H��H�H�H��H��$�H��$�H��$�\$(H�D$0H�|$8H�|$hH��H�D$`tH��$H��$H���À�tZH�4$H�] ���\$��t&H�H��$H�H��$H����HDŽ$HDŽ$H����H����H�$H��$H�\$��\$���+H��$�H��$�H��$�H��$�H��$�H�H�CH��$�H����H��H��H��$�H��$�H��$�H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$H��$H���É����H��$H�$H��$H�\$��\$��t HDŽ$HDŽ$H����H��$H��$H��$H��$H����H�4$H�]0�Ӌ\$�� +��t HDŽ$HDŽ$H����H��$�H�$H��$�H�[ ���\$����H��$�H�$H��$�H�\$�H�\$H�\$@H�D$H�L$ H��$�H��H��$��GH�$H�L$��\$���+H��$�H��$�H��$�H��$�H��$�H�H�CH��$�H����H��H��H��$�H��$�H��$�H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$H��$H���É����H�\$@H�$�HDŽ$HDŽ$H������0 +*"runtime.morestack�"path/filepath.Rel��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils.Matches� +�*path/filepath.SkipDir�*path/filepath.SkipDir�os.IsPermission�type.string�runtime.convT2E� 2runtime.writebarrieriface� 6go.string."can't stat '%s'"� +fmt.Errorf� os.IsNotExist� +� +�os.Open�os.IsPermission�type.string�runtime.convT2E�2runtime.writebarrieriface�Vgo.string."no permission to read from '%s'"�fmt.Errorf� os.(*File).Close��."".autotmp_1659"type.interface {}"".autotmp_1658*type.*[1]interface {}"".autotmp_1657&type.[]interface {}"".autotmp_1656�"type.interface {}"".autotmp_1654/&type.[]interface {}"".autotmp_1653type.error"".autotmp_1652type.string"".autotmp_1651(type.[1]interface {}"".autotmp_1649type.bool"".autotmp_1647type.bool"".autotmp_1645otype.string"".autotmp_1644O(type.[1]interface {}"".autotmp_1643type.bool"".&excludes�type.*[]string "".err�type.error"".currentFile�type.*os.File "".err�type.error "".err�type.error"".relFilePath�type.string "".~r3`type.error "".err@type.error"".f  type.os.FileInfo"".filePathtype.string�"����y��>�������N��'��8�����4��� d�GNb&  +$�( ( + (:.�/0lg�vG��v74Tgclocals·0e5d6e03d8b052993869281db2167ff7Tgclocals·8638ac1ded2e05617036c77f7600dfac�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tar.go�"".func·009��eH� %H;aw���H��pH�ZH�\$(H�D$@H�D$HH�D$0H�D$8H�H�$�H�D$H�D$ H�$H�<$��H�\$@H�\$H�\$HH�\$�H�\$ H�\$ H�1�H9�tOH�L$ H�D$PH�L$XH�D$0H�D$`H�L$8H�L$hH�H�$H�\$(H�+H�l$H�\$`H�\$�H��p�H�H�$H�H�\$H�H�\$�H�D$낉%�E��� + "runtime.morestack�.type.errors.errorString�"runtime.newobject�4runtime.writebarrierstring�Bgo.itab.*errors.errorString.error�type.chan error�"runtime.chansend1�0type.*errors.errorString�type.error�Bgo.itab.*errors.errorString.error� runtime.typ2Itab�"".autotmp_1669?type.error"".autotmp_1668�0type.*errors.errorString"".autotmp_1667type.error"".autotmp_16660type.*errors.errorString "".~r0type.errorerrors.text·2_type.string"".&errChannel� type.*chan error����@�h#�@R2b-Tgclocals·0372b889336bbdf612862c172920463dTgclocals·0730e324c95d53ccaec07bf254f1f516�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tls.go�"".func·010��eH� %H;aw���H��0H�ZH�\$H�ZH�+H�,$�H�\$H�\$ H�\$H�\$(H�H�$H�\$H�+H�l$H�\$ H�\$�H��0� + + "runtime.morestack^8crypto/tls.(*Conn).Handshake�type.chan error�"runtime.chansend1`"".autotmp_1672type.error"".&errChannel/ type.*chan error`X_��'G.;Tgclocals·73423680ca5f2d7df4fe760a82d507fbTgclocals·f1ce4f14231620ac9cd58e5cd8e6fa2d�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tls.go�"".init��eH� %H;aw���H��`���t���uH��`�� �����������������������������H�H�,$H��H��H�H��H�L$H�D$H�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H��H�L$H�D$H�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H��H�L$H�D$H�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H��H�T$H�L$H�D$ H�H�$H�T$HH�T$H�L$PH�L$H�D$XH�D$�H�H�,$H��H��H�H��H�L$H�D$H�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H��H�L$H�D$H�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H��H�L$H�D$H�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H��H�L$H�D$H�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H��H�L$H�D$H�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H��H�L$H�D$H�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H��H�L$H�D$H�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H��H�L$H�D$H�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H��H�L$H�D$H�H�$H�L$8H�L$H�D$@H�D$��H��`î + 0runtime.morestack_noctxt:"".initdone·R"".initdone·p"runtime.throwinit�"".initdone·��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils.init��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive.init�sync.init�math.init��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.init��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir.init��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts.init�time.init�strconv.init�runtime.init�reflect.init�$path/filepath.init�net/url.init�,net/http/httputil.init�net/http.init�net.init�io/ioutil.init� crypto/x509.init�crypto/tls.init�bufio.init�strings.init�path.init�os.init�io.init�fmt.init�$encoding/json.init�(encoding/base64.init�bytes.init�pgo.string."Failed to read authentication from dockercfg"�errors.New�""".AuthParseError�2runtime.writebarrieriface�8go.string."invalid endpoint"�errors.New�*"".ErrInvalidEndpoint�2runtime.writebarrieriface�Zgo.string."cannot connect to Docker endpoint"�errors.New�."".ErrConnectionRefused�2runtime.writebarrieriface� go.string."1.12"� "".NewAPIVersion� "".apiVersion112�2runtime.writebarrierslice�Hgo.string."container already exists"�errors.New�8"".ErrContainerAlreadyExists� 2runtime.writebarrieriface� bgo.string."no listeners present to receive event"� errors.New� """.ErrNoListeners� +2runtime.writebarrieriface� +jgo.string."listener already exists for docker events"� +errors.New� +6"".ErrListenerAlreadyExists� 2runtime.writebarrieriface� 2go.string."no such image"� errors.New� """.ErrNoSuchImage� 2runtime.writebarrieriface� �go.string."missing remote repository e.g. 'github.com/user/repo'"� errors.New� """.ErrMissingRepo� 2runtime.writebarrieriface� Bgo.string."missing output stream"� errors.New�2"".ErrMissingOutputStream�2runtime.writebarrieriface��go.string."image build may not be provided BOTH context dir and input stream"�errors.New�,"".ErrMultipleContexts�2runtime.writebarrieriface�hgo.string."must specify at least one name to export"�errors.New�,"".ErrMustSpecifyNames�2runtime.writebarrieriface�Dgo.string."network already exists"�errors.New�4"".ErrNetworkAlreadyExists�2runtime.writebarrieriface�"".initdone·�"".autotmp_1686/$type."".APIVersion"".autotmp_1685type.error"".autotmp_1684type.error"".autotmp_1683type.error"".autotmp_1682type.error"".autotmp_1681type.error"".autotmp_1680type.error"".autotmp_1679type.error"".autotmp_1678type.error"".autotmp_1677type.error"".autotmp_1676type.error"".autotmp_1675type.error"".autotmp_1674Otype.error ����� *�H�H��H H���H.HHW1H4HHDHHHHH�H� 7�Tgclocals·3280bececceccd33cb74587feedb1f9fTgclocals·7b2d1dc8e692ba633cb2c876407e20f2�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tls.go�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image.go�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/network.go�>type..hash."".AuthConfiguration��eH� %H;aw���H�� H�\$(H�$H�<$��H�D$H�\$8H�\$�H�D$H�\$(H�$H�<$��H�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$tgH�$ H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$0H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�R����%���� + 0runtime.morestack_noctxt�runtime.strhash�runtime.strhash�runtime.strhash�runtime.strhash@@"".autotmp_1690type.uintptr"".autotmp_1689type.uintptr"".autotmp_1688type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p4type.*"".AuthConfiguration@�?@1�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�:type..eq."".AuthConfiguration��eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH����H�H�CH9���H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ����H�\$PH���uH�SH�CH�\$XH���WH�sH�KH9��<H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ���H�\$PH����H�s H�K(H�\$XH����H�S H�C(H9���H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t}H�\$PH��tnH�S0H�C8H�\$XH��tWH�s0H�K8H9�u@H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ��t +�D$hH��H��D$hH��HÉ륉��D$hH��HÉ�+����� ����D$hH��HÉ����������D$hH��HÉ���������� + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring� runtime.eqstring� runtime.eqstring@�"".autotmp_1698type.string"".autotmp_1697type.string"".autotmp_1696type.string"".autotmp_1695type.string"".autotmp_1694type.string"".autotmp_1693type.string"".autotmp_1692?type.string"".autotmp_1691type.string "".~r30type.bool"".s type.uintptr"".q4type.*"".AuthConfiguration"".p4type.*"".AuthConfigurationJ���� ���������� v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�(type..hash.[8]string��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�runtime.strhash@` "".autotmp_1701type.int"".autotmp_1700type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*[8]string`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�$type..eq.[8]string��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$`H����H��Hk�H�H�3H�KH�\$hH��tvH��Hk�H�H�H�CH9�uVH�t$HH�4$H�L$PH�L$H�T$8H�T$H�D$@H�D$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt� runtime.eqstring@�"".autotmp_1705?type.string"".autotmp_1704type.string"".autotmp_1703_type.int"".autotmp_1702Otype.int "".~r30type.bool"".s type.uintptr"".qtype.*[8]string"".ptype.*[8]string&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�Dtype..hash.[8]"".AuthConfiguration��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�@H�H�$H�D$@H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�>type..hash."".AuthConfiguration@` "".autotmp_1708type.int"".autotmp_1707type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p:type.*[8]"".AuthConfiguration`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�@type..eq.[8]"".AuthConfiguration� +� +eH� %H;aw���H��h1�H�D$(H�l$(H9��H�D$0H�L$pH���EH�\$xH��Hk�@H�H���%H��Hk�@H�H�L$@H���H�1H�IH�\$8H����H�H�CH9���H�t$XH�4$H�L$`H�L$H�T$HH�T$H�D$PH�D$��\$ ���jH�\$@H����H�SH�CH�\$8H���jH�sH�KH9��3H�T$HH�$H�D$PH�D$H�t$XH�t$H�L$`H�L$��\$ ����H�\$@H��� H�s H�K(H�\$8H����H�S H�C(H9���H�t$XH�4$H�L$`H�L$H�T$HH�T$H�D$PH�D$��\$ ����H�\$@H����H�S0H�C8H�\$8H��tsH�s0H�K8H9�uYH�T$HH�$H�D$PH�D$H�t$XH�t$H�L$`H�L$��\$ ��t#H�D$0H��H�l$(H9������Ƅ$�H��h�Ƅ$�H��hÉ뉉�o���������������������q������������������������� + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring� runtime.eqstring� runtime.eqstring@� "".autotmp_1720type.string"".autotmp_1719type.string"".autotmp_1718type.string"".autotmp_1717type.string"".autotmp_1716type.string"".autotmp_1715type.string"".autotmp_1714?type.string"".autotmp_1713type.string"".autotmp_1712_4type.*"".AuthConfiguration"".autotmp_1711O4type.*"".AuthConfiguration"".autotmp_1710type.int"".autotmp_1709otype.int "".~r30type.bool"".s type.uintptr"".q:type.*[8]"".AuthConfiguration"".p:type.*[8]"".AuthConfiguration&���� ��P�����Tgclocals·fa7203fd5ed88aea99b7be572f707eb0Tgclocals·65526a5f07004f02424fe51b799cdd23�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�(type..hash.[3]string��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�runtime.strhash@` "".autotmp_1723type.int"".autotmp_1722type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*[3]string`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�$type..eq.[3]string��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$`H����H��Hk�H�H�3H�KH�\$hH��tvH��Hk�H�H�H�CH9�uVH�t$HH�4$H�L$PH�L$H�T$8H�T$H�D$@H�D$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt� runtime.eqstring@�"".autotmp_1727?type.string"".autotmp_1726type.string"".autotmp_1725_type.int"".autotmp_1724Otype.int "".~r30type.bool"".s type.uintptr"".qtype.*[3]string"".ptype.*[3]string&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�(type..hash.[2]string��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�runtime.strhash@` "".autotmp_1730type.int"".autotmp_1729type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*[2]string`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�$type..eq.[2]string��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$`H����H��Hk�H�H�3H�KH�\$hH��tvH��Hk�H�H�H�CH9�uVH�t$HH�4$H�L$PH�L$H�T$8H�T$H�D$@H�D$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt� runtime.eqstring@�"".autotmp_1734?type.string"".autotmp_1733type.string"".autotmp_1732_type.int"".autotmp_1731Otype.int "".~r30type.bool"".s type.uintptr"".qtype.*[2]string"".ptype.*[2]string&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�4type..hash."".dockerConfig��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|runtime.strhash�runtime.strhash@@ +"".autotmp_1736type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*"".dockerConfig@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�0type..eq."".dockerConfig��eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH����H�H�CH9���H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t}H�\$PH��tnH�SH�CH�\$XH��tWH�sH�KH9�u@H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ��t +�D$hH��H��D$hH��HÉ륉��D$hH��HÉ�,�������� + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring@�"".autotmp_1740type.string"".autotmp_1739type.string"".autotmp_1738?type.string"".autotmp_1737type.string "".~r30type.bool"".s type.uintptr"".q*type.*"".dockerConfig"".p*type.*"".dockerConfig2���� ������ v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�:type..hash.[8]"".dockerConfig��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk� H�H�$H�D$ H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�4type..hash."".dockerConfig@` "".autotmp_1743type.int"".autotmp_1742type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p0type.*[8]"".dockerConfig`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�6type..eq.[8]"".dockerConfig��eH� %H;aw���H��h1�H�D$(H�l$(H9��!H�D$0H�L$pH���GH�\$xH��Hk� H�H���'H��Hk� H�H�L$@H���H�1H�IH�\$8H����H�H�CH9���H�t$XH�4$H�L$`H�L$H�T$HH�T$H�D$PH�D$��\$ ����H�\$@H����H�SH�CH�\$8H��tsH�sH�KH9�uYH�T$HH�$H�D$PH�D$H�t$XH�t$H�L$`H�L$��\$ ��t#H�D$0H��H�l$(H9������Ƅ$�H��h�Ƅ$�H��hÉ뉉�o������������������������� + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring@�"".autotmp_1751type.string"".autotmp_1750type.string"".autotmp_1749?type.string"".autotmp_1748type.string"".autotmp_1747_*type.*"".dockerConfig"".autotmp_1746O*type.*"".dockerConfig"".autotmp_1745type.int"".autotmp_1744otype.int "".~r30type.bool"".s type.uintptr"".q0type.*[8]"".dockerConfig"".p0type.*[8]"".dockerConfig&���� ��2���iqTgclocals·fa7203fd5ed88aea99b7be572f707eb0Tgclocals·65526a5f07004f02424fe51b799cdd23�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�.type..hash."".APIEvents��eH� %H;aw���H�� H�\$(H�$H�<$��H�D$H�\$8H�\$�H�D$H�\$(H�$H�<$��H�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$tgH�$ H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$0H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�R����%���� + 0runtime.morestack_noctxt�runtime.strhash�runtime.strhash�runtime.strhash�runtime.memhash@@"".autotmp_1755type.uintptr"".autotmp_1754type.uintptr"".autotmp_1753type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p$type.*"".APIEvents@�?@1�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�*type..eq."".APIEvents��eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH����H�H�CH9��hH�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ���.H�\$PH���H�SH�CH�\$XH����H�sH�KH9���H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ����H�\$PH����H�s H�K(H�\$XH��txH�S H�C(H9�uaH�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t+H�l$PH�]0L�D$XI�h0H9�t +�D$hH��H��D$hH��H��D$hH��HÉ넉�j����D$hH��HÉ������������D$hH��HÉ�w�����Z��� + + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring� runtime.eqstring@�"".autotmp_1761type.string"".autotmp_1760type.string"".autotmp_1759type.string"".autotmp_1758type.string"".autotmp_1757?type.string"".autotmp_1756type.string "".~r30type.bool"".s type.uintptr"".q$type.*"".APIEvents"".p$type.*"".APIEventsJ���� �� �������� v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�>"".(*eventMonitoringState).Lock@@H�\$H�\$H�|$t��%��&(sync.(*RWMutex).Lock""..this:type.*"".eventMonitoringState   Tgclocals·519efd86263089ddb84df3cfe7fd2992Tgclocals·3280bececceccd33cb74587feedb1f9f�@"".(*eventMonitoringState).RLock@@H�\$H�\$H�|$t��%��&*sync.(*RWMutex).RLock""..this:type.*"".eventMonitoringState   Tgclocals·519efd86263089ddb84df3cfe7fd2992Tgclocals·3280bececceccd33cb74587feedb1f9f�D"".(*eventMonitoringState).RLocker�dH�D$H�D$H�\$H�\$H�|$t��%��J.sync.(*RWMutex).RLocker0 "".~r1 type.sync.Locker""..this:type.*"".eventMonitoringState@@@Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�D"".(*eventMonitoringState).RUnlock@@H�\$H�\$H�|$t��%��&.sync.(*RWMutex).RUnlock""..this:type.*"".eventMonitoringState   + Tgclocals·519efd86263089ddb84df3cfe7fd2992Tgclocals·3280bececceccd33cb74587feedb1f9f�B"".(*eventMonitoringState).Unlock@@H�\$H�\$H�|$t��%��&,sync.(*RWMutex).Unlock""..this:type.*"".eventMonitoringState   Tgclocals·519efd86263089ddb84df3cfe7fd2992Tgclocals·3280bececceccd33cb74587feedb1f9f�<"".(*eventMonitoringState).Add`LH�\$H�\$H�|$t H�D$��%��2*sync.(*WaitGroup).Add sync.delta·2type.int""..this:type.*"".eventMonitoringState000Tgclocals·cd30d2bcfdea04ed7c49639580b4bd08Tgclocals·3280bececceccd33cb74587feedb1f9f�>"".(*eventMonitoringState).Done`LH�\$H�\$H�|$t H�D$��%��2,sync.(*WaitGroup).Done""..this:type.*"".eventMonitoringState000Tgclocals·519efd86263089ddb84df3cfe7fd2992Tgclocals·3280bececceccd33cb74587feedb1f9f�>"".(*eventMonitoringState).Wait`LH�\$H�\$H�|$t H�D$��%��2,sync.(*WaitGroup).Wait""..this:type.*"".eventMonitoringState000Tgclocals·519efd86263089ddb84df3cfe7fd2992Tgclocals·3280bececceccd33cb74587feedb1f9f�."".(*APIVersion).String��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$@H�D$HH�\$81�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�t$8H��t+H�,$H��H�H�H��H�L$H�D$ H�L$@H�D$HH��0É�� + 0runtime.morestack_noctxt�$go.string."docker"�,go.string."APIVersion"�$go.string."String"�"runtime.panicwrap�("".APIVersion.String0` "".~r0type.string""..this&type.*"".APIVersion`�_`�� �ATgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�2"".(*APIVersion).LessThan��eH� %H;aw���H��8H�Y H��t H�|$@H9;uH�#H�\$@1�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�t$@H��t>H�,$H��H�H�H�H�\$HH�\$H�\$PH�\$ H�\$XH�\$(��\$0�\$`H��8É� + 0runtime.morestack_noctxt~$go.string."docker"�,go.string."APIVersion"�(go.string."LessThan"�"runtime.panicwrap�,"".APIVersion.LessThanPp "".~r1@type.bool"".other$type."".APIVersion""..this&type.*"".APIVersionp�op�� +}cTgclocals·14c45952157723c8762210d9c661bf29Tgclocals·3280bececceccd33cb74587feedb1f9f�D"".(*APIVersion).LessThanOrEqualTo��eH� %H;aw���H��8H�Y H��t H�|$@H9;uH�#H�\$@1�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�t$@H��t>H�,$H��H�H�H�H�\$HH�\$H�\$PH�\$ H�\$XH�\$(��\$0�\$`H��8É� + 0runtime.morestack_noctxt~$go.string."docker"�,go.string."APIVersion"�:go.string."LessThanOrEqualTo"�"runtime.panicwrap�>"".APIVersion.LessThanOrEqualToPp "".~r1@type.bool"".other$type."".APIVersion""..this&type.*"".APIVersionp�op�� +}cTgclocals·14c45952157723c8762210d9c661bf29Tgclocals·3280bececceccd33cb74587feedb1f9f�8"".(*APIVersion).GreaterThan��eH� %H;aw���H��8H�Y H��t H�|$@H9;uH�#H�\$@1�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�t$@H��t>H�,$H��H�H�H�H�\$HH�\$H�\$PH�\$ H�\$XH�\$(��\$0�\$`H��8É� + 0runtime.morestack_noctxt~$go.string."docker"�,go.string."APIVersion"�.go.string."GreaterThan"�"runtime.panicwrap�2"".APIVersion.GreaterThanPp "".~r1@type.bool"".other$type."".APIVersion""..this&type.*"".APIVersionp�op�� +}cTgclocals·14c45952157723c8762210d9c661bf29Tgclocals·3280bececceccd33cb74587feedb1f9f�J"".(*APIVersion).GreaterThanOrEqualTo��eH� %H;aw���H��8H�Y H��t H�|$@H9;uH�#H�\$@1�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�t$@H��t>H�,$H��H�H�H�H�\$HH�\$H�\$PH�\$ H�\$XH�\$(��\$0�\$`H��8É� + 0runtime.morestack_noctxt~$go.string."docker"�,go.string."APIVersion"�@go.string."GreaterThanOrEqualTo"�"runtime.panicwrap�D"".APIVersion.GreaterThanOrEqualToPp "".~r1@type.bool"".other$type."".APIVersion""..this&type.*"".APIVersionp�op�� +}cTgclocals·14c45952157723c8762210d9c661bf29Tgclocals·3280bececceccd33cb74587feedb1f9f�0"".(*APIVersion).compare��eH� %H;aw���H��8H�Y H��t H�|$@H9;uH�#H�\$@1�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�t$@H��t?H�,$H��H�H�H�H�\$HH�\$H�\$PH�\$ H�\$XH�\$(�H�\$0H�\$`H��8É� + 0runtime.morestack_noctxt~$go.string."docker"�,go.string."APIVersion"�&go.string."compare"�"runtime.panicwrap�*"".APIVersion.comparePp "".~r1@type.int"".other$type."".APIVersion""..this&type.*"".APIVersionp�op�� +}cTgclocals·14c45952157723c8762210d9c661bf29Tgclocals·3280bececceccd33cb74587feedb1f9f�Ltype..hash."".AttachToContainerOptions��eH� %H;aw���H�� H�\$(H�$H�<$��H�D$H�\$8H�\$�H�D$H�\$(H�$H�<$�hH�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$�&H�$ H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$��H�$0H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$��H�$@H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$tgH�$HH�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$PH�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�R����%�����%������%�����%�O��� + 0runtime.morestack_noctxt�runtime.strhash�"runtime.interhash�"runtime.interhash�"runtime.interhash�runtime.memhash�runtime.memhash�runtime.memhash@@"".autotmp_1774type.uintptr"".autotmp_1773type.uintptr"".autotmp_1772type.uintptr"".autotmp_1771type.uintptr"".autotmp_1770type.uintptr"".autotmp_1769type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".pBtype.*"".AttachToContainerOptions@�?@O�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�Htype..eq."".AttachToContainerOptions��eH� %H�D$�H;Aw���H��H��$�H���rH�3H�KH��$�H���RH�H�CH9��2H�t$xH�4$H��$�H�L$H�T$hH�T$H�D$pH�D$��\$ ����H��$�H����H�KH�sH��$�H����H�CH�SH9���H�D$HH�$H�T$PH�T$H�L$XH�L$H�t$`H�t$��\$ ���`H��$�H���GH�K H�s(H��$�H���&H�C H�S(H9��H�D$(H�$H�T$0H�T$H�L$8H�L$H�t$@H�t$��\$ ����H��$�H����H�K0H�s8H��$�H����H�C0H�S8H9��pH�D$(H�$H�T$0H�T$H�L$8H�L$H�t$@H�t$��\$ ���6H��$�H�$H�<$�H�$@H��$�H�\$H�|$��H�D$@H�D$�H��$�H�$H�<$��H�$@H��$�H�\$H�|$��H�D$@H�D$�H��$�H��$��\$��uƄ$�H�Ĉ�H�ZHH�iHH9�tƄ$�H�Ĉ��ZP�iP@8�tƄ$�H�Ĉ�Ƅ$�H�ĈÉ%�l����%�B����%� ����%�����Ƅ$�H�ĈÉ�h�����G���Ƅ$�H�ĈÉ����������Ƅ$�H�ĈÉ�>��������Ƅ$�H�ĈÉ��������� +*0runtime.morestack_noctxt� runtime.eqstring�runtime.ifaceeq�runtime.ifaceeq�runtime.ifaceeq�  runtime.memequal� + runtime.memequal@�"".autotmp_1783type.io.Writer"".autotmp_1782type.io.Writer"".autotmp_1781�type.io.Writer"".autotmp_1780�type.io.Writer"".autotmp_1779type.io.Reader"".autotmp_1778_type.io.Reader"".autotmp_1777?type.string"".autotmp_1776type.string "".~r30type.bool"".s type.uintptr"".qBtype.*"".AttachToContainerOptions"".pBtype.*"".AttachToContainerOptionsn"����������?������������Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·af3107c17ee1ab6f9f33230b5c7e3062�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�"".(*Port).Port��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$@H�D$HH�\$81�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�t$8H��t)H�,$H��H�H��H�L$H�D$H�L$@H�D$HH��0É�� + 0runtime.morestack_noctxt�$go.string."docker"� go.string."Port"� go.string."Port"�"runtime.panicwrap�"".Port.Port0` "".~r0type.string""..thistype.*"".Port`�_`� � �ATgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f� "".(*Port).Proto��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$@H�D$HH�\$81�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�t$8H��t)H�,$H��H�H��H�L$H�D$H�L$@H�D$HH��0É�� + 0runtime.morestack_noctxt�$go.string."docker"� go.string."Port"�"go.string."Proto"�"runtime.panicwrap�"".Port.Proto0` "".~r0type.string""..thistype.*"".Port`�_`�"� �ATgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�*type..hash.[8]"".Port��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�runtime.strhash@` "".autotmp_1788type.int"".autotmp_1787type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p type.*[8]"".Port`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�&type..eq.[8]"".Port��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$`H����H��Hk�H�H�3H�KH�\$hH��tvH��Hk�H�H�H�CH9�uVH�t$HH�4$H�L$PH�L$H�T$8H�T$H�D$@H�D$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt� runtime.eqstring@�"".autotmp_1792?type."".Port"".autotmp_1791type."".Port"".autotmp_1790_type.int"".autotmp_1789Otype.int "".~r30type.bool"".s type.uintptr"".q type.*[8]"".Port"".p type.*[8]"".Port&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�Htype..hash."".CommitContainerOptions��eH� %H;aw���H�� H�\$(H�$H�<$�cH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$�&H�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$��H�$ H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$��H�$0H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$tgH�$@H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$PH�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�R����%�����%������%���� + 0runtime.morestack_noctxt�runtime.strhash�runtime.strhash�runtime.strhash�runtime.strhash�runtime.strhash�runtime.memhash@@"".autotmp_1798type.uintptr"".autotmp_1797type.uintptr"".autotmp_1796type.uintptr"".autotmp_1795type.uintptr"".autotmp_1794type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p>type.*"".CommitContainerOptions@�?@E�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�Dtype..eq."".CommitContainerOptions� � eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH����H�H�CH9��zH�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ���@H�\$PH���*H�SH�CH�\$XH��� H�sH�KH9���H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ����H�\$PH����H�s H�K(H�\$XH����H�S H�C(H9��hH�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ���.H�\$PH���H�S0H�C8H�\$XH����H�s0H�K8H9���H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ����H�\$PH����H�s@H�KHH�\$XH��txH�S@H�CHH9�uaH�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t+H�l$PH�]PL�D$XI�hPH9�t +�D$hH��H��D$hH��H��D$hH��HÉ넉�j����D$hH��HÉ������������D$hH��HÉ�v�����X����D$hH��HÉ������������D$hH��HÉ�e�����H��� + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring� runtime.eqstring� runtime.eqstring� runtime.eqstring@�"".autotmp_1808type.string"".autotmp_1807type.string"".autotmp_1806type.string"".autotmp_1805type.string"".autotmp_1804type.string"".autotmp_1803type.string"".autotmp_1802type.string"".autotmp_1801type.string"".autotmp_1800?type.string"".autotmp_1799type.string "".~r30type.bool"".s type.uintptr"".q>type.*"".CommitContainerOptions"".p>type.*"".CommitContainerOptionsb���� �� ������������ v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�(type..hash."".Change��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|runtime.strhash�runtime.memhash@@ +"".autotmp_1810type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*"".Change@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�$type..eq."".Change��eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH��twH�H�CH9�uaH�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t+H�l$PH�]L�D$XI�hH9�t +�D$hH��H��D$hH��H��D$hH��HÉ녉�l��� + 0runtime.morestack_noctxt� runtime.eqstring@� "".autotmp_1812?type.string"".autotmp_1811type.string "".~r30type.bool"".s type.uintptr"".qtype.*"".Change"".ptype.*"".Change2���� �� ���� +nRTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�Ltype..hash."".CopyFromContainerOptions��eH� %H;aw���H�� H�\$(H�$H�<$��H�D$H�\$8H�\$�H�D$H�\$(H�$H�<$tgH�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$ H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�W��� + + 0runtime.morestack_noctxt�"runtime.interhash�runtime.strhash�runtime.strhash@@ "".autotmp_1815type.uintptr"".autotmp_1814type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".pBtype.*"".CopyFromContainerOptions@�?@'�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�Htype..eq."".CopyFromContainerOptions��eH� %H;aw���H��hH�\$xH����H� H�sH�\$pH���fH�H�SH9��IH�D$HH�$H�T$PH�T$H�L$XH�L$H�t$`H�t$��\$ ���H�\$pH����H�sH�KH�\$xH����H�SH�CH9���H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ����H�\$pH��ttH�S H�C(H�\$xH��t]H�s H�K(H9�uCH�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ��t Ƅ$�H��h�Ƅ$�H��hÉ량�Ƅ$�H��hÉ���������Ƅ$�H��hÉ������v��� + + 0runtime.morestack_noctxt�runtime.ifaceeq� runtime.eqstring� runtime.eqstring@�"".autotmp_1821type.string"".autotmp_1820type.string"".autotmp_1819type.string"".autotmp_1818_type.string"".autotmp_1817?type.io.Writer"".autotmp_1816type.io.Writer "".~r30type.bool"".s type.uintptr"".qBtype.*"".CopyFromContainerOptions"".pBtype.*"".CopyFromContainerOptions>���� �������� v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·e13351f28add7c60853cb3aac0a0e34e�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�4type..hash."".KeyValuePair��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|runtime.strhash�runtime.strhash@@ +"".autotmp_1823type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*"".KeyValuePair@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�0type..eq."".KeyValuePair��eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH����H�H�CH9���H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t}H�\$PH��tnH�SH�CH�\$XH��tWH�sH�KH9�u@H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ��t +�D$hH��H��D$hH��HÉ륉��D$hH��HÉ�,�������� + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring@�"".autotmp_1827type.string"".autotmp_1826type.string"".autotmp_1825?type.string"".autotmp_1824type.string "".~r30type.bool"".s type.uintptr"".q*type.*"".KeyValuePair"".p*type.*"".KeyValuePair2���� ������ v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�2type..hash."".PortBinding��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|runtime.strhash�runtime.strhash@@ +"".autotmp_1829type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p(type.*"".PortBinding@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�.type..eq."".PortBinding��eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH����H�H�CH9���H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t}H�\$PH��tnH�SH�CH�\$XH��tWH�sH�KH9�u@H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ��t +�D$hH��H��D$hH��HÉ륉��D$hH��HÉ�,�������� + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring@�"".autotmp_1833type.string"".autotmp_1832type.string"".autotmp_1831?type.string"".autotmp_1830type.string "".~r30type.bool"".s type.uintptr"".q(type.*"".PortBinding"".p(type.*"".PortBinding2���� ������ v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�6type..hash."".RestartPolicy��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|runtime.strhash�runtime.memhash@@ +"".autotmp_1835type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p,type.*"".RestartPolicy@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�2type..eq."".RestartPolicy��eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH��twH�H�CH9�uaH�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t+H�l$PH�]L�D$XI�hH9�t +�D$hH��H��D$hH��H��D$hH��HÉ녉�l��� + 0runtime.morestack_noctxt� runtime.eqstring@� "".autotmp_1837?type.string"".autotmp_1836type.string "".~r30type.bool"".s type.uintptr"".q,type.*"".RestartPolicy"".p,type.*"".RestartPolicy2���� �� ���� +nRTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�(type..hash."".Device��eH� %H;aw���H�� H�\$(H�$H�<$��H�D$H�\$8H�\$�H�D$H�\$(H�$H�<$tgH�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$ H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�W��� + + 0runtime.morestack_noctxt�runtime.strhash�runtime.strhash�runtime.strhash@@ "".autotmp_1840type.uintptr"".autotmp_1839type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*"".Device@�?@'�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�$type..eq."".Device��eH� %H;aw���H��HH�\$PH���sH�3H�KH�\$XH���VH�H�CH9��<H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ���H�\$PH����H�SH�CH�\$XH����H�sH�KH9���H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ��t}H�\$PH��tnH�s H�K(H�\$XH��tWH�S H�C(H9�u@H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t +�D$hH��H��D$hH��HÉ륉��D$hH��HÉ�+����� ����D$hH��HÉ��������� + + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring� runtime.eqstring@�"".autotmp_1846type.string"".autotmp_1845type.string"".autotmp_1844type.string"".autotmp_1843type.string"".autotmp_1842?type.string"".autotmp_1841type.string "".~r30type.bool"".s type.uintptr"".qtype.*"".Device"".ptype.*"".Device>���� �������� v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�(type..hash."".ULimit��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|runtime.strhash�runtime.memhash@@ +"".autotmp_1848type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*"".ULimit@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�$type..eq."".ULimit��eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH����H�H�CH9�uxH�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$�H�L$PH�D$X�\$ ��t8H�YH�hH9�t +�D$hH��H�H�YH�hH9�t +�D$hH��H��D$hH��H��D$hH��HÉ�k�����N��� + 0runtime.morestack_noctxt� runtime.eqstring@� "".autotmp_1850?type.string"".autotmp_1849type.string "".~r30type.bool"".s type.uintptr"".qtype.*"".ULimit"".ptype.*"".ULimit>������ �� ���� +rnTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�Htype..hash."".CreateContainerOptions��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|runtime.strhash�runtime.memhash@@ +"".autotmp_1852type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p>type.*"".CreateContainerOptions@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�Dtype..eq."".CreateContainerOptions��eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH����H�H�CH9�uxH�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$�H�L$PH�D$X�\$ ��t8H�YH�hH9�t +�D$hH��H�H�YH�hH9�t +�D$hH��H��D$hH��H��D$hH��HÉ�k�����N��� + 0runtime.morestack_noctxt� runtime.eqstring@� "".autotmp_1854?type.string"".autotmp_1853type.string "".~r30type.bool"".s type.uintptr"".q>type.*"".CreateContainerOptions"".p>type.*"".CreateContainerOptions>������ �� ���� +rnTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�&type..hash."".State��eH� %H;aw���H�� H�\$(H�$H�<$�cH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$�&H�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$��H�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$��H�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$tgH�$(H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$@H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�R����%�����%������%���� + 0runtime.morestack_noctxt�runtime.memhash�runtime.memhash�runtime.memhash�runtime.strhash�(type..hash.time.Time�(type..hash.time.Time@@"".autotmp_1860type.uintptr"".autotmp_1859type.uintptr"".autotmp_1858type.uintptr"".autotmp_1857type.uintptr"".autotmp_1856type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*"".State@�?@E�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�"type..eq."".State��eH� %H;aw���H��HH�\$PH�$H�<$��H�\$XH�\$H�|$��H�D$�H�\$PH�$H�<$�H�\$XH�\$H�|$�]H�D$�H�L$PH�T$X�\$��u +�D$hH��H�H�YH�jH9�t +�D$hH��H�H�YH�jH9�t +�D$hH��H�H�qH�I H��H�RH�C H9���H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$�H�t$PH�T$X�\$ ����H��H����H��(H��H��ttH��(H�H�(H9�u[�Y�h9�uQH�YH�hH9�uDH��H��@H��H��@H�H�)H9�u!�X�i9�uH�XH�iH9�u +�D$hH��H��D$hH��H��D$hH��HÉ눉�t����D$hH��HÉ%�����%�u����%�G����%�%��� + + 0runtime.morestack_noctxt�$runtime.memequal32�$runtime.memequal32� runtime.eqstring@�"".autotmp_1867type.*time.Time"".autotmp_1866type.*time.Time"".autotmp_1863?type.string"".autotmp_1862type.string "".~r30type.bool"".s type.uintptr"".qtype.*"".State"".ptype.*"".Stated����������� �� ����7�� M�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�*type..hash."".APIPort��eH� %H;aw���H�� H�\$(H�$H�<$��H�D$H�\$8H�\$�H�D$H�\$(H�$H�<$tgH�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$ H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�W��� + + 0runtime.morestack_noctxt�runtime.memhash�runtime.strhash�runtime.strhash@@ "".autotmp_1870type.uintptr"".autotmp_1869type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p type.*"".APIPort@�?@'�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�&type..eq."".APIPort��eH� %H;aw���H��HH�L$PH�D$XH�H�(H9�t +�D$hH��H�H�YH�hH9�t +�D$hH��H�H�qH�IH�PH�@H9���H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t}H�\$PH��tnH�S H�C(H�\$XH��tWH�s H�K(H9�u@H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ��t +�D$hH��H��D$hH��HÉ륉��D$hH��H� + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring@�"".autotmp_1874type.string"".autotmp_1873type.string"".autotmp_1872?type.string"".autotmp_1871type.string "".~r30type.bool"".s type.uintptr"".q type.*"".APIPort"".p type.*"".APIPortD�������� ��� ����Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�4type..hash.[8]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_1877type.int"".autotmp_1876type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[8]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�0type..eq.[8]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_1881?"type.interface {}"".autotmp_1880"type.interface {}"".autotmp_1879_type.int"".autotmp_1878Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[8]interface {}"".p*type.*[8]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�,type..hash."".Endpoint��eH� %H;aw���H�� H�\$(H�$H�<$��H�D$H�\$8H�\$�H�D$H�\$(H�$H�<$tgH�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$ H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�W��� + + 0runtime.morestack_noctxt�runtime.strhash�runtime.strhash�runtime.strhash@@ "".autotmp_1884type.uintptr"".autotmp_1883type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p"type.*"".Endpoint@�?@'�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�(type..eq."".Endpoint��eH� %H;aw���H��HH�\$PH���sH�3H�KH�\$XH���VH�H�CH9��<H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ���H�\$PH����H�SH�CH�\$XH����H�sH�KH9���H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ��t}H�\$PH��tnH�s H�K(H�\$XH��tWH�S H�C(H9�u@H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t +�D$hH��H��D$hH��HÉ륉��D$hH��HÉ�+����� ����D$hH��HÉ��������� + + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring� runtime.eqstring@�"".autotmp_1890type.string"".autotmp_1889type.string"".autotmp_1888type.string"".autotmp_1887type.string"".autotmp_1886?type.string"".autotmp_1885type.string "".~r30type.bool"".s type.uintptr"".q"type.*"".Endpoint"".p"type.*"".Endpoint>���� �������� v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�Htype..hash."".ExportContainerOptions��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|runtime.strhash�"runtime.interhash@@ +"".autotmp_1892type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p>type.*"".ExportContainerOptions@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�Dtype..eq."".ExportContainerOptions��eH� %H;aw���H��hH�\$pH����H�3H�KH�\$xH����H�H�CH9���H�t$XH�4$H�L$`H�L$H�T$HH�T$H�D$PH�D$��\$ ����H�\$xH��ttH�KH�sH�\$pH��t]H�CH�SH9�uCH�D$(H�$H�T$0H�T$H�L$8H�L$H�t$@H�t$��\$ ��t Ƅ$�H��h�Ƅ$�H��hÉ량�Ƅ$�H��hÉ��������� + 0runtime.morestack_noctxt� runtime.eqstring�runtime.ifaceeq@�"".autotmp_1896type.io.Writer"".autotmp_1895_type.io.Writer"".autotmp_1894?type.string"".autotmp_1893type.string "".~r30type.bool"".s type.uintptr"".q>type.*"".ExportContainerOptions"".p>type.*"".ExportContainerOptions2���� ������ v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·e13351f28add7c60853cb3aac0a0e34e�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�@type..hash."".ExportImageOptions��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|runtime.strhash�"runtime.interhash@@ +"".autotmp_1898type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p6type.*"".ExportImageOptions@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�H��$��]PL��$�A�hP@8�tƄ$�H�Ĉ�Ƅ$�H�Ĉ�Ƅ$�H�ĈÉ�h�����G���Ƅ$�H�ĈÉ����������Ƅ$�H�ĈÉ�;��������Ƅ$�H�ĈÉ���������Ƅ$�H�ĈÉ� ��������� +*0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring� runtime.eqstring�runtime.ifaceeq� runtime.ifaceeq@�"".autotmp_1918�type.io.Writer"".autotmp_1917�type.io.Writer"".autotmp_1916type.io.Reader"".autotmp_1915_type.io.Reader"".autotmp_1914type.string"".autotmp_1913type.string"".autotmp_1912type.string"".autotmp_1911type.string"".autotmp_1910?type.string"".autotmp_1909type.string "".~r30type.bool"".s type.uintptr"".q6type.*"".ImportImageOptions"".p6type.*"".ImportImageOptionsb"��������������������Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·af3107c17ee1ab6f9f33230b5c7e3062�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�Dtype..hash."".KillContainerOptions��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|runtime.strhash�runtime.memhash@@ +"".autotmp_1920type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p:type.*"".KillContainerOptions@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�@type..eq."".KillContainerOptions��eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH��twH�H�CH9�uaH�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t+H�l$PH�]L�D$XI�hH9�t +�D$hH��H��D$hH��H��D$hH��HÉ녉�l��� + 0runtime.morestack_noctxt� runtime.eqstring@� "".autotmp_1922?type.string"".autotmp_1921type.string "".~r30type.bool"".s type.uintptr"".q:type.*"".KillContainerOptions"".p:type.*"".KillContainerOptions2���� �� ���� +nRTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�2type..hash."".LogsOptions��eH� %H;aw���H�� H�\$(H�$H�<$��H�D$H�\$8H�\$�H�D$H�\$(H�$H�<$��H�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$�hH�$ H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$�&H�$0H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$��H�$8H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$��H�$@H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$tgH�$HH�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$XH�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�R����%�����%������%�����%�J����%� ��� + 0runtime.morestack_noctxt�runtime.strhash�"runtime.interhash�"runtime.interhash�runtime.memhash�runtime.memhash�runtime.memhash�runtime.strhash�runtime.memhash@@"".autotmp_1930type.uintptr"".autotmp_1929type.uintptr"".autotmp_1928type.uintptr"".autotmp_1927type.uintptr"".autotmp_1926type.uintptr"".autotmp_1925type.uintptr"".autotmp_1924type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p(type.*"".LogsOptions@�?@i�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�.type..eq."".LogsOptions� � eH� %H;aw���H��hH�\$pH���$H�3H�KH�\$xH���H�H�CH9���H�t$XH�4$H�L$`H�L$H�T$HH�T$H�D$PH�D$��\$ ����H�\$xH����H�KH�sH�\$pH���|H�CH�SH9��^H�D$(H�$H�T$0H�T$H�L$8H�L$H�t$@H�t$��\$ ���$H�\$xH���H�K H�s(H�\$pH����H�C H�S(H9���H�D$(H�$H�T$0H�T$H�L$8H�L$H�t$@H�t$��\$ ����H�\$pH�$H�<$�xH�$0H�\$xH�\$H�|$�QH�D$0H�D$�H�\$pH�$H�<$�H�$0H�\$xH�\$H�|$��H�D$0H�D$�H�T$pH�L$x�\$��u Ƅ$�H��h�H�Z8H�i8H9�t Ƅ$�H��h��Z@�i@@8�t Ƅ$�H��h�H��H�RHH�CPH�qHH�IPH9�uhH�T$HH�$H�D$PH�D$H�t$XH�t$H�L$`H�L$��\$ ��t2H�l$p�]XL�D$xA�hX@8�t Ƅ$�H��h�Ƅ$�H��h�Ƅ$�H��hÉ%������%������%�����%�|���Ƅ$�H��hÉ� ���������Ƅ$�H��hÉ�}�����_���Ƅ$�H��hÉ����������� + 0runtime.morestack_noctxt� runtime.eqstring�runtime.ifaceeq�runtime.ifaceeq� runtime.memequal� runtime.memequal� + runtime.eqstring@�"".autotmp_1939type.string"".autotmp_1938type.string"".autotmp_1937type.io.Writer"".autotmp_1936type.io.Writer"".autotmp_1935type.io.Writer"".autotmp_1934_type.io.Writer"".autotmp_1933?type.string"".autotmp_1932type.string "".~r30type.bool"".s type.uintptr"".q(type.*"".LogsOptions"".p(type.*"".LogsOptionsz��������r�� �� ��<�������� v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·e13351f28add7c60853cb3aac0a0e34e�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�type.*"".RemoveContainerOptions@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�Dtype..eq."".RemoveContainerOptions��eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH����H�H�CH9�uxH�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$�H�L$PH�D$X�\$ ��t8�Y�h@8�t +�D$hH��H��Y�h@8�t +�D$hH��H��D$hH��H��D$hH��HÉ�k�����N��� + 0runtime.morestack_noctxt� runtime.eqstring@� "".autotmp_1969?type.string"".autotmp_1968type.string "".~r30type.bool"".s type.uintptr"".q>type.*"".RemoveContainerOptions"".p>type.*"".RemoveContainerOptions>������ �� ���� +rnTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�Htype..hash."".RenameContainerOptions��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|runtime.strhash�runtime.strhash@@ +"".autotmp_1971type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p>type.*"".RenameContainerOptions@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�Dtype..eq."".RenameContainerOptions��eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH����H�H�CH9���H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t}H�\$PH��tnH�SH�CH�\$XH��tWH�sH�KH9�u@H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ��t +�D$hH��H��D$hH��HÉ륉��D$hH��HÉ�,�������� + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring@�"".autotmp_1975type.string"".autotmp_1974type.string"".autotmp_1973?type.string"".autotmp_1972type.string "".~r30type.bool"".s type.uintptr"".q>type.*"".RenameContainerOptions"".p>type.*"".RenameContainerOptions2���� ������ v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�8type..hash."".APIImageSearch��eH� %H;aw���H�� H�\$(H�$H�<$��H�D$H�\$8H�\$�H�D$H�\$(H�$H�<$��H�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$tgH�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$(H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�R����%���� + 0runtime.morestack_noctxt�runtime.strhash�runtime.memhash�runtime.strhash�runtime.memhash@@"".autotmp_1979type.uintptr"".autotmp_1978type.uintptr"".autotmp_1977type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p.type.*"".APIImageSearch@�?@1�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�4type..eq."".APIImageSearch��eH� %H;aw���H��HH�\$PH���)H�3H�KH�\$XH��� H�H�CH9���H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$�H�D$PH�L$X�\$ �����X�i@8�t +�D$hH��H��X�i@8�t +�D$hH��H�H�PH�@ H�qH�I H9�uaH�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ��t+H�l$PH�](L�D$XI�h(H9�t +�D$hH��H��D$hH��H��D$hH��H��D$hH��HÉ����������� + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring@�"".autotmp_1983type.string"".autotmp_1982type.string"".autotmp_1981?type.string"".autotmp_1980type.string "".~r30type.bool"".s type.uintptr"".q.type.*"".APIImageSearch"".p.type.*"".APIImageSearchV������k�� �� �� ���� v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go����� �� ������ v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�.type..hash."".doOptions��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|(runtime.nilinterhash�runtime.memhash@@ +"".autotmp_2015type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p$type.*"".doOptions@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�*type..eq."".doOptions��eH� %H;aw���H��HH�\$XH����H� H�sH�\$PH��txH�H�SH9�ubH�D$(H�$H�T$0H�T$H�L$8H�L$H�t$@H�t$��\$ ��t,H�l$P�]L�D$XA�h@8�t +�D$hH��H��D$hH��H��D$hH��HÉ넉�k��� + 0runtime.morestack_noctxt�runtime.efaceeq@� "".autotmp_2017?"type.interface {}"".autotmp_2016"type.interface {} "".~r30type.bool"".s type.uintptr"".q$type.*"".doOptions"".p$type.*"".doOptions2���� �� �� �� +nRTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�6type..hash."".hijackOptions��eH� %H;aw���H�� H�\$(H�$H�<$�!H�D$ H�\$8H�\$�H�D$H�\$(H�$H�<$��H�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$��H�$ H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$tgH�$0H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$@H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�R����%�����%����� + 0runtime.morestack_noctxt�runtime.memhash�"runtime.interhash�"runtime.interhash�"runtime.interhash�(runtime.nilinterhash@@"".autotmp_2022type.uintptr"".autotmp_2021type.uintptr"".autotmp_2020type.uintptr"".autotmp_2019type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p,type.*"".hijackOptions@�?@;�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�2type..eq."".hijackOptions� +� +eH� %H�D$�H;Aw���H��H��$�H��$�H�H�(H9�tƄ$�H�Ĉ��Z�h@8�tƄ$�H�Ĉ�H�HH�pH�BH�RH9���H�D$hH�$H�T$pH�T$H�L$xH�L$H��$�H�t$��\$ ����H��$�H����H�K H�s(H��$�H����H�C H�S(H9��aH�D$HH�$H�T$PH�T$H�L$XH�L$H�t$`H�t$��\$ ���'H��$�H���H�K0H�s8H��$�H����H�C0H�S8H9���H�D$HH�$H�T$PH�T$H�L$XH�L$H�t$`H�t$��\$ ����H��$�H��t}H�K@H�sHH��$�H��tcH�C@H�SHH9�uFH�D$(H�$H�T$0H�T$H�L$8H�L$H�t$@H�t$��\$ ��tƄ$�H�Ĉ�Ƅ$�H�ĈÉ뙉�|���Ƅ$�H�ĈÉ� ���������Ƅ$�H�ĈÉ�w�����V���Ƅ$�H�Ĉ� +*0runtime.morestack_noctxt�runtime.ifaceeq�runtime.ifaceeq�runtime.ifaceeq�runtime.efaceeq@�"".autotmp_2030�"type.interface {}"".autotmp_2029�"type.interface {}"".autotmp_2028type.io.Writer"".autotmp_2027type.io.Writer"".autotmp_2026type.io.Writer"".autotmp_2025_type.io.Writer"".autotmp_2024?type.io.Reader"".autotmp_2023type.io.Reader "".~r30type.bool"".s type.uintptr"".q,type.*"".hijackOptions"".p,type.*"".hijackOptions\"�*������������������Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·af3107c17ee1ab6f9f33230b5c7e3062�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�4type..hash.[2]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_2033type.int"".autotmp_2032type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[2]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�0type..eq.[2]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_2037?"type.interface {}"".autotmp_2036"type.interface {}"".autotmp_2035_type.int"".autotmp_2034Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[2]interface {}"".p*type.*[2]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�4type..hash.[1]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_2040type.int"".autotmp_2039type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[1]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�0type..eq.[1]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_2044?"type.interface {}"".autotmp_2043"type.interface {}"".autotmp_2042_type.int"".autotmp_2041Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[1]interface {}"".p*type.*[1]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�.type..hash."".dockerEnv��eH� %H;aw���H�� H�\$(H�$H�<$��H�D$H�\$8H�\$�H�D$H�\$(H�$H�<$tgH�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�W��� + + 0runtime.morestack_noctxt�runtime.strhash�runtime.memhash�runtime.strhash@@ "".autotmp_2047type.uintptr"".autotmp_2046type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p$type.*"".dockerEnv@�?@'�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�*type..eq."".dockerEnv��eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH����H�H�CH9���H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$�H�D$PH�L$X�\$ ��tv�X�i@8�t +�D$hH��H�H�PH�@ H�qH�I H9�u@H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ��t +�D$hH��H��D$hH��H��D$hH��HÉ�)����� ��� + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring@�"".autotmp_2051type.string"".autotmp_2050type.string"".autotmp_2049?type.string"".autotmp_2048type.string "".~r30type.bool"".s type.uintptr"".q$type.*"".dockerEnv"".p$type.*"".dockerEnv>����T�� �� ���� v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�&type..hash."".Error��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|runtime.memhash�runtime.strhash@@ +"".autotmp_2053type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*"".Error@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�"type..eq."".Error��eH� %H;aw���H��HH�L$PH�D$XH�H�(H9�t +�D$hH��H�H�qH�IH�PH�@H9�u@H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t +�D$hH��H��D$hH��H� + 0runtime.morestack_noctxt� runtime.eqstring@� "".autotmp_2055?type.string"".autotmp_2054type.string "".~r30type.bool"".s type.uintptr"".qtype.*"".Error"".ptype.*"".Error*���T�� � �� +u+Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�2type..hash."".jsonMessage��eH� %H;aw���H�� H�\$(H�$H�<$��H�D$H�\$8H�\$�H�D$H�\$(H�$H�<$��H�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$tgH�$ H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$0H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�R����%���� + 0runtime.morestack_noctxt�runtime.strhash�runtime.strhash�runtime.strhash�runtime.strhash@@"".autotmp_2059type.uintptr"".autotmp_2058type.uintptr"".autotmp_2057type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p(type.*"".jsonMessage@�?@1�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�.type..eq."".jsonMessage��eH� %H;aw���H��HH�\$PH����H�3H�KH�\$XH����H�H�CH9���H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ����H�\$PH���uH�SH�CH�\$XH���WH�sH�KH9��<H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ���H�\$PH����H�s H�K(H�\$XH����H�S H�C(H9���H�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t}H�\$PH��tnH�S0H�C8H�\$XH��tWH�s0H�K8H9�u@H�T$(H�$H�D$0H�D$H�t$8H�t$H�L$@H�L$��\$ ��t +�D$hH��H��D$hH��HÉ륉��D$hH��HÉ�+����� ����D$hH��HÉ����������D$hH��HÉ���������� + 0runtime.morestack_noctxt� runtime.eqstring� runtime.eqstring� runtime.eqstring� runtime.eqstring@�"".autotmp_2067type.string"".autotmp_2066type.string"".autotmp_2065type.string"".autotmp_2064type.string"".autotmp_2063type.string"".autotmp_2062type.string"".autotmp_2061?type.string"".autotmp_2060type.string "".~r30type.bool"".s type.uintptr"".q(type.*"".jsonMessage"".p(type.*"".jsonMessageJ���� ���������� v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�4type..hash.[3]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_2070type.int"".autotmp_2069type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[3]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�0type..eq.[3]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_2074?"type.interface {}"".autotmp_2073"type.interface {}"".autotmp_2072_type.int"".autotmp_2071Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[3]interface {}"".p*type.*[3]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�:type..hash."".NoSuchContainer��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|runtime.strhash�"runtime.interhash@@ +"".autotmp_2076type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p0type.*"".NoSuchContainer@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�6type..eq."".NoSuchContainer��eH� %H;aw���H��hH�\$pH����H�3H�KH�\$xH����H�H�CH9���H�t$XH�4$H�L$`H�L$H�T$HH�T$H�D$PH�D$��\$ ����H�\$xH��ttH�KH�sH�\$pH��t]H�CH�SH9�uCH�D$(H�$H�T$0H�T$H�L$8H�L$H�t$@H�t$��\$ ��t Ƅ$�H��h�Ƅ$�H��hÉ량�Ƅ$�H��hÉ��������� + 0runtime.morestack_noctxt� runtime.eqstring�runtime.ifaceeq@�"".autotmp_2080type.error"".autotmp_2079_type.error"".autotmp_2078?type.string"".autotmp_2077type.string "".~r30type.bool"".s type.uintptr"".q0type.*"".NoSuchContainer"".p0type.*"".NoSuchContainer2���� ������ v�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·e13351f28add7c60853cb3aac0a0e34e�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�(type..hash.[1]string��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�runtime.strhash@` "".autotmp_2083type.int"".autotmp_2082type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*[1]string`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�$type..eq.[1]string��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$`H����H��Hk�H�H�3H�KH�\$hH��tvH��Hk�H�H�H�CH9�uVH�t$HH�4$H�L$PH�L$H�T$8H�T$H�D$@H�D$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt� runtime.eqstring@�"".autotmp_2087?type.string"".autotmp_2086type.string"".autotmp_2085_type.int"".autotmp_2084Otype.int "".~r30type.bool"".s type.uintptr"".qtype.*[1]string"".ptype.*[1]string&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�\go.interface { CloseWrite() error }.CloseWrite��eH� %H;aw���H��H�Y H��t H�|$ H9;uH�#H�D$0H�D$8H�\$(H�$H�\$ H�[ ��H�L$H�D$H�L$0H�D$8H��� + 0runtime.morestack_noctxt� +@0 "".~r0 type.error""..thisJtype.interface { CloseWrite() error }0T/p$p +TTgclocals·78fd77a07ab543a063c3a3049973febeTgclocals·3280bececceccd33cb74587feedb1f9f�4type..hash."".tlsClientCon��eH� %H;aw���H�� H�\$(H�$H�<$tbH�D$H�\$8H�\$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%� + 0runtime.morestack_noctxt|runtime.memhash�"runtime.interhash@@ +"".autotmp_2090type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*"".tlsClientCon@h?@�� +=cTgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�0type..eq."".tlsClientCon��eH� %H;aw���H��HH�T$PH�D$XH�H�(H9�t +�D$hH��H�H�HH�pH�BH�RH9�u@H�D$(H�$H�T$0H�T$H�L$8H�L$H�t$@H�t$��\$ ��t +�D$hH��H��D$hH��H� + 0runtime.morestack_noctxt�runtime.ifaceeq@� "".autotmp_2092?type.net.Conn"".autotmp_2091type.net.Conn "".~r30type.bool"".s type.uintptr"".q*type.*"".tlsClientCon"".p*type.*"".tlsClientCon*���T�� � �� +u+Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go�0"".(*tlsClientCon).Close`HH�D$H�D$H�\$H�+H�l$�@0crypto/tls.(*Conn).Close0 "".~r1type.error""..this*type.*"".tlsClientCon00&0Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�*"".tlsClientCon.Close��eH� %H;aw���H��H�Y H��t H�|$ H9;uH�#H�D$8H�D$@H�\$ H�$�H�L$H�D$H�L$8H�D$@H��� + 0runtime.morestack_noctxt�0crypto/tls.(*Conn).CloseP0 "".~r10type.error""..this(type."".tlsClientCon0N/p(p +K%Tgclocals·57e1009a600f832f844e0e3c49ba5a89Tgclocals·3280bececceccd33cb74587feedb1f9f�D"".(*tlsClientCon).ConnectionState@type.crypto/tls.ConnectionState""..this*type.*"".tlsClientCon  * Tgclocals·40de35fb9b773b345d1ee7cba691ea13Tgclocals·3280bececceccd33cb74587feedb1f9f�>"".tlsClientCon.ConnectionState��eH� %H�D$�H;Aw���H���H�Y H��tH��$H9;uH�#H��$1��H��$H�$�H�\$H��$�H��H���H��$�H��$H��H���H���� +*0runtime.morestack_noctxt�� runtime.duffzero�Dcrypto/tls.(*Conn).ConnectionState�� runtime.duffcopy�� runtime.duffcopy��"".autotmp_2094�>type.crypto/tls.ConnectionState "".~r10>type.crypto/tls.ConnectionState""..this(type."".tlsClientCon"�s� �,� +VJTgclocals·25609300e15c97db07af80faee4d2fd6Tgclocals·b0f264e78fa38c77ad79fe8a353279f7�8"".(*tlsClientCon).Handshake`HH�D$H�D$H�\$H�+H�l$�@8crypto/tls.(*Conn).Handshake0 "".~r1type.error""..this*type.*"".tlsClientCon00.0Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�2"".tlsClientCon.Handshake��eH� %H;aw���H��H�Y H��t H�|$ H9;uH�#H�D$8H�D$@H�\$ H�$�H�L$H�D$H�L$8H�D$@H��� + 0runtime.morestack_noctxt�8crypto/tls.(*Conn).HandshakeP0 "".~r10type.error""..this(type."".tlsClientCon0N/p0p +K%Tgclocals·57e1009a600f832f844e0e3c49ba5a89Tgclocals·3280bececceccd33cb74587feedb1f9f�8"".(*tlsClientCon).LocalAddr`HH�D$H�D$H�\$H�+H�l$�@8crypto/tls.(*Conn).LocalAddr0 "".~r1type.net.Addr""..this*type.*"".tlsClientCon0020Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�2"".tlsClientCon.LocalAddr��eH� %H;aw���H��H�Y H��t H�|$ H9;uH�#H�D$8H�D$@H�\$ H�$�H�L$H�D$H�L$8H�D$@H��� + 0runtime.morestack_noctxt�8crypto/tls.(*Conn).LocalAddrP0 "".~r10type.net.Addr""..this(type."".tlsClientCon0N/p4p +K%Tgclocals·57e1009a600f832f844e0e3c49ba5a89Tgclocals·3280bececceccd33cb74587feedb1f9f�>"".(*tlsClientCon).OCSPResponse`ZH�D$H�D$H�D$ H�\$H�+H�l$�R>crypto/tls.(*Conn).OCSPResponse@ "".~r1type.[]uint8""..this*type.*"".tlsClientCon0060Tgclocals·13d3af77a5bf02af6db4588efb2ea811Tgclocals·3280bececceccd33cb74587feedb1f9f�8"".tlsClientCon.OCSPResponse��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�D$@H�D$HH�D$PH�\$(H�$�H�T$H�L$H�D$H�T$@H�L$HH�D$PH�� � + 0runtime.morestack_noctxt�>crypto/tls.(*Conn).OCSPResponse`@ "".~r10type.[]uint8""..this(type."".tlsClientCon@a?�8� +T,Tgclocals·da455f41cf2a78c8890074a4a256bdd4Tgclocals·3280bececceccd33cb74587feedb1f9f�."".(*tlsClientCon).Read`ZH�D$0H�D$8H�D$(H�\$H�+H�l$�R.crypto/tls.(*Conn).Readp"crypto/tls.err·2Ptype.errorcrypto/tls.n·1@type.intcrypto/tls.b·4type.[]uint8""..this*type.*"".tlsClientCon00:0Tgclocals·9877a4ef732a0f966b889793f9b99b87Tgclocals·3280bececceccd33cb74587feedb1f9f�("".tlsClientCon.Read��eH� %H;aw���H��8H�Y H��t H�|$@H9;uH�#H�D$xHDŽ$�H�\$@H�$H�\$XH�\$H�\$`H�\$H�\$hH�\$�H�T$ H�L$(H�D$0H�T$pH�L$xH��$�H��8� + 0runtime.morestack_noctxt�.crypto/tls.(*Conn).Read�p"crypto/tls.err·2ptype.errorcrypto/tls.n·1`type.intcrypto/tls.b·40type.[]uint8""..this(type."".tlsClientConp|o +�<� +l4Tgclocals·0273bd9c87bb10f67d516fbf00fd7767Tgclocals·3280bececceccd33cb74587feedb1f9f�:"".(*tlsClientCon).RemoteAddr`HH�D$H�D$H�\$H�+H�l$�@:crypto/tls.(*Conn).RemoteAddr0 "".~r1type.net.Addr""..this*type.*"".tlsClientCon00>0Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�4"".tlsClientCon.RemoteAddr��eH� %H;aw���H��H�Y H��t H�|$ H9;uH�#H�D$8H�D$@H�\$ H�$�H�L$H�D$H�L$8H�D$@H��� + 0runtime.morestack_noctxt�:crypto/tls.(*Conn).RemoteAddrP0 "".~r10type.net.Addr""..this(type."".tlsClientCon0N/p@p +K%Tgclocals·57e1009a600f832f844e0e3c49ba5a89Tgclocals·3280bececceccd33cb74587feedb1f9f�<"".(*tlsClientCon).SetDeadline`HH�D$(H�D$0H�\$H�+H�l$�@�6"".tlsClientCon.SetDeadline��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$hH�D$pH�\$8H�$H�\$PH�\$�\$X�\$H�\$`H�\$�H�L$ H�D$(H�L$hH�D$pH��0� + 0runtime.morestack_noctxt��D"".(*tlsClientCon).SetReadDeadline`HH�D$(H�D$0H�\$H�+H�l$�@Dcrypto/tls.(*Conn).SetReadDeadline` "".~r2@type.errorcrypto/tls.t·3type.time.Time""..this*type.*"".tlsClientCon00F0Tgclocals·86b4418f46455e3a0eb577619691d10fTgclocals·3280bececceccd33cb74587feedb1f9f�>"".tlsClientCon.SetReadDeadline��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$hH�D$pH�\$8H�$H�\$PH�\$�\$X�\$H�\$`H�\$�H�L$ H�D$(H�L$hH�D$pH��0� + 0runtime.morestack_noctxt�Dcrypto/tls.(*Conn).SetReadDeadline�` "".~r2`type.errorcrypto/tls.t·30type.time.Time""..this(type."".tlsClientCon`j_ �H� +g)Tgclocals·be9b149192cd561578dd28b30f28e84fTgclocals·3280bececceccd33cb74587feedb1f9f�F"".(*tlsClientCon).SetWriteDeadline`HH�D$(H�D$0H�\$H�+H�l$�@Fcrypto/tls.(*Conn).SetWriteDeadline` "".~r2@type.errorcrypto/tls.t·3type.time.Time""..this*type.*"".tlsClientCon00J0Tgclocals·86b4418f46455e3a0eb577619691d10fTgclocals·3280bececceccd33cb74587feedb1f9f�@"".tlsClientCon.SetWriteDeadline��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$hH�D$pH�\$8H�$H�\$PH�\$�\$X�\$H�\$`H�\$�H�L$ H�D$(H�L$hH�D$pH��0� + 0runtime.morestack_noctxt�Fcrypto/tls.(*Conn).SetWriteDeadline�` "".~r2`type.errorcrypto/tls.t·30type.time.Time""..this(type."".tlsClientCon`j_ �L� +g)Tgclocals·be9b149192cd561578dd28b30f28e84fTgclocals·3280bececceccd33cb74587feedb1f9f�B"".(*tlsClientCon).VerifyHostname`HH�D$ H�D$(H�\$H�+H�l$�@Bcrypto/tls.(*Conn).VerifyHostnameP "".~r20type.error$crypto/tls.host·3type.string""..this*type.*"".tlsClientCon00N0Tgclocals·14c45952157723c8762210d9c661bf29Tgclocals·3280bececceccd33cb74587feedb1f9f�<"".tlsClientCon.VerifyHostname��eH� %H;aw���H��(H�Y H��t H�|$0H9;uH�#H�D$XH�D$`H�\$0H�$H�\$HH�\$H�\$PH�\$�H�L$H�D$ H�L$XH�D$`H��(� + 0runtime.morestack_noctxt�Bcrypto/tls.(*Conn).VerifyHostnamepP "".~r2Ptype.error$crypto/tls.host·30type.string""..this(type."".tlsClientConPbO�P� +_!Tgclocals·be4f16eacaf744756abcb34364e01385Tgclocals·3280bececceccd33cb74587feedb1f9f�0"".(*tlsClientCon).Write`ZH�D$(H�D$0H�D$8H�\$H�+H�l$�R0crypto/tls.(*Conn).Writep "".~r3Ptype.error "".~r2@type.intcrypto/tls.b·4type.[]uint8""..this*type.*"".tlsClientCon00R0Tgclocals·9877a4ef732a0f966b889793f9b99b87Tgclocals·3280bececceccd33cb74587feedb1f9f�*"".tlsClientCon.Write��eH� %H;aw���H��8H�Y H��t H�|$@H9;uH�#H�D$xHDŽ$�H�\$@H�$H�\$XH�\$H�\$`H�\$H�\$hH�\$�H�T$ H�L$(H�D$0H�T$pH�L$xH��$�H��8� + 0runtime.morestack_noctxt�0crypto/tls.(*Conn).Write�p "".~r3ptype.error "".~r2`type.intcrypto/tls.b·40type.[]uint8""..this(type."".tlsClientConp|o +�T� +l4Tgclocals·0273bd9c87bb10f67d516fbf00fd7767Tgclocals·3280bececceccd33cb74587feedb1f9f�Z"".(*tlsClientCon).crypto/tls.clientHandshake`HH�D$H�D$H�\$H�+H�l$�@Dcrypto/tls.(*Conn).clientHandshake0 "".~r1type.error""..this*type.*"".tlsClientCon00V0Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�T"".tlsClientCon.crypto/tls.clientHandshake��eH� %H;aw���H��H�Y H��t H�|$ H9;uH�#H�D$8H�D$@H�\$ H�$�H�L$H�D$H�L$8H�D$@H��� + 0runtime.morestack_noctxt�Dcrypto/tls.(*Conn).clientHandshakeP0 "".~r10type.error""..this(type."".tlsClientCon0N/pXp +K%Tgclocals·57e1009a600f832f844e0e3c49ba5a89Tgclocals·3280bececceccd33cb74587feedb1f9f�V"".(*tlsClientCon).crypto/tls.decryptTicket@@H�D$(�D$0H�\$H�+H�l$�8@crypto/tls.(*Conn).decryptTicket` "".~r3Ptype.bool "".~r2@:type.*crypto/tls.sessionState.crypto/tls.encrypted·4type.[]uint8""..this*type.*"".tlsClientCon  Z Tgclocals·9f0d5ba6770c4a1ed4fa771547e96df1Tgclocals·3280bececceccd33cb74587feedb1f9f�P"".tlsClientCon.crypto/tls.decryptTicket��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�\$8H�$H�\$PH�\$H�\$XH�\$H�\$`H�\$�H�L$ �\$(H�L$h�\$pH��0� + 0runtime.morestack_noctxt�@crypto/tls.(*Conn).decryptTicket�` "".~r3ptype.bool "".~r2`:type.*crypto/tls.sessionState.crypto/tls.encrypted·40type.[]uint8""..this(type."".tlsClientCon`Y_ �\� +W)Tgclocals·4e44481e9dee421443081e94ffaa0dd2Tgclocals·3280bececceccd33cb74587feedb1f9f�V"".(*tlsClientCon).crypto/tls.encryptTicket�~H�D$H�D$ H�D$(H�D$0H�D$8H�\$H�+H�l$�v@crypto/tls.(*Conn).encryptTicketp "".~r3Ptype.error "".~r2 type.[]uint8&crypto/tls.state·4:type.*crypto/tls.sessionState""..this*type.*"".tlsClientCon@@^@Tgclocals·9877a4ef732a0f966b889793f9b99b87Tgclocals·3280bececceccd33cb74587feedb1f9f�P"".tlsClientCon.crypto/tls.encryptTicket��eH� %H;aw���H��8H�Y H��t H�|$@H9;uH�#H�D$`H�D$hH�D$pH�D$xHDŽ$�H�\$@H�$H�\$XH�\$�H�t$H�l$H�T$ H�L$(H�D$0H�t$`H�l$hH�T$pH�L$xH��$�H��8� + 0runtime.morestack_noctxt�@crypto/tls.(*Conn).encryptTicket�p "".~r3ptype.error "".~r2@type.[]uint8&crypto/tls.state·40:type.*crypto/tls.sessionState""..this(type."".tlsClientConp�o�`� +sMTgclocals·0273bd9c87bb10f67d516fbf00fd7767Tgclocals·3280bececceccd33cb74587feedb1f9f�V"".(*tlsClientCon).crypto/tls.readHandshake�lH�D$H�D$H�D$ H�D$(H�\$H�+H�l$�d@crypto/tls.(*Conn).readHandshakeP "".~r20type.error "".~r1"type.interface {}""..this*type.*"".tlsClientCon@@b@Tgclocals·5dfce38b1d248a3900c6ec750de77702Tgclocals·3280bececceccd33cb74587feedb1f9f�P"".tlsClientCon.crypto/tls.readHandshake��eH� %H;aw���H��(H�Y H��t H�|$0H9;uH�#H�D$HH�D$PH�D$XH�D$`H�\$0H�$�H�l$H�T$H�L$H�D$ H�l$HH�T$PH�L$XH�D$`H��(� + 0runtime.morestack_noctxt�@crypto/tls.(*Conn).readHandshakepP "".~r2Ptype.error "".~r10"type.interface {}""..this(type."".tlsClientConPtO�d� +]3Tgclocals·d93d6c9fc85d7888b8b1832756680f45Tgclocals·3280bececceccd33cb74587feedb1f9f�P"".(*tlsClientCon).crypto/tls.readRecord`HH�D$H�D$ H�\$H�+H�l$�@:crypto/tls.(*Conn).readRecord@ "".~r2 type.error$crypto/tls.want·34type.crypto/tls.recordType""..this*type.*"".tlsClientCon00f0Tgclocals·6a2e5ab2d393a1bfd331903fbd0fd425Tgclocals·3280bececceccd33cb74587feedb1f9f�J"".tlsClientCon.crypto/tls.readRecord��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�D$HH�D$PH�\$(H�$�\$@�\$�H�L$H�D$H�L$HH�D$PH�� � + 0runtime.morestack_noctxt�:crypto/tls.(*Conn).readRecord`@ "".~r2@type.error$crypto/tls.want·304type.crypto/tls.recordType""..this(type."".tlsClientCon@W?�h� +T,Tgclocals·c776d40308d3cc87dab399555a94d3caTgclocals·3280bececceccd33cb74587feedb1f9f�N"".(*tlsClientCon).crypto/tls.sendAlert`HH�D$H�D$ H�\$H�+H�l$�@8crypto/tls.(*Conn).sendAlert@ "".~r2 type.error"crypto/tls.err·3*type.crypto/tls.alert""..this*type.*"".tlsClientCon00j0Tgclocals·6a2e5ab2d393a1bfd331903fbd0fd425Tgclocals·3280bececceccd33cb74587feedb1f9f�H"".tlsClientCon.crypto/tls.sendAlert��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�D$HH�D$PH�\$(H�$�\$@�\$�H�L$H�D$H�L$HH�D$PH�� � + 0runtime.morestack_noctxt�8crypto/tls.(*Conn).sendAlert`@ "".~r2@type.error"crypto/tls.err·30*type.crypto/tls.alert""..this(type."".tlsClientCon@W?�l� +T,Tgclocals·c776d40308d3cc87dab399555a94d3caTgclocals·3280bececceccd33cb74587feedb1f9f�Z"".(*tlsClientCon).crypto/tls.sendAlertLocked`HH�D$H�D$ H�\$H�+H�l$�@Dcrypto/tls.(*Conn).sendAlertLocked@ "".~r2 type.error"crypto/tls.err·3*type.crypto/tls.alert""..this*type.*"".tlsClientCon00n0Tgclocals·6a2e5ab2d393a1bfd331903fbd0fd425Tgclocals·3280bececceccd33cb74587feedb1f9f�T"".tlsClientCon.crypto/tls.sendAlertLocked��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�D$HH�D$PH�\$(H�$�\$@�\$�H�L$H�D$H�L$HH�D$PH�� � + 0runtime.morestack_noctxt�Dcrypto/tls.(*Conn).sendAlertLocked`@ "".~r2@type.error"crypto/tls.err·30*type.crypto/tls.alert""..this(type."".tlsClientCon@W?�p� +T,Tgclocals·c776d40308d3cc87dab399555a94d3caTgclocals·3280bececceccd33cb74587feedb1f9f�Z"".(*tlsClientCon).crypto/tls.serverHandshake`HH�D$H�D$H�\$H�+H�l$�@Dcrypto/tls.(*Conn).serverHandshake0 "".~r1type.error""..this*type.*"".tlsClientCon00r0Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�T"".tlsClientCon.crypto/tls.serverHandshake��eH� %H;aw���H��H�Y H��t H�|$ H9;uH�#H�D$8H�D$@H�\$ H�$�H�L$H�D$H�L$8H�D$@H��� + 0runtime.morestack_noctxt�Dcrypto/tls.(*Conn).serverHandshakeP0 "".~r10type.error""..this(type."".tlsClientCon0N/ptp +K%Tgclocals·57e1009a600f832f844e0e3c49ba5a89Tgclocals·3280bececceccd33cb74587feedb1f9f�X"".(*tlsClientCon).crypto/tls.tryCipherSuite@6H�D$8H�\$H�+H�l$�.Bcrypto/tls.(*Conn).tryCipherSuitep "".~r6`8type.*crypto/tls.cipherSuite*crypto/tls.ecdsaOk·7Vtype.bool0crypto/tls.ellipticOk·6Ttype.bool*crypto/tls.version·5Ptype.uint16Fcrypto/tls.supportedCipherSuites·4 type.[]uint16 crypto/tls.id·3type.uint16""..this*type.*"".tlsClientCon  v Tgclocals·a99c50f5f5d34b1bf54d8ece6dad05c2Tgclocals·3280bececceccd33cb74587feedb1f9f�R"".tlsClientCon.crypto/tls.tryCipherSuite��eH� %H;aw���H��8H�Y H��t H�|$@H9;uH�#H�\$@H�$H�\$Xf�\$H�\$`H�\$H�\$hH�\$H�\$pH�\$ H�\$xf�\$(�\$z�\$*�\${�\$+�H�\$0H��$�H��8� + 0runtime.morestack_noctxt�Bcrypto/tls.(*Conn).tryCipherSuite�p "".~r6�8type.*crypto/tls.cipherSuite*crypto/tls.ecdsaOk·7vtype.bool0crypto/tls.ellipticOk·6ttype.bool*crypto/tls.version·5ptype.uint16Fcrypto/tls.supportedCipherSuites·4@type.[]uint16 crypto/tls.id·30type.uint16""..this(type."".tlsClientConp{o �x� +!Tgclocals·adf7fd756b6e86afbfe88b4b789f56a2Tgclocals·3280bececceccd33cb74587feedb1f9f�R"".(*tlsClientCon).crypto/tls.writeRecord`ZH�D$8H�D$@H�D$0H�\$H�+H�l$�R�L"".tlsClientCon.crypto/tls.writeRecord��eH� %H;aw���H��@H�Y H��t H�|$HH9;uH�#HDŽ$�HDŽ$�H�\$HH�$�\$`�\$H�\$hH�\$H�\$pH�\$H�\$xH�\$ �H�T$(H�L$0H�D$8H��$�H��$�H��$�H��@� + 0runtime.morestack_noctxt��,4go.itab.*os.File.io.Reader� go.string."HOME"0*HOME go.string."HOME"�&go.string.".docker"00.docker &go.string.".docker"�.go.string."config.json"@8 config.json .go.string."config.json"�,go.string.".dockercfg"@6 +.dockercfg ,go.string.".dockercfg"�Tgclocals·2e2a9972ea9ced3a58f9e7510cf4914d��0� " "��  �Tgclocals·b8a8407971613b03b21a64dc1e56fba0PP�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·9edc1f6d8fc7336ae101b48cbf822a45 + �"go.string."auths"0,auths "go.string."auths"�Tgclocals·74b1ee12d224f81a4fda25605fab855d``$ �Tgclocals·e94084972e98c8fdf7f2203a35ca807a88 + �go.string.":"0$: go.string.":"�Tgclocals·1c0f8a36a8ada2462e7c582fc8286897��R ��V����V�����V�����V�Tgclocals·f565a1229afec041643831d3cd6a3b7dHH�.go.string."conf is nil"@8 conf is nil .go.string."conf is nil"� go.string."POST"0*POST go.string."POST"�"go.string."/auth"0,/auth "go.string."/auth"�>go.string."auth error (%d): %s"PHauth error (%d): %s >go.string."auth error (%d): %s"�Tgclocals·25ee8e11891a6b427c03a740dc761f96@@(����Tgclocals·b29a376724b9675f7c9e576a6dabc1e0(( + + +�go.string."C"0$C go.string."C"�go.string."A"0$A go.string."A"�go.string."D"0$D go.string."D"�"go.string."%s %s"0,%s %s "go.string."%s %s"�Tgclocals·0b4080736ceb8b2da6f0b7e8a876e6b8(("���Tgclocals·6d340c3bdac448a6ef1256f331f68dd3((�go.string."."0$. go.string."."�Lgo.string."Unable to parse version %q"`VUnable to parse version %q Lgo.string."Unable to parse version %q"�xgo.string."Unable to parse version %q: %q is not an integer"��0Unable to parse version %q: %q is not an integer xgo.string."Unable to parse version %q: %q is not an integer"�Tgclocals·979c84cf2ee7fa703a7cd5365c579635��>�< �< +�� �?��?�Tgclocals·56fad8922133a82d7e9abffb05067a58HH�Tgclocals·37f4150aca71c16b472a5e6f54a4a2bc((�Tgclocals·c45f1008acf31f9ce337f7dfa1fa0204(( +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·d83eab2a3f0aa562c88b153605ebed26��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·d83eab2a3f0aa562c88b153605ebed26��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·d83eab2a3f0aa562c88b153605ebed26��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·d83eab2a3f0aa562c88b153605ebed26��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·d83eab2a3f0aa562c88b153605ebed26��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·5dfce38b1d248a3900c6ec750de77702 +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·8e51ba8a606dfe7bf8ea610f35b1860a""�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a3afb5a83dcf14cc57a3d3da3be3a7df"�Tgclocals·63a71a9d82a0cb5094b44aef6b6fe396PP"*(�Tgclocals·677e212df4ff2dc5d1bd7207f0cb343fPP""""""""�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·c984d5bd78e9da313cca302adec9d408""�Tgclocals·f27fde19da2a9a9e0264e00d44cbb36a(( �Tgclocals·e11b7011fe7d18f281fa367784f98637((""""""�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·0528ab8f76149a707fd2f0025c2178a3�go.string."://"0(:// go.string."://"�hgo.string."could not split %s into two parts by ://"�r(could not split %s into two parts by :// hgo.string."could not split %s into two parts by ://"�,go.string."https://%s"@6 +https://%s ,go.string."https://%s"�(go.string."cert.pem"@2cert.pem (go.string."cert.pem"�&go.string."key.pem"00key.pem &go.string."key.pem"�$go.string."ca.pem"0.ca.pem $go.string."ca.pem"�Tgclocals·bae70cbfa95aa7f2f402b02d37b0b239�� <������  �� � �� � �Tgclocals·7df0f47b43308a447b1c5362b1e10571xx +�,Bgo.itab.*errors.errorString.error�,bgo.itab.*net/http.Transport.net/http.RoundTripper�Tgo.string."Both cert and key are required"`^Both cert and key are required Tgo.string."Both cert and key are required"�Hgo.string."Could not add RootCA pem"`RCould not add RootCA pem Hgo.string."Could not add RootCA pem"�Tgclocals·6eba1717ce5fb698f2f3dfc9fb10f9ab��`  � �  � � �����  +  +�  +�  +  + *   +  + + + +� +� +�Tgclocals·7d1b151141fa06142d6be12f40c65cbc�� "�"�"�"�"�"�"�"�"�"�"�"�"�"�"�"�"�"�"�"�"�"�"�"�"�"�"��Tgclocals·44e348188e22fef6300f71ab26e45197 ��Tgclocals·d7e8a62d22b1cde6d92b17a55c33fe8f �,.go.itab.*"".Error.error�$go.string."/_ping"0./_ping $go.string."/_ping"�go.string."GET"0(GET go.string."GET"�Tgclocals·de8e430848d9f174d000c8b092bf66b4((�Tgclocals·6d340c3bdac448a6ef1256f331f68dd3((�(go.string."/version"@2/version (go.string."/version"��go.string."Received unexpected status %d while trying to retrieve the server version"��IReceived unexpected status %d while trying to retrieve the server version �go.string."Received unexpected status %d while trying to retrieve the server version"�,go.string."ApiVersion"@6 +ApiVersion ,go.string."ApiVersion"�Tgclocals·3bb049eac63b5508d15152b6e410d69f88 < <��Tgclocals·31c26a3eb5003a6c37416d296e2bd48788 +�,>go.itab.*bytes.Buffer.io.Reader�,go.string."User-Agent"@6 +User-Agent ,go.string."User-Agent"�6go.string."go-dockerclient"@@go-dockerclient 6go.string."go-dockerclient"�0go.string."Content-Type"@: Content-Type 0go.string."Content-Type"�8go.string."application/json"PBapplication/json 8go.string."application/json"�,go.string."plain/text"@6 +plain/text ,go.string."plain/text"� go.string."unix"0*unix go.string."unix"�go.itab.*bytes.Reader.io.Reader�go.string."PUT"0(PUT go.string."PUT"�*go.string."%s %s\x0d"0.%s %s  *go.string."%s %s\x0d"�Tgclocals·147deda5d3defe8e1d522f194155c84f� � ����j�������  , � � � � � � �Tgclocals·7e331f181b2554581236d61d5561e53d�� ��������������������������������������������������������������������go.string."tcp"0(tcp go.string."tcp"�Tgclocals·65dfd25068bbba2abebc869f9ef9f7a5��T���������� +������������������������������*��*� *� *�*�(�������Tgclocals·84e82484f467e1dc08e5640e075b9b76��"���>�������������������������������������������������������>���>���>���>���>���>���>���>���>���>���>���>���>�go.string."/"0$/ go.string."/"�(go.string."%s/v%s%s"@2%s/v%s%s (go.string."%s/v%s%s"� go.string."%s%s"0*%s%s go.string."%s%s"�Tgclocals·514c3d378a44440bceb597de19ebfbf7pp." �?"�?" �"��Tgclocals·61dac2719f307a892a4a15123f2e6a2d@@ + + + + + + +�go.string."qs"0&qs go.string."qs"�go.string."-"0$- go.string."-"�Tgclocals·928ad969a3698656dfa33d91e2ca9cd1��Jhhh��JB +h�Tgclocals·c69849cba6bf70a13b3371331d258b5088�go.string."1"0$1 go.string."1"�Tgclocals·a157d5303e3a20a2392145ef25e4599b��x  �� �   �Tgclocals·b8c550e5e1ba1f11f1bc237b9d0f0dc8�� ��������������������Tgclocals·e1ae6533a9e39048ba0735a2264ce16a �Tgclocals·3e69739b44630d52358b28c7a0e238fa  + �go.string."/containers/create?"PH/containers/create? >go.string."/containers/create?"�Tgclocals·2ca41a02a2a5788f97a4be1897b36700pp*������Tgclocals·3f5a7d1842b14039f35be09ef67df5f8@@�������$go.string."always"0.always $go.string."always"�Tgclocals·0528ab8f76149a707fd2f0025c2178a3�Tgclocals·0528ab8f76149a707fd2f0025c2178a3�,go.string."on-failure"@6 +on-failure ,go.string."on-failure"�Tgclocals·0528ab8f76149a707fd2f0025c2178a3�Tgclocals·2d8f3a7439ca173dec4205ff264b0edc�go.string."no"0&no go.string."no"�Tgclocals·0528ab8f76149a707fd2f0025c2178a3�Tgclocals·0528ab8f76149a707fd2f0025c2178a3�,Rgo.itab.*"".ContainerAlreadyRunning.error�$go.string."/start"0./start $go.string."/start"�Tgclocals·1497b0fbec88b963d1dc5f4cf942151688  �Tgclocals·fc96ae191c2547955912928601e8595988 ������,Jgo.itab.*"".ContainerNotRunning.error�Hgo.string."/containers/%s/stop?t=%d"`R/containers/%s/stop?t=%d Hgo.string."/containers/%s/stop?t=%d"�Tgclocals·5c42a9dee0c88889a167a0d13b7c2026``(���Tgclocals·42785a4ae44025160cf24924f7d01efb88 JJJJJ�Ngo.string."/containers/%s/restart?t=%d"`X/containers/%s/restart?t=%d Ngo.string."/containers/%s/restart?t=%d"�Tgclocals·5bacfca50b7e6b4494d1c0d96e8b2c7cPP&��?��?�Tgclocals·1da38d5d89527cd2ab312249704d85d700 JJJJ�@go.string."/containers/%s/pause"PJ/containers/%s/pause @go.string."/containers/%s/pause"�Tgclocals·556e2b84f9ef2d507be121d828e30b96PP"�< �<�Tgclocals·fe0d626f6a1a9cb0d3493cb8c292091b00 + + + + +�Dgo.string."/containers/%s/unpause"PN/containers/%s/unpause Dgo.string."/containers/%s/unpause"�Tgclocals·556e2b84f9ef2d507be121d828e30b96PP"�< �<�Tgclocals·fe0d626f6a1a9cb0d3493cb8c292091b00 + + + + +�.go.string."?ps_args=%s"@8 ?ps_args=%s .go.string."?ps_args=%s"�@go.string."/containers/%s/top%s"PJ/containers/%s/top%s @go.string."/containers/%s/top%s"�Tgclocals·6a9f496e2cfbe0515ededb4c2d64a743��:    " �"� +�Tgclocals·950e6e6b9e7c3fe47672289f0a6f6e8bPP���������,@go.itab.*io.PipeReader.io.Reader�Tgclocals·1bc79a478470e6209b0ca20d87c54bd3��8 "* *"*"*"*@�"*@�@��" +@�� +@�� @� @��@� �@� @� @��@�@�@�@�,@��@��Tgclocals·378f3e900c220a5cad1989c9c06023bd�������������������������������������������������$go.string."/kill?"0./kill? $go.string."/kill?"�Tgclocals·1705812f15ec71868ae696027438a358(( �Tgclocals·bd92ef728a38faac78badef3588d832f(( JJJ�go.string."?"0$? go.string."?"�$go.string."DELETE"0.DELETE $go.string."DELETE"�Tgclocals·1705812f15ec71868ae696027438a358(( �Tgclocals·bd92ef728a38faac78badef3588d832f(( JJJ�>go.string."/containers/%s/copy"PH/containers/%s/copy >go.string."/containers/%s/copy"�Tgclocals·4e703ba17638508264f032b5f70033cd��<�� �++�Tgclocals·387212f77114c618c992aca1d7f6e2d3PP���������"go.string."/wait"0,/wait "go.string."/wait"�Tgclocals·ec5d02e01ec699817d1c71b60a3fa4d088| �Tgclocals·eda57d60e805297221010beefc01cf3d88 + + + + +�(go.string."/commit?"@2/commit? (go.string."/commit?"�Tgclocals·81f50624430372e4ea6dfcf6f12b35fd``6""" �Tgclocals·9de5ccc45996de67be5d048aaa3bec9388����������������(go.string."/attach?"@2/attach? (go.string."/attach?"�Tgclocals·504ff4b35adae90119a7640e996e47cf@@4��n�Tgclocals·acacf92de43844ed4d32a3b47fdcea71((��f��f��f�go.string."all"0(all go.string."all"�$go.string."/logs?"0./logs? $go.string."/logs?"�Tgclocals·75d51badeb043219a3daddeb664f46bc@@4 ���Tgclocals·2c6bb9a575800b4fd811118aedd59a39((�n%�n%�n%�go.string."h"0$h go.string."h"�go.string."w"0$w go.string."w"�(go.string."/resize?"@2/resize? (go.string."/resize?"�Tgclocals·564b0cd8045e1e3a560aecfc019285da�� *((�(� � "��"� ��Tgclocals·78ce512784b85b97418b7726f81bf730XX JJJJJJJJJ�Bgo.string."/containers/%s/export"PL/containers/%s/export Bgo.string."/containers/%s/export"�Tgclocals·fcc516824ce26c0001e09dc1d3d75478PP&�<�Tgclocals·48afd233022498cd45f3b0138014243e00�����>go.string."No such container: "PHNo such container:  >go.string."No such container: "�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Ngo.string."Container already running: "`XContainer already running:  Ngo.string."Container already running: "�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Fgo.string."Container not running: "PPContainer not running:  Fgo.string."Container not running: "�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·14c45952157723c8762210d9c661bf29 + +�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�go.string." \t"0&  go.string." \t"�go.string."0"0$0 go.string."0"�"go.string."false"0,false "go.string."false"� go.string."none"0*none go.string."none"�Tgclocals·23c4785fa8abd7e258acfe91c9f325f3  �Tgclocals·9ff42bf311af152488d11f0f78c8d5ce  + +�go.string."="0$= go.string."="�Tgclocals·2d894b3b66dff3ff7aaa2a78013804f9��$  ��Tgclocals·f774b632f7ff7d029527413a83030842HHJJJJJJJ�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·6fac742cdcfec8bff38f6662e683bbda00��Tgclocals·2cda55eacf8f3a391cf15caecdfeef0600JJJJ�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·6fac742cdcfec8bff38f6662e683bbda00��Tgclocals·2cda55eacf8f3a391cf15caecdfeef0600JJJJ�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·528c559c9193f2a671691be2686ab724��Tgclocals·299a4d24490b926d38628658bb77eeb1PP$��Tgclocals·bb06efbb6a26e0f286c10766fad350d700�����Tgclocals·80f0398afc092a879ad303c2fec80b66@@ *(�Tgclocals·f09ff24693e6d72e9e2f82319a6e45a0@@ + + + + + +�Tgclocals·61e2515c69061b8fed0e66ece719f936 �Tgclocals·ff7af1025fb7deae6ebf3487eab30c33 ���Tgclocals·f9166171185d1f1926264897a0c959c1(( �Tgclocals·46b690808f7e1a8626f300054e53774f(( +����Tgclocals·8f12e5afe7e149987419843938d69919``, +�Z�Tgclocals·784852ecd61fa458e8af6c57e3ee02b888 +.....�go.string."%v"0&%v go.string."%v"�Tgclocals·7c09a673592d13ccf4305e509b0c4fdf�� F�� < < � �Tgclocals·f2bff8318847e30874c64d3cd9d3a459pp +�������������Tgclocals·f3e8856499aee240134cb47f88c6cd55  +�Tgclocals·2148c3737b2bb476685a1100a2e8343e �Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·551282070bdf4bca9f3b8ada2a8f2d2a(( �Tgclocals·4ab27d0e7d4f80bb5765ef5f61de5fe5(( +� +�Tgclocals·158185e77a15ce9170c1aa92e62cd73e00 +�Tgclocals·00180cfd7eeeff04c22905d29bdac05200 +� + +�Tgclocals·29f0050a5ee7c2b9348a75428171d7de �Tgclocals·ac5bea9c8a91f5fb1d31bdacc5067b57 �Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·d3486bc7ce1948dc22d7ad1c0be2887a +�Tgclocals·0115f8d53b75c1696444f08ad03251d9�Tgclocals·fa051c55663fc115869f36c85a0645b9  +��Tgclocals·0115f8d53b75c1696444f08ad03251d9�Tgclocals·a9282ac20787dc3025c0916068a42263 .�Tgclocals·d963a621632aab39c7173f3f69ae3f91�� J ������i��������i����� � �Tgclocals·8e6ff68ca952ded665cfa894236f9944XX  + + + + + + + + +�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·0115f8d53b75c1696444f08ad03251d9�Tgclocals·cf86db206769ec68369d07e260728f65 �Tgclocals·0115f8d53b75c1696444f08ad03251d9�Tgclocals·cf86db206769ec68369d07e260728f65 �Tgclocals·770683613b64aeb90b5472e68a988b48  + +�Tgclocals·9d97800b9eac7aaad25644c1094f6baa  + +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·e8c55b930b09fa5028b5e4b78b8932dc +�,Bgo.itab.*crypto/tls.Conn.net.Conn�&go.string."/events"00/events &go.string."/events"�*go.string."?since=%d"@4 ?since=%d *go.string."?since=%d"�Tgclocals·c1f40b05e3ffba0283c820006999a7cf��<(��(�(<(�� +� +,��� + �(�"��(�(� ��Tgclocals·8c02cd934f4d00aa05beba150d4d3e04�� �������������������>go.string."/containers/%s/exec"PH/containers/%s/exec >go.string."/containers/%s/exec"�Tgclocals·b0ce568b8ee350283c34690ddf2c6892��6 |@�@� �Tgclocals·5a21f577b603cab6ea76228d95a69547PP&�&�&�&�&�&�&�&��,8go.itab.*"".NoSuchExec.error�4go.string."/exec/%s/start"@>/exec/%s/start 4go.string."/exec/%s/start"�Tgclocals·49a517ef1dc6f4c5ce9b594512df2da4��H�< �<��� �� ��i���Tgclocals·1bba016e9a05211bda029a0d75dbaa68HHJ��J��J��J��J��J��J���/images/%s/get 4go.string."/images/%s/get"�Tgclocals·fabba7188ed1b3b8bac23e1a07c66457@@$/�Tgclocals·740354061e4e9c9d9a50f05557f21f54((����0go.string."/images/get?"@: /images/get? 0go.string."/images/get?"�Tgclocals·d83f68254ae1224f9001a252749abef2((���Tgclocals·0629ba7d00f7a57ad6e2352df47e7bb3(( + + + �Tgclocals·9b3781349ecf8ea1253d7ba626d001b4``" ( �"�Tgclocals·10971996d6a01a6d477c3318892d070f88��n��n��n��n��n�6go.string."application/tar"@@application/tar 6go.string."application/tar"�*go.string."/build?%s"@4 /build?%s *go.string."/build?%s"�Tgclocals·f991e5818d95c260e9075daec3edcda1�� D �� + +  + +" +�Tgclocals·689d5e2b826a4f8fae61c828d739d7d9�� >�T�n""��T�n""��T�n""��T�n""��T�n""��T�n""��T�n""��T�n""��T�n""��&go.string."/tag?%s"00/tag?%s &go.string."/tag?%s"�Tgclocals·c6e5a101f01f70a879acdb3760944b0d((���Tgclocals·23803564b4b262dab15001f621fd3b37((�H�H�H�Tgclocals·2b892b6166a29da84b4f26d3316f1499  +�Tgclocals·d7e8a62d22b1cde6d92b17a55c33fe8f �,>go.itab.*bytes.Buffer.io.Writer�6go.string."X-Registry-Auth"@@X-Registry-Auth 6go.string."X-Registry-Auth"�:go.string."X-Registry-Config"PDX-Registry-Config :go.string."X-Registry-Config"�Tgclocals·7a03355e34b75c37acf5eff7bf674ad4��H"�"���"��*�""" ��" �*�"�Tgclocals·afcded8c13354e18af605d7f21ec25fe�� �@go.string."/images/search?term="PJ/images/search?term= @go.string."/images/search?term="�Tgclocals·52cf122e9547c47353d18ce23d85402a00�Tgclocals·766148fb4da5bf1af59ee4d8b91fb45400 + + + +�Tgclocals·41bb44495be0a59dc118277b1d9139f988 +�Tgclocals·9cf15d8275d9c299f023024ca604cf9088�"go.string."/info"0,/info "go.string."/info"�Tgclocals·41bb44495be0a59dc118277b1d9139f988 +�Tgclocals·9cf15d8275d9c299f023024ca604cf9088�Tgclocals·8d600a433c6aaa81a4fe446d95c5546b �Tgclocals·ca1ebfc68aaed1d083688775167e5178  �*go.string."/networks"@4 /networks *go.string."/networks"�Tgclocals·ae0db13c3cc1bbe7f0a2ea62d05cf906((�Tgclocals·6d3fa487f5e45db9cb9199d2a5e0e216(( �,>go.itab.*"".NoSuchNetwork.error�,go.string."/networks/"@6 +/networks/ ,go.string."/networks/"�Tgclocals·0c8fa0fcc4836d09a64d3d20b95663fe00�Tgclocals·be34fa03b4e4d696adaf8f647f7704fd00 + + + +�Tgclocals·35aa8cef5f531c9de8f76600ceb85b27��"|""  +�Tgclocals·81381a8f40f0e35a38db28a8bb50de11HH��������>go.string."No such network: %s"PHNo such network: %s >go.string."No such network: %s"�Tgclocals·403a8d79fd24b295e8557f6970497aa3((���Tgclocals·6d340c3bdac448a6ef1256f331f68dd3((�2go.string.".dockerignore"@< .dockerignore 2go.string.".dockerignore"�jgo.string."cannot match .dockerfile: '%s', error: %s"�t)cannot match .dockerfile: '%s', error: %s jgo.string."cannot match .dockerfile: '%s', error: %s"�Tgclocals·4df3d887804869ca0d16462c47a4175f�� T � (   �  � �(       �Tgclocals·d01647b6fcc19f6b40c264ab6c580992xx """""""""""""�Tgclocals·7546955fbaa0a8a5c520077bd4d4710500 ""�"�Tgclocals·e3c75ef39e8363f5b00a257bd2be7adb00""""�Zgo.string."error reading .dockerignore: '%s'"pd!error reading .dockerignore: '%s' Zgo.string."error reading .dockerignore: '%s'"�go.string."\n"0$ + go.string."\n"�Tgclocals·738657360054077c9c40a6546341a136pp0�"�" �� ��Tgclocals·2f519926ed4d9241bccfb3ede7c3f0ba@@�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·a08e9001cb8f9d822225de3b8e406515�,Bgo.itab.*"".tlsClientCon.net.Conn�Tgclocals·a6f85fd4ba75b8cdab35ab28e50023d9��0 + +�+� ( ��( �( ( (  �Tgclocals·0389232f9bf0423206204d8b27e58130��������������������Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·d85453ba2fc2b16513844b65495ea6c3"�,>go.itab.*bufio.Reader.io.Reader�Tgclocals·51d2fd2674ba9ccfd7abd80151d2e03288�* �Tgclocals·7c13896baab3273e10662a9a37b348ce�Tgclocals·2b592d649ecec7c5b5fac74b8e09bee800*"��Tgclocals·0372b889336bbdf612862c172920463d�Tgclocals·50e42ec547586bf00be346cef54257da88* +� +�Tgclocals·7c13896baab3273e10662a9a37b348ce�,@go.itab.*io.PipeWriter.io.Writer�Tgo.string."/containers/%s/stats?stream=%v"`^/containers/%s/stats?stream=%v Tgo.string."/containers/%s/stats?stream=%v"�Tgclocals·df3c8560fdbead80e4ddce1ccdbd1147�� +D� + �� +�� +��������"���Tgclocals·fb05dbbfacbbe47b8b1eb4226ce34430 +�Tgclocals·f34a2133376bc2e71ee31cc35164f3d3@@*j�eZ�h�eZ��Tgclocals·73423680ca5f2d7df4fe760a82d507fb�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·48a37d9114fa45f0336e02e754d41f88�� (���� ��������� ���Tgclocals·8e6ff68ca952ded665cfa894236f9944XX  + + + + + + + + +�6go.string."can't stat '%s'"@@can't stat '%s' 6go.string."can't stat '%s'"�Vgo.string."no permission to read from '%s'"``no permission to read from '%s' Vgo.string."no permission to read from '%s'"�Tgclocals·8638ac1ded2e05617036c77f7600dfac``&���Tgclocals·0e5d6e03d8b052993869281db2167ff788� � � � � �Tgclocals·0730e324c95d53ccaec07bf254f1f51600 + �Tgclocals·0372b889336bbdf612862c172920463d�Tgclocals·f1ce4f14231620ac9cd58e5cd8e6fa2d((,�Tgclocals·73423680ca5f2d7df4fe760a82d507fb�go.string."EOF"0(EOF go.string."EOF"�pgo.string."Failed to read authentication from dockercfg"�z,Failed to read authentication from dockercfg pgo.string."Failed to read authentication from dockercfg"�8go.string."invalid endpoint"PBinvalid endpoint 8go.string."invalid endpoint"�Zgo.string."cannot connect to Docker endpoint"pd!cannot connect to Docker endpoint Zgo.string."cannot connect to Docker endpoint"� go.string."1.12"0*1.12 go.string."1.12"�Hgo.string."container already exists"`Rcontainer already exists Hgo.string."container already exists"�bgo.string."no listeners present to receive event"pl%no listeners present to receive event bgo.string."no listeners present to receive event"�jgo.string."listener already exists for docker events"�t)listener already exists for docker events jgo.string."listener already exists for docker events"�2go.string."no such image"@< no such image 2go.string."no such image"��go.string."missing remote repository e.g. 'github.com/user/repo'"��5missing remote repository e.g. 'github.com/user/repo' �go.string."missing remote repository e.g. 'github.com/user/repo'"�Bgo.string."missing output stream"PLmissing output stream Bgo.string."missing output stream"��go.string."image build may not be provided BOTH context dir and input stream"��Aimage build may not be provided BOTH context dir and input stream �go.string."image build may not be provided BOTH context dir and input stream"�hgo.string."must specify at least one name to export"�r(must specify at least one name to export hgo.string."must specify at least one name to export"�Dgo.string."network already exists"PNnetwork already exists Dgo.string."network already exists"�Tgclocals·7b2d1dc8e692ba633cb2c876407e20f2 +�Tgclocals·3280bececceccd33cb74587feedb1f9f�*""".AuthParseError type.error�**"".ErrInvalidEndpoint type.error�*."".ErrConnectionRefused type.error�* "".apiVersion1120$type."".APIVersion�*8"".ErrContainerAlreadyExists type.error�*""".ErrNoListeners type.error�*6"".ErrListenerAlreadyExists type.error�*"".EOFEvent$type.*"".APIEvents""".statictmp_1673�*""".ErrNoSuchImage type.error�*""".ErrMissingRepo type.error�*2"".ErrMissingOutputStream type.error�*,"".ErrMultipleContexts type.error�*,"".ErrMustSpecifyNames type.error�*4"".ErrNetworkAlreadyExists type.error�""".statictmp_0068`type.[3]string`   &go.string.".docker"@ .go.string."config.json"�""".statictmp_0073@type.[2]string@ +  ,go.string.".dockercfg"�""".statictmp_0222@type.[2]string@  (go.string."cert.pem"�""".statictmp_0225@type.[2]string@  &go.string."key.pem"�""".statictmp_0228@type.[2]string@  $go.string."ca.pem"�""".statictmp_0552@type.[2]string@  &go.string.".docker"�""".statictmp_0876�*type."".hijackOptions�""".statictmp_1200�*type."".hijackOptions�""".statictmp_1318�*type."".streamOptions�""".statictmp_1373�*type."".streamOptions�""".statictmp_1509 type.[1]string  go.string."."�""".statictmp_1511@type.[2]string  2go.string.".dockerignore"�""".statictmp_1545@type.[2]string@  go.string."."�""".statictmp_1558@type.[2]string@   2go.string.".dockerignore"�*""".statictmp_1673p"type."".APIEvents  go.string."EOF"�,"".initdone·type.uint8�P"".NewAuthConfigurationsFromDockerCfg·fJ"".NewAuthConfigurationsFromDockerCfg�os.Getenv·fos.Getenv�:runtime.writebarrierstring·f4runtime.writebarrierstring�path.Join·fpath.Join�&runtime.typ2Itab·f runtime.typ2Itab�os.Open·fos.Open�6"".NewAuthConfigurations·f0"".NewAuthConfigurations�,runtime.throwreturn·f&runtime.throwreturn�."".parseDockerConfig·f("".parseDockerConfig�""".authConfigs·f"".authConfigs�(runtime.newobject·f"runtime.newobject�6bytes.(*Buffer).ReadFrom·f0bytes.(*Buffer).ReadFrom�*runtime.panicslice·f$runtime.panicslice�4encoding/json.Unmarshal·f.encoding/json.Unmarshal�:runtime.mapaccess2_faststr·f4runtime.mapaccess2_faststr�$runtime.makemap·fruntime.makemap�4runtime.writebarrierptr·f.runtime.writebarrierptr�,runtime.mapiterinit·f&runtime.mapiterinit�,runtime.mapiternext·f&runtime.mapiternext�Vencoding/base64.(*Encoding).DecodeString·fPencoding/base64.(*Encoding).DecodeString�8runtime.slicebytetostring·f2runtime.slicebytetostring� strings.Split·fstrings.Split�*runtime.panicindex·f$runtime.panicindex�*runtime.mapassign1·f$runtime.mapassign1�2"".(*Client).AuthCheck·f,"".(*Client).AuthCheck�fmt.Errorf·ffmt.Errorf�$"".(*Client).do·f"".(*Client).do�$runtime.convT2E·fruntime.convT2E�8runtime.writebarrieriface·f2runtime.writebarrieriface�,"".(*Change).String·f&"".(*Change).String�fmt.Sprintf·ffmt.Sprintf�&"".NewAPIVersion·f "".NewAPIVersion�&strings.Contains·f strings.Contains�(runtime.makeslice·f"runtime.makeslice�strconv.Atoi·fstrconv.Atoi�."".APIVersion.String·f("".APIVersion.String�strconv.Itoa·fstrconv.Itoa�0runtime.concatstring2·f*runtime.concatstring2�2"".APIVersion.LessThan·f,"".APIVersion.LessThan�0"".APIVersion.compare·f*"".APIVersion.compare�D"".APIVersion.LessThanOrEqualTo·f>"".APIVersion.LessThanOrEqualTo�8"".APIVersion.GreaterThan·f2"".APIVersion.GreaterThan�J"".APIVersion.GreaterThanOrEqualTo·fD"".APIVersion.GreaterThanOrEqualTo�"".NewClient·f"".NewClient�0"".NewVersionedClient·f*"".NewVersionedClient�$"".NewTLSClient·f"".NewTLSClient�6"".NewVersionedTLSClient·f0"".NewVersionedTLSClient�6"".NewTLSClientFromBytes·f0"".NewTLSClientFromBytes�H"".NewVersionedTLSClientFromBytes·fB"".NewVersionedTLSClientFromBytes�&"".parseEndpoint·f "".parseEndpoint�8runtime.writebarrierslice·f2runtime.writebarrierslice�8"".NewVersionnedTLSClient·f2"".NewVersionnedTLSClient�*io/ioutil.ReadFile·f$io/ioutil.ReadFile�,"".NewClientFromEnv·f&"".NewClientFromEnv�>"".NewVersionedClientFromEnv·f8"".NewVersionedClientFromEnv�$"".getDockerEnv·f"".getDockerEnv�"strings.SplitN·fstrings.SplitN�*path/filepath.Join·f$path/filepath.Join�2crypto/tls.X509KeyPair·f,crypto/tls.X509KeyPair�4runtime.writebarrierfat·f.runtime.writebarrierfat�Zcrypto/x509.(*CertPool).AppendCertsFromPEM·fTcrypto/x509.(*CertPool).AppendCertsFromPEM�>"".(*Client).checkAPIVersion·f8"".(*Client).checkAPIVersion�R"".(*Client).getServerAPIVersionString·fL"".(*Client).getServerAPIVersionString�("".(*Client).Ping·f""".(*Client).Ping�:runtime.mapaccess1_faststr·f4runtime.mapaccess1_faststr�*runtime.assertE2T2·f$runtime.assertE2T2�0encoding/json.Marshal·f*encoding/json.Marshal�,runtime.deferreturn·f&runtime.deferreturn�&runtime.eqstring·f runtime.eqstring�,"".(*Client).getURL·f&"".(*Client).getURL�,net/http.NewRequest·f&net/http.NewRequest�,net/http.Header.Set·f&net/http.Header.Set�net.Dial·fnet.Dial�(runtime.deferproc·f"runtime.deferproc�$runtime.convI2I·fruntime.convI2I�$bufio.NewReader·fbufio.NewReader�8net/http.(*Request).Write·f2net/http.(*Request).Write�0net/http.ReadResponse·f*net/http.ReadResponse�0net/http.(*Client).Do·f*net/http.(*Client).Do�(io/ioutil.ReadAll·f"io/ioutil.ReadAll�,"".(*Client).stream·f&"".(*Client).stream�time.Now·ftime.Now� time.Time.Add·ftime.Time.Add�,net/http.Header.Get·f&net/http.Header.Get�io.Copy·fio.Copy�Dencoding/json.(*Decoder).Decode·f>encoding/json.(*Decoder).Decode�$runtime.ifaceeq·fruntime.ifaceeq�fmt.Fprint·ffmt.Fprint�fmt.Fprintf·ffmt.Fprintf�fmt.Fprintln·ffmt.Fprintln��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.StdCopy·f�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.StdCopy�,"".(*Client).hijack·f&"".(*Client).hijack�"".tlsDial·f"".tlsDial�Dnet/http/httputil.NewClientConn·f>net/http/httputil.NewClientConn�Pnet/http/httputil.(*ClientConn).Close·fJnet/http/httputil.(*ClientConn).Close�Jnet/http/httputil.(*ClientConn).Do·fDnet/http/httputil.(*ClientConn).Do�(runtime.chansend1·f"runtime.chansend1�(runtime.chanrecv1·f"runtime.chanrecv1�Rnet/http/httputil.(*ClientConn).Hijack·fLnet/http/httputil.(*ClientConn).Hijack�&runtime.makechan·f runtime.makechan�"".func·001·f"".func·001�$runtime.newproc·fruntime.newproc�"".func·002·f"".func·002�0net/url.(*URL).String·f*net/url.(*URL).String�(strings.TrimRight·f"strings.TrimRight�""".queryString·f"".queryString�$reflect.ValueOf·freflect.ValueOf�*reflect.Value.Kind·f$reflect.Value.Kind�*reflect.Value.Elem·f$reflect.Value.Elem�2reflect.Value.NumField·f,reflect.Value.NumField�*reflect.Value.Type·f$reflect.Value.Type�0reflect.StructTag.Get·f*reflect.StructTag.Get�$strings.ToLower·fstrings.ToLower�,reflect.Value.Field·f&reflect.Value.Field�2"".addQueryStringValue·f,"".addQueryStringValue�0net/url.Values.Encode·f*net/url.Values.Encode�*reflect.Value.Bool·f$reflect.Value.Bool�(runtime.growslice·f"runtime.growslice�(reflect.Value.Int·f"reflect.Value.Int�(strconv.FormatInt·f"strconv.FormatInt�,reflect.Value.Float·f&reflect.Value.Float�,strconv.FormatFloat·f&strconv.FormatFloat�.reflect.Value.String·f(reflect.Value.String�,reflect.Value.IsNil·f&reflect.Value.IsNil�4reflect.Value.Interface·f.reflect.Value.Interface�0reflect.Value.MapKeys·f*reflect.Value.MapKeys�(reflect.Value.Len·f"reflect.Value.Len�,reflect.Value.Index·f&reflect.Value.Index�"".newError·f"".newError�("".(*Error).Error·f""".(*Error).Error� net/url.Parse·fnet/url.Parse�(runtime.cmpstring·f"runtime.cmpstring�(net.SplitHostPort·f"net.SplitHostPort�*runtime.assertI2T2·f$runtime.assertI2T2�&strconv.ParseInt·f strconv.ParseInt�4"".getDefaultDockerHost·f."".getDefaultDockerHost��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir.Get·f�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir.Get�(path/filepath.Abs·f"path/filepath.Abs��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts.ValidateHost·f�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts.ValidateHost�<"".(*Client).ListContainers·f6"".(*Client).ListContainers�"".Port.Port·f"".Port.Port� "".Port.Proto·f"".Port.Proto�*"".(*State).String·f$"".(*State).String� time.Time.Sub·ftime.Time.Sub�N"".(*NetworkSettings).PortMappingAPI·fH"".(*NetworkSettings).PortMappingAPI�"".parsePort·f"".parsePort�(strconv.ParseUint·f"strconv.ParseUint�>"".(*Client).RenameContainer·f8"".(*Client).RenameContainer�0runtime.concatstring3·f*runtime.concatstring3�@"".(*Client).InspectContainer·f:"".(*Client).InspectContainer�@"".(*Client).ContainerChanges·f:"".(*Client).ContainerChanges�>"".(*Client).CreateContainer·f8"".(*Client).CreateContainer�&"".AlwaysRestart·f "".AlwaysRestart�,"".RestartOnFailure·f&"".RestartOnFailure�$"".NeverRestart·f"".NeverRestart�<"".(*Client).StartContainer·f6"".(*Client).StartContainer�:"".(*Client).StopContainer·f4"".(*Client).StopContainer�@"".(*Client).RestartContainer·f:"".(*Client).RestartContainer�<"".(*Client).PauseContainer·f6"".(*Client).PauseContainer�@"".(*Client).UnpauseContainer·f:"".(*Client).UnpauseContainer�8"".(*Client).TopContainer·f2"".(*Client).TopContainer�*"".(*Client).Stats·f$"".(*Client).Stats�io.Pipe·fio.Pipe�"".func·003·f"".func·003�"".func·004·f"".func·004�(runtime.closechan·f"runtime.closechan�"".func·005·f"".func·005�:"".(*Client).KillContainer·f4"".(*Client).KillContainer�0runtime.concatstring4·f*runtime.concatstring4�>"".(*Client).RemoveContainer·f8"".(*Client).RemoveContainer�B"".(*Client).CopyFromContainer·f<"".(*Client).CopyFromContainer�:"".(*Client).WaitContainer·f4"".(*Client).WaitContainer�>"".(*Client).CommitContainer·f8"".(*Client).CommitContainer�B"".(*Client).AttachToContainer·f<"".(*Client).AttachToContainer�("".(*Client).Logs·f""".(*Client).Logs�D"".(*Client).ResizeContainerTTY·f>"".(*Client).ResizeContainerTTY�>"".(*Client).ExportContainer·f8"".(*Client).ExportContainer�<"".(*NoSuchContainer).Error·f6"".(*NoSuchContainer).Error�L"".(*ContainerAlreadyRunning).Error·fF"".(*ContainerAlreadyRunning).Error�D"".(*ContainerNotRunning).Error·f>"".(*ContainerNotRunning).Error� "".(*Env).Get·f"".(*Env).Get� "".(*Env).Map·f"".(*Env).Map�&"".(*Env).Exists·f "".(*Env).Exists�("".(*Env).GetBool·f""".(*Env).GetBool�strings.Trim·fstrings.Trim�("".(*Env).SetBool·f""".(*Env).SetBool�&"".(*Env).GetInt·f "".(*Env).GetInt�*"".(*Env).GetInt64·f$"".(*Env).GetInt64�&"".(*Env).SetInt·f "".(*Env).SetInt�*"".(*Env).SetInt64·f$"".(*Env).SetInt64�("".(*Env).GetJSON·f""".(*Env).GetJSON�8runtime.stringtoslicebyte·f2runtime.stringtoslicebyte�("".(*Env).SetJSON·f""".(*Env).SetJSON�("".(*Env).GetList·f""".(*Env).GetList�("".(*Env).SetList·f""".(*Env).SetList� "".(*Env).Set·f"".(*Env).Set�&"".(*Env).Decode·f "".(*Env).Decode�("".(*Env).SetAuto·f""".(*Env).SetAuto�@"".(*Client).AddEventListener·f:"".(*Client).AddEventListener�N"".(*eventMonitoringState).isEnabled·fH"".(*eventMonitoringState).isEnabled�f"".(*eventMonitoringState).enableEventMonitoring·f`"".(*eventMonitoringState).enableEventMonitoring�R"".(*eventMonitoringState).addListener·fL"".(*eventMonitoringState).addListener�F"".(*Client).RemoveEventListener·f@"".(*Client).RemoveEventListener�X"".(*eventMonitoringState).removeListener·fR"".(*eventMonitoringState).removeListener�h"".(*eventMonitoringState).disableEventMonitoring·fb"".(*eventMonitoringState).disableEventMonitoring�.sync.(*RWMutex).Lock·f(sync.(*RWMutex).Lock�2sync.(*RWMutex).Unlock·f,sync.(*RWMutex).Unlock�("".listenerExists·f""".listenerExists�0sync.(*WaitGroup).Add·f*sync.(*WaitGroup).Add�X"".(*eventMonitoringState).closeListeners·fR"".(*eventMonitoringState).closeListeners�V"".(*eventMonitoringState).monitorEvents·fP"".(*eventMonitoringState).monitorEvents�2sync.(*WaitGroup).Wait·f,sync.(*WaitGroup).Wait�R"".(*eventMonitoringState).noListeners·fL"".(*eventMonitoringState).noListeners�time.Sleep·ftime.Sleep�\"".(*eventMonitoringState).connectWithRetry·fV"".(*eventMonitoringState).connectWithRetry�time.After·ftime.After�(runtime.newselect·f"runtime.newselect�,runtime.selectrecv2·f&runtime.selectrecv2�X"".(*eventMonitoringState).updateLastSeen·fR"".(*eventMonitoringState).updateLastSeen�N"".(*eventMonitoringState).sendEvent·fH"".(*eventMonitoringState).sendEvent�*runtime.selectrecv·f$runtime.selectrecv�"".func·006·f"".func·006�&runtime.selectgo·f runtime.selectgo�0sync/atomic.LoadInt64·f*sync/atomic.LoadInt64�6"".(*Client).eventHijack·f0"".(*Client).eventHijack�math.Pow·fmath.Pow�0sync.(*RWMutex).RLock·f*sync.(*RWMutex).RLock�4sync.(*RWMutex).RUnlock·f.sync.(*RWMutex).RUnlock�2sync.(*WaitGroup).Done·f,sync.(*WaitGroup).Done�2sync/atomic.StoreInt64·f,sync/atomic.StoreInt64�$crypto/tls.Dial·fcrypto/tls.Dial�"".func·007·f"".func·007�4"".(*Client).CreateExec·f."".(*Client).CreateExec�2"".(*Client).StartExec·f,"".(*Client).StartExec�:"".(*Client).ResizeExecTTY·f4"".(*Client).ResizeExecTTY�6"".(*Client).InspectExec·f0"".(*Client).InspectExec�2"".(*NoSuchExec).Error·f,"".(*NoSuchExec).Error�4"".(*Client).ListImages·f."".(*Client).ListImages�8"".(*Client).ImageHistory·f2"".(*Client).ImageHistory�6"".(*Client).RemoveImage·f0"".(*Client).RemoveImage�F"".(*Client).RemoveImageExtended·f@"".(*Client).RemoveImageExtended�8"".(*Client).InspectImage·f2"".(*Client).InspectImage�6runtime.writebarrierfat3·f0runtime.writebarrierfat3�2"".(*Client).PushImage·f,"".(*Client).PushImage�*"".headersWithAuth·f$"".headersWithAuth�2"".(*Client).PullImage·f,"".(*Client).PullImage�6"".(*Client).createImage·f0"".(*Client).createImage�2"".(*Client).LoadImage·f,"".(*Client).LoadImage�6"".(*Client).ExportImage·f0"".(*Client).ExportImage�8"".(*Client).ExportImages·f2"".(*Client).ExportImages�6"".(*Client).ImportImage·f0"".(*Client).ImportImage�"".isURL·f"".isURL�4"".(*Client).BuildImage·f."".(*Client).BuildImage�*"".createTarStream·f$"".createTarStream�0"".(*Client).TagImage·f*"".(*Client).TagImage�*runtime.efacethash·f$runtime.efacethash�,runtime.assertE2TOK·f&runtime.assertE2TOK�Dencoding/json.(*Encoder).Encode·f>encoding/json.(*Encoder).Encode�Zencoding/base64.(*Encoding).EncodeToString·fTencoding/base64.(*Encoding).EncodeToString�8"".(*Client).SearchImages·f2"".(*Client).SearchImages�."".(*Client).Version·f("".(*Client).Version�("".(*Client).Info·f""".(*Client).Info�0"".ParseRepositoryTag·f*"".ParseRepositoryTag�(strings.LastIndex·f"strings.LastIndex�8"".(*Client).ListNetworks·f2"".(*Client).ListNetworks�6"".(*Client).NetworkInfo·f0"".(*Client).NetworkInfo�:"".(*Client).CreateNetwork·f4"".(*Client).CreateNetwork�8"".(*NoSuchNetwork).Error·f2"".(*NoSuchNetwork).Error�."".parseDockerignore·f("".parseDockerignore��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils.Matches·f�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils.Matches�$runtime.convI2E·fruntime.convI2E�<"".validateContextDirectory·f6"".validateContextDirectory��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive.TarWithOptions·f�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive.TarWithOptions�"".func·008·f"".func·008�*path/filepath.Walk·f$path/filepath.Walk� os.IsNotExist·fos.IsNotExist�@"".(*tlsClientCon).CloseWrite·f:"".(*tlsClientCon).CloseWrite�*runtime.assertI2I2·f$runtime.assertI2I2�."".tlsDialWithDialer·f("".tlsDialWithDialer�"".func·009·f"".func·009�"time.AfterFunc·ftime.AfterFunc�*net.(*Dialer).Dial·f$net.(*Dialer).Dial�>crypto/tls.(*Conn).Handshake·f8crypto/tls.(*Conn).Handshake�"".func·010·f"".func·010�(runtime.assertI2I·f"runtime.assertI2I�.runtime.selectnbrecv·f(runtime.selectnbrecv�2io.(*PipeReader).Close·f,io.(*PipeReader).Close�2io.(*PipeWriter).Close·f,io.(*PipeWriter).Close�(path/filepath.Rel·f"path/filepath.Rel�$os.IsPermission·fos.IsPermission�&os.(*File).Close·f os.(*File).Close�"".init·f"".init�(runtime.throwinit·f"runtime.throwinit��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils.init·f�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils.init��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive.init·f�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive.init�sync.init·fsync.init�math.init·fmath.init��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.init·f�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.init��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir.init·f�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir.init��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts.init·f�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts.init�time.init·ftime.init�strconv.init·fstrconv.init�runtime.init·fruntime.init�reflect.init·freflect.init�*path/filepath.init·f$path/filepath.init�net/url.init·fnet/url.init�2net/http/httputil.init·f,net/http/httputil.init� net/http.init·fnet/http.init�net.init·fnet.init�"io/ioutil.init·fio/ioutil.init�&crypto/x509.init·f crypto/x509.init�$crypto/tls.init·fcrypto/tls.init�bufio.init·fbufio.init�strings.init·fstrings.init�path.init·fpath.init�os.init·fos.init�io.init·fio.init�fmt.init·ffmt.init�*encoding/json.init·f$encoding/json.init�.encoding/base64.init·f(encoding/base64.init�bytes.init·fbytes.init�errors.New·ferrors.New�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�Ftype..hashfunc."".AuthConfiguration>type..hash."".AuthConfiguration�Btype..eqfunc."".AuthConfiguration:type..eq."".AuthConfiguration� &type..alg.[8]string0bruntime.gcbits.0x48484848484848480000000000000000P*go.string."[8]string"p.go.weak.type.*[8]string�"runtime.zerovalue�type.string�type.[]string�>go.typelink.[8]string/[8]stringtype.[8]string�Lgo.string."[]docker.AuthConfiguration"`V[]docker.AuthConfiguration Lgo.string."[]docker.AuthConfiguration"�6type.[]"".AuthConfiguration���+�� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000PLgo.string."[]docker.AuthConfiguration"pHgo.weak.type.*[]"".AuthConfiguration�"runtime.zerovalue�2type."".AuthConfiguration�zgo.typelink.[]docker.AuthConfiguration/[]"".AuthConfiguration6type.[]"".AuthConfiguration�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·65526a5f07004f02424fe51b799cdd23  +�Tgclocals·fa7203fd5ed88aea99b7be572f707eb0 �Ltype..hashfunc.[8]"".AuthConfigurationDtype..hash.[8]"".AuthConfiguration�Htype..eqfunc.[8]"".AuthConfiguration@type..eq.[8]"".AuthConfiguration�Btype..alg.[8]"".AuthConfiguration Ltype..hashfunc.[8]"".AuthConfigurationHtype..eqfunc.[8]"".AuthConfiguration�,@type..gc.[8]"".AuthConfigurationB�Htype..gcprog.[8]"".AuthConfigurationff�Ngo.string."[8]docker.AuthConfiguration"`X[8]docker.AuthConfiguration Ngo.string."[8]docker.AuthConfiguration"�8type.[8]"".AuthConfiguration����X]Q Btype..alg.[8]"".AuthConfiguration0@type..gc.[8]"".AuthConfiguration@Htype..gcprog.[8]"".AuthConfigurationPNgo.string."[8]docker.AuthConfiguration"pJgo.weak.type.*[8]"".AuthConfiguration�"runtime.zerovalue�2type."".AuthConfiguration�6type.[]"".AuthConfiguration�~go.typelink.[8]docker.AuthConfiguration/[8]"".AuthConfiguration8type.[8]"".AuthConfiguration�ngo.string."*map.bucket[string]docker.AuthConfiguration"�x+*map.bucket[string]docker.AuthConfiguration ngo.string."*map.bucket[string]docker.AuthConfiguration"�Xtype.*map.bucket[string]"".AuthConfiguration��i�g/6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pngo.string."*map.bucket[string]docker.AuthConfiguration"pjgo.weak.type.**map.bucket[string]"".AuthConfiguration�"runtime.zerovalue�Vtype.map.bucket[string]"".AuthConfiguration�,^type..gc.map.bucket[string]"".AuthConfigurationT�ftype..gcprog.map.bucket[string]"".AuthConfiguration22����ff�lgo.string."map.bucket[string]docker.AuthConfiguration"�v*map.bucket[string]docker.AuthConfiguration lgo.string."map.bucket[string]docker.AuthConfiguration"� go.string."keys"0*keys go.string."keys"�$go.string."values"0.values $go.string."values"�(go.string."overflow"@2overflow (go.string."overflow"�Vtype.map.bucket[string]"".AuthConfiguration���Ի��Y�� � runtime.algarray0^type..gc.map.bucket[string]"".AuthConfiguration@ftype..gcprog.map.bucket[string]"".AuthConfigurationPlgo.string."map.bucket[string]docker.AuthConfiguration"phgo.weak.type.*map.bucket[string]"".AuthConfiguration�"runtime.zerovalue��Vtype.map.bucket[string]"".AuthConfiguration� go.string."keys"�type.[8]string�$go.string."values"�8type.[8]"".AuthConfiguration�(go.string."overflow"�Xtype.*map.bucket[string]"".AuthConfiguration�bruntime.gcbits.0x44844800000000000000000000000000 D�H�fgo.string."map.hdr[string]docker.AuthConfiguration"pp'map.hdr[string]docker.AuthConfiguration fgo.string."map.hdr[string]docker.AuthConfiguration"�&go.string."buckets"00buckets &go.string."buckets"�,go.string."oldbuckets"@6 +oldbuckets ,go.string."oldbuckets"�Ptype.map.hdr[string]"".AuthConfiguration��0��R  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000Pfgo.string."map.hdr[string]docker.AuthConfiguration"pbgo.weak.type.*map.hdr[string]"".AuthConfiguration�"runtime.zerovalue��Ptype.map.hdr[string]"".AuthConfiguration�&go.string."buckets"�Xtype.*map.bucket[string]"".AuthConfiguration�,go.string."oldbuckets"�Xtype.*map.bucket[string]"".AuthConfiguration�^go.string."map[string]docker.AuthConfiguration"ph#map[string]docker.AuthConfiguration ^go.string."map[string]docker.AuthConfiguration"�Htype.map[string]"".AuthConfiguration����:c5@� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P^go.string."map[string]docker.AuthConfiguration"pZgo.weak.type.*map[string]"".AuthConfiguration�"runtime.zerovalue�type.string�2type."".AuthConfiguration�Vtype.map.bucket[string]"".AuthConfiguration�Ptype.map.hdr[string]"".AuthConfiguration��go.typelink.map[string]docker.AuthConfiguration/map[string]"".AuthConfigurationHtype.map[string]"".AuthConfiguration�Jgo.string."docker.AuthConfigurations"`Tdocker.AuthConfigurations Jgo.string."docker.AuthConfigurations"�&go.string."Configs"00Configs &go.string."Configs"�8go.string."json:\"configs\""@>json:"configs" 8go.string."json:\"configs\""�go.typelink.[3]string/[3]stringtype.[3]string�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�0type..hashfunc.[2]string(type..hash.[2]string�,type..eqfunc.[2]string$type..eq.[2]string�&type..alg.[2]string 0type..hashfunc.[2]string,type..eqfunc.[2]string�bruntime.gcbits.0x48480000000000000000000000000000 HH�*go.string."[2]string"@4 [2]string *go.string."[2]string"�type.[2]string�� PX�� &type..alg.[2]string0bruntime.gcbits.0x48480000000000000000000000000000P*go.string."[2]string"p.go.weak.type.*[2]string�"runtime.zerovalue�type.string�type.[]string�>go.typelink.[2]string/[2]stringtype.[2]string�,go.string."*[3]string"@6 +*[3]string ,go.string."*[3]string"�type.*[3]string�� ++� 6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."*[3]string"p0go.weak.type.**[3]string�"runtime.zerovalue�type.[3]string�,go.string."*[2]string"@6 +*[2]string ,go.string."*[2]string"�type.*[2]string�� f<6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."*[2]string"p0go.weak.type.**[2]string�"runtime.zerovalue�type.[2]string�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�go.string."docker.dockerConfig"PHdocker.dockerConfig >go.string."docker.dockerConfig"� go.string."Auth"0*Auth go.string."Auth"�2go.string."json:\"auth\""@8 json:"auth" 2go.string."json:\"auth\""�4go.string."json:\"email\""@: json:"email" 4go.string."json:\"email\""�0go.string."dockerConfig"@: dockerConfig 0go.string."dockerConfig"�(type."".dockerConfig�� 0��� 2type..alg."".dockerConfig0bruntime.gcbits.0x48480000000000000000000000000000P>go.string."docker.dockerConfig"p*type.*"".dockerConfig�"runtime.zerovalue��(type."".dockerConfig� go.string."Auth"�type.string�2go.string."json:\"auth\""�"go.string."Email"�type.string�4go.string."json:\"email\""`�(type."".dockerConfig�0go.string."dockerConfig"�"go.importpath."".��(type."".dockerConfig�Bgo.string."[]docker.dockerConfig"PL[]docker.dockerConfig Bgo.string."[]docker.dockerConfig"�,type.[]"".dockerConfig��=hl � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000PBgo.string."[]docker.dockerConfig"p>go.weak.type.*[]"".dockerConfig�"runtime.zerovalue�(type."".dockerConfig�fgo.typelink.[]docker.dockerConfig/[]"".dockerConfig,type.[]"".dockerConfig�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·65526a5f07004f02424fe51b799cdd23  +�Tgclocals·fa7203fd5ed88aea99b7be572f707eb0 �Btype..hashfunc.[8]"".dockerConfig:type..hash.[8]"".dockerConfig�>type..eqfunc.[8]"".dockerConfig6type..eq.[8]"".dockerConfig�8type..alg.[8]"".dockerConfig Btype..hashfunc.[8]"".dockerConfig>type..eqfunc.[8]"".dockerConfig�bruntime.gcbits.0x48484848484848484848484848484848 HHHHHHHHHHHHHHHH�Dgo.string."[8]docker.dockerConfig"PN[8]docker.dockerConfig Dgo.string."[8]docker.dockerConfig"�.type.[8]"".dockerConfig���ei� 8type..alg.[8]"".dockerConfig0bruntime.gcbits.0x48484848484848484848484848484848PDgo.string."[8]docker.dockerConfig"p@go.weak.type.*[8]"".dockerConfig�"runtime.zerovalue�(type."".dockerConfig�,type.[]"".dockerConfig�jgo.typelink.[8]docker.dockerConfig/[8]"".dockerConfig.type.[8]"".dockerConfig�dgo.string."*map.bucket[string]docker.dockerConfig"pn&*map.bucket[string]docker.dockerConfig dgo.string."*map.bucket[string]docker.dockerConfig"�Ntype.*map.bucket[string]"".dockerConfig������6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pdgo.string."*map.bucket[string]docker.dockerConfig"p`go.weak.type.**map.bucket[string]"".dockerConfig�"runtime.zerovalue�Ltype.map.bucket[string]"".dockerConfig�,Ttype..gc.map.bucket[string]"".dockerConfig4�\type..gcprog.map.bucket[string]"".dockerConfig00����f�bgo.string."map.bucket[string]docker.dockerConfig"pl%map.bucket[string]docker.dockerConfig bgo.string."map.bucket[string]docker.dockerConfig"�Ltype.map.bucket[string]"".dockerConfig����)tY�� � runtime.algarray0Ttype..gc.map.bucket[string]"".dockerConfig@\type..gcprog.map.bucket[string]"".dockerConfigPbgo.string."map.bucket[string]docker.dockerConfig"p^go.weak.type.*map.bucket[string]"".dockerConfig�"runtime.zerovalue��Ltype.map.bucket[string]"".dockerConfig� go.string."keys"�type.[8]string�$go.string."values"�.type.[8]"".dockerConfig�(go.string."overflow"�Ntype.*map.bucket[string]"".dockerConfig�\go.string."map.hdr[string]docker.dockerConfig"pf"map.hdr[string]docker.dockerConfig \go.string."map.hdr[string]docker.dockerConfig"�Ftype.map.hdr[string]"".dockerConfig��0K�  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000P\go.string."map.hdr[string]docker.dockerConfig"pXgo.weak.type.*map.hdr[string]"".dockerConfig�"runtime.zerovalue��Ftype.map.hdr[string]"".dockerConfig�&go.string."buckets"�Ntype.*map.bucket[string]"".dockerConfig�,go.string."oldbuckets"�Ntype.*map.bucket[string]"".dockerConfig�Tgo.string."map[string]docker.dockerConfig"`^map[string]docker.dockerConfig Tgo.string."map[string]docker.dockerConfig"�>type.map[string]"".dockerConfig��H��5 � � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PTgo.string."map[string]docker.dockerConfig"pPgo.weak.type.*map[string]"".dockerConfig�"runtime.zerovalue�type.string�(type."".dockerConfig�Ltype.map.bucket[string]"".dockerConfig�Ftype.map.hdr[string]"".dockerConfig��go.typelink.map[string]docker.dockerConfig/map[string]"".dockerConfig>type.map[string]"".dockerConfig�Xgo.string."[]map[string]docker.dockerConfig"pb []map[string]docker.dockerConfig Xgo.string."[]map[string]docker.dockerConfig"�Btype.[]map[string]"".dockerConfig���� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000PXgo.string."[]map[string]docker.dockerConfig"pTgo.weak.type.*[]map[string]"".dockerConfig�"runtime.zerovalue�>type.map[string]"".dockerConfig��go.typelink.[]map[string]docker.dockerConfig/[]map[string]"".dockerConfigBtype.[]map[string]"".dockerConfig�bruntime.gcbits.0x88888888000000000000000000000000 �����Zgo.string."[8]map[string]docker.dockerConfig"pd![8]map[string]docker.dockerConfig Zgo.string."[8]map[string]docker.dockerConfig"�Dtype.[8]map[string]"".dockerConfig��@���� � runtime.algarray0bruntime.gcbits.0x88888888000000000000000000000000PZgo.string."[8]map[string]docker.dockerConfig"pVgo.weak.type.*[8]map[string]"".dockerConfig�"runtime.zerovalue�>type.map[string]"".dockerConfig�Btype.[]map[string]"".dockerConfig��go.typelink.[8]map[string]docker.dockerConfig/[8]map[string]"".dockerConfigDtype.[8]map[string]"".dockerConfig�zgo.string."*map.bucket[string]map[string]docker.dockerConfig"��1*map.bucket[string]map[string]docker.dockerConfig zgo.string."*map.bucket[string]map[string]docker.dockerConfig"�dtype.*map.bucket[string]map[string]"".dockerConfig��aخ�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pzgo.string."*map.bucket[string]map[string]docker.dockerConfig"pvgo.weak.type.**map.bucket[string]map[string]"".dockerConfig�"runtime.zerovalue�btype.map.bucket[string]map[string]"".dockerConfig�bruntime.gcbits.0x84848484848484848488888888000000 ��������������xgo.string."map.bucket[string]map[string]docker.dockerConfig"��0map.bucket[string]map[string]docker.dockerConfig xgo.string."map.bucket[string]map[string]docker.dockerConfig"�btype.map.bucket[string]map[string]"".dockerConfig���&����� � runtime.algarray0bruntime.gcbits.0x84848484848484848488888888000000Pxgo.string."map.bucket[string]map[string]docker.dockerConfig"ptgo.weak.type.*map.bucket[string]map[string]"".dockerConfig�"runtime.zerovalue��btype.map.bucket[string]map[string]"".dockerConfig� go.string."keys"�type.[8]string�$go.string."values"�Dtype.[8]map[string]"".dockerConfig�(go.string."overflow"�dtype.*map.bucket[string]map[string]"".dockerConfig�rgo.string."map.hdr[string]map[string]docker.dockerConfig"�|-map.hdr[string]map[string]docker.dockerConfig rgo.string."map.hdr[string]map[string]docker.dockerConfig"�\type.map.hdr[string]map[string]"".dockerConfig��0+;�  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000Prgo.string."map.hdr[string]map[string]docker.dockerConfig"pngo.weak.type.*map.hdr[string]map[string]"".dockerConfig�"runtime.zerovalue��\type.map.hdr[string]map[string]"".dockerConfig�&go.string."buckets"�dtype.*map.bucket[string]map[string]"".dockerConfig�,go.string."oldbuckets"�dtype.*map.bucket[string]map[string]"".dockerConfig�jgo.string."map[string]map[string]docker.dockerConfig"�t)map[string]map[string]docker.dockerConfig jgo.string."map[string]map[string]docker.dockerConfig"�Ttype.map[string]map[string]"".dockerConfig��O�9�5� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pjgo.string."map[string]map[string]docker.dockerConfig"pfgo.weak.type.*map[string]map[string]"".dockerConfig�"runtime.zerovalue�type.string�>type.map[string]"".dockerConfig�btype.map.bucket[string]map[string]"".dockerConfig�\type.map.hdr[string]map[string]"".dockerConfig��go.typelink.map[string]map[string]docker.dockerConfig/map[string]map[string]"".dockerConfigTtype.map[string]map[string]"".dockerConfig�lgo.string."*map[string]map[string]docker.dockerConfig"�v**map[string]map[string]docker.dockerConfig lgo.string."*map[string]map[string]docker.dockerConfig"�Vtype.*map[string]map[string]"".dockerConfig���u6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Plgo.string."*map[string]map[string]docker.dockerConfig"phgo.weak.type.**map[string]map[string]"".dockerConfig�"runtime.zerovalue�Ttype.map[string]map[string]"".dockerConfig�Vgo.string."*map[string]docker.dockerConfig"``*map[string]docker.dockerConfig Vgo.string."*map[string]docker.dockerConfig"�@type.*map[string]"".dockerConfig�����6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PVgo.string."*map[string]docker.dockerConfig"pRgo.weak.type.**map[string]"".dockerConfig�"runtime.zerovalue�>type.map[string]"".dockerConfig�&go.string."[]uint8"00[]uint8 &go.string."[]uint8"�type.[]uint8���~.8 � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P&go.string."[]uint8"p*go.weak.type.*[]uint8�"runtime.zerovalue�type.uint8�6go.typelink.[]uint8/[]uint8type.[]uint8�^go.string."*map.hdr[string]docker.dockerConfig"ph#*map.hdr[string]docker.dockerConfig ^go.string."*map.hdr[string]docker.dockerConfig"�Htype.*map.hdr[string]"".dockerConfig��B���6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P^go.string."*map.hdr[string]docker.dockerConfig"pZgo.weak.type.**map.hdr[string]"".dockerConfig�"runtime.zerovalue�Ftype.map.hdr[string]"".dockerConfig�*go.string."[]uintptr"@4 []uintptr *go.string."[]uintptr"�type.[]uintptr���3�] � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P*go.string."[]uintptr"p.go.weak.type.*[]uintptr�"runtime.zerovalue�type.uintptr�>go.typelink.[]uintptr/[]uintptrtype.[]uintptr�^runtime.gcbits.0x000000000000000000000000000000 �,go.string."[4]uintptr"@6 +[4]uintptr ,go.string."[4]uintptr"�type.[4]uintptr�� l<�  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P,go.string."[4]uintptr"p0go.weak.type.*[4]uintptr�"runtime.zerovalue�type.uintptr�type.[]uintptr�Bgo.typelink.[4]uintptr/[4]uintptrtype.[4]uintptr�bruntime.gcbits.0x88888844440000000000000000000000 ���DD�^go.string."map.iter[string]docker.dockerConfig"ph#map.iter[string]docker.dockerConfig ^go.string."map.iter[string]docker.dockerConfig"�go.string."key"0(key go.string."key"�go.string."val"0(val go.string."val"�go.string."t"0$t go.string."t"� go.string."bptr"0*bptr go.string."bptr"�"go.string."other"0,other "go.string."other"�Htype.map.iter[string]"".dockerConfig��P�b$ (0( � runtime.algarray0bruntime.gcbits.0x88888844440000000000000000000000P^go.string."map.iter[string]docker.dockerConfig"pZgo.weak.type.*map.iter[string]"".dockerConfig�"runtime.zerovalue��Htype.map.iter[string]"".dockerConfig�go.string."key"�type.*string�go.string."val"�*type.*"".dockerConfig�go.string."t"�type.*uint8�go.string."h"�Htype.*map.hdr[string]"".dockerConfig�&go.string."buckets"�Ntype.*map.bucket[string]"".dockerConfig� go.string."bptr"�Ntype.*map.bucket[string]"".dockerConfig�"go.string."other"�type.[4]uintptr�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�6type..hashfunc."".APIEvents.type..hash."".APIEvents�2type..eqfunc."".APIEvents*type..eq."".APIEvents�,type..alg."".APIEvents 6type..hashfunc."".APIEvents2type..eqfunc."".APIEvents�bruntime.gcbits.0x48484884848444000000000000000000 HHH���D�8go.string."docker.APIEvents"PBdocker.APIEvents 8go.string."docker.APIEvents"�$go.string."Status"0.Status $go.string."Status"�~go.string."json:\"Status,omitempty\" yaml:\"Status,omitempty\""��/json:"Status,omitempty" yaml:"Status,omitempty" ~go.string."json:\"Status,omitempty\" yaml:\"Status,omitempty\""�go.string."ID"0&ID go.string."ID"�ngo.string."json:\"ID,omitempty\" yaml:\"ID,omitempty\""pp'json:"ID,omitempty" yaml:"ID,omitempty" ngo.string."json:\"ID,omitempty\" yaml:\"ID,omitempty\""� go.string."From"0*From go.string."From"�vgo.string."json:\"From,omitempty\" yaml:\"From,omitempty\""�x+json:"From,omitempty" yaml:"From,omitempty" vgo.string."json:\"From,omitempty\" yaml:\"From,omitempty\""� go.string."Time"0*Time go.string."Time"�vgo.string."json:\"Time,omitempty\" yaml:\"Time,omitempty\""�x+json:"Time,omitempty" yaml:"Time,omitempty" vgo.string."json:\"Time,omitempty\" yaml:\"Time,omitempty\""�*go.string."APIEvents"@4 APIEvents *go.string."APIEvents"�"type."".APIEvents��8���� 0, ,type..alg."".APIEvents0bruntime.gcbits.0x48484884848444000000000000000000P8go.string."docker.APIEvents"p$type.*"".APIEvents�"runtime.zerovalue��"type."".APIEvents�$go.string."Status"�type.string�~go.string."json:\"Status,omitempty\" yaml:\"Status,omitempty\""�go.string."ID"�type.string�ngo.string."json:\"ID,omitempty\" yaml:\"ID,omitempty\""� go.string."From"�type.string�vgo.string."json:\"From,omitempty\" yaml:\"From,omitempty\""� go.string."Time"�type.int64�vgo.string."json:\"Time,omitempty\" yaml:\"Time,omitempty\""`�"type."".APIEvents�*go.string."APIEvents"�"go.importpath."".��"type."".APIEvents�:go.string."*docker.APIEvents"PD*docker.APIEvents :go.string."*docker.APIEvents"�$type.*"".APIEvents����6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."*docker.APIEvents"p6go.weak.type.**"".APIEvents�"runtime.zerovalue�"type."".APIEvents�Dgo.string."chan *docker.APIEvents"PNchan *docker.APIEvents Dgo.string."chan *docker.APIEvents"�.type.chan *"".APIEvents��P@�D2 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PDgo.string."chan *docker.APIEvents"p@go.weak.type.*chan *"".APIEvents�"runtime.zerovalue�$type.*"".APIEvents�jgo.typelink.chan *docker.APIEvents/chan *"".APIEvents.type.chan *"".APIEvents�,go.string."chan error"@6 +chan error ,go.string."chan error"�type.chan error��"��]2 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."chan error"p0go.weak.type.*chan error�"runtime.zerovalue�type.error�Bgo.typelink.chan error/chan errortype.chan error�Hgo.string."chan<- *docker.APIEvents"`Rchan<- *docker.APIEvents Hgo.string."chan<- *docker.APIEvents"�2type.chan<- *"".APIEvents��� �Q2 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PHgo.string."chan<- *docker.APIEvents"pDgo.weak.type.*chan<- *"".APIEvents�"runtime.zerovalue�$type.*"".APIEvents�rgo.typelink.chan<- *docker.APIEvents/chan<- *"".APIEvents2type.chan<- *"".APIEvents�Lgo.string."[]chan<- *docker.APIEvents"`V[]chan<- *docker.APIEvents Lgo.string."[]chan<- *docker.APIEvents"�6type.[]chan<- *"".APIEvents��%�= � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000PLgo.string."[]chan<- *docker.APIEvents"pHgo.weak.type.*[]chan<- *"".APIEvents�"runtime.zerovalue�2type.chan<- *"".APIEvents�zgo.typelink.[]chan<- *docker.APIEvents/[]chan<- *"".APIEvents6type.[]chan<- *"".APIEvents�bruntime.gcbits.0x44448484884844444448888844000000 DD���HDDDH��D�Ngo.string."docker.eventMonitoringState"`Xdocker.eventMonitoringState Ngo.string."docker.eventMonitoringState"�&go.string."enabled"00enabled &go.string."enabled"�(go.string."lastSeen"@2lastSeen (go.string."lastSeen"� go.string."errC"0*errC go.string."errC"�*go.string."listeners"@4 listeners *go.string."listeners"�@go.string."eventMonitoringState"PJeventMonitoringState @go.string."eventMonitoringState"�8type."".eventMonitoringState��h^}� 08@HP4 � runtime.algarray0bruntime.gcbits.0x44448484884844444448888844000000PNgo.string."docker.eventMonitoringState"p:type.*"".eventMonitoringState�"runtime.zerovalue��8type."".eventMonitoringState�"type.sync.RWMutex�&type.sync.WaitGroup�&go.string."enabled"�"go.importpath."".�type.bool�(go.string."lastSeen"�"go.importpath."".�type.*int64�go.string."C"�.type.chan *"".APIEvents� go.string."errC"�"go.importpath."".�type.chan error�*go.string."listeners"�"go.importpath."".�6type.[]chan<- *"".APIEvents`�8type."".eventMonitoringState�@go.string."eventMonitoringState"�"go.importpath."".��8type."".eventMonitoringState�Pgo.string."*docker.eventMonitoringState"`Z*docker.eventMonitoringState Pgo.string."*docker.eventMonitoringState"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·519efd86263089ddb84df3cfe7fd2992�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·519efd86263089ddb84df3cfe7fd2992�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·519efd86263089ddb84df3cfe7fd2992�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·519efd86263089ddb84df3cfe7fd2992�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·cd30d2bcfdea04ed7c49639580b4bd08�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·519efd86263089ddb84df3cfe7fd2992�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·519efd86263089ddb84df3cfe7fd2992�fgo.string."func(*docker.eventMonitoringState, int)"pp'func(*docker.eventMonitoringState, int) fgo.string."func(*docker.eventMonitoringState, int)"�Ptype.func(*"".eventMonitoringState, int)�� @{13 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pfgo.string."func(*docker.eventMonitoringState, int)"pbgo.weak.type.*func(*"".eventMonitoringState, int)�"runtime.zerovalue��Ptype.func(*"".eventMonitoringState, int)��Ptype.func(*"".eventMonitoringState, int)�:type.*"".eventMonitoringState�type.int�\go.string."func(*docker.eventMonitoringState)"pf"func(*docker.eventMonitoringState) \go.string."func(*docker.eventMonitoringState)"�Ftype.func(*"".eventMonitoringState)��u�v3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P\go.string."func(*docker.eventMonitoringState)"pXgo.weak.type.*func(*"".eventMonitoringState)�"runtime.zerovalue��Ftype.func(*"".eventMonitoringState)��Ftype.func(*"".eventMonitoringState)�:type.*"".eventMonitoringState�tgo.string."func(*docker.eventMonitoringState) sync.Locker"�~.func(*docker.eventMonitoringState) sync.Locker tgo.string."func(*docker.eventMonitoringState) sync.Locker"�^type.func(*"".eventMonitoringState) sync.Locker���S�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ptgo.string."func(*docker.eventMonitoringState) sync.Locker"ppgo.weak.type.*func(*"".eventMonitoringState) sync.Locker�"runtime.zerovalue��^type.func(*"".eventMonitoringState) sync.Locker��^type.func(*"".eventMonitoringState) sync.Locker�:type.*"".eventMonitoringState� type.sync.Locker��go.string."func(*docker.eventMonitoringState, chan<- *docker.APIEvents) error"��Bfunc(*docker.eventMonitoringState, chan<- *docker.APIEvents) error �go.string."func(*docker.eventMonitoringState, chan<- *docker.APIEvents) error"�~type.func(*"".eventMonitoringState, chan<- *"".APIEvents) error���Uv�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.eventMonitoringState, chan<- *docker.APIEvents) error"p�go.weak.type.*func(*"".eventMonitoringState, chan<- *"".APIEvents) error�"runtime.zerovalue��~type.func(*"".eventMonitoringState, chan<- *"".APIEvents) error��~type.func(*"".eventMonitoringState, chan<- *"".APIEvents) error�:type.*"".eventMonitoringState�2type.chan<- *"".APIEvents�type.error��go.string."func(*docker.eventMonitoringState, *docker.Client) error"��8func(*docker.eventMonitoringState, *docker.Client) error �go.string."func(*docker.eventMonitoringState, *docker.Client) error"�jtype.func(*"".eventMonitoringState, *"".Client) error��o��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.eventMonitoringState, *docker.Client) error"p|go.weak.type.*func(*"".eventMonitoringState, *"".Client) error�"runtime.zerovalue��jtype.func(*"".eventMonitoringState, *"".Client) error��jtype.func(*"".eventMonitoringState, *"".Client) error�:type.*"".eventMonitoringState�type.*"".Client�type.error�hgo.string."func(*docker.eventMonitoringState) error"�r(func(*docker.eventMonitoringState) error hgo.string."func(*docker.eventMonitoringState) error"�Rtype.func(*"".eventMonitoringState) error����o�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Phgo.string."func(*docker.eventMonitoringState) error"pdgo.weak.type.*func(*"".eventMonitoringState) error�"runtime.zerovalue��Rtype.func(*"".eventMonitoringState) error��Rtype.func(*"".eventMonitoringState) error�:type.*"".eventMonitoringState�type.error�fgo.string."func(*docker.eventMonitoringState) bool"pp'func(*docker.eventMonitoringState) bool fgo.string."func(*docker.eventMonitoringState) bool"�Ptype.func(*"".eventMonitoringState) bool���m��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pfgo.string."func(*docker.eventMonitoringState) bool"pbgo.weak.type.*func(*"".eventMonitoringState) bool�"runtime.zerovalue��Ptype.func(*"".eventMonitoringState) bool��Ptype.func(*"".eventMonitoringState) bool�:type.*"".eventMonitoringState�type.bool�|go.string."func(*docker.eventMonitoringState, *docker.Client)"��2func(*docker.eventMonitoringState, *docker.Client) |go.string."func(*docker.eventMonitoringState, *docker.Client)"�^type.func(*"".eventMonitoringState, *"".Client)��"8:"3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P|go.string."func(*docker.eventMonitoringState, *docker.Client)"ppgo.weak.type.*func(*"".eventMonitoringState, *"".Client)�"runtime.zerovalue��^type.func(*"".eventMonitoringState, *"".Client)��^type.func(*"".eventMonitoringState, *"".Client)�:type.*"".eventMonitoringState�type.*"".Client��go.string."func(*docker.eventMonitoringState, *docker.APIEvents)"��5func(*docker.eventMonitoringState, *docker.APIEvents) �go.string."func(*docker.eventMonitoringState, *docker.APIEvents)"�dtype.func(*"".eventMonitoringState, *"".APIEvents)���43 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.eventMonitoringState, *docker.APIEvents)"pvgo.weak.type.*func(*"".eventMonitoringState, *"".APIEvents)�"runtime.zerovalue��dtype.func(*"".eventMonitoringState, *"".APIEvents)��dtype.func(*"".eventMonitoringState, *"".APIEvents)�:type.*"".eventMonitoringState�$type.*"".APIEvents�go.string."Add"0(Add go.string."Add"�*go.string."func(int)"@4 func(int) *go.string."func(int)"�type.func(int)�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P*go.string."func(int)"p.go.weak.type.*func(int)�"runtime.zerovalue��type.func(int)��type.func(int)�type.int� go.string."Done"0*Done go.string."Done"�$go.string."func()"0.func() $go.string."func()"�type.func()������3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P$go.string."func()"p(go.weak.type.*func()�"runtime.zerovalue��type.func()��type.func()� go.string."Lock"0*Lock go.string."Lock"�"go.string."RLock"0,RLock "go.string."RLock"�&go.string."RLocker"00RLocker &go.string."RLocker"�closeListeners 4go.string."closeListeners"�8go.string."connectWithRetry"PBconnectWithRetry 8go.string."connectWithRetry"�Lgo.string."func(*docker.Client) error"`Vfunc(*docker.Client) error Lgo.string."func(*docker.Client) error"�6type.func(*"".Client) error��j��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PLgo.string."func(*docker.Client) error"pHgo.weak.type.*func(*"".Client) error�"runtime.zerovalue��6type.func(*"".Client) error��6type.func(*"".Client) error�type.*"".Client�type.error�Dgo.string."disableEventMonitoring"PNdisableEventMonitoring Dgo.string."disableEventMonitoring"�0go.string."func() error"@: func() error 0go.string."func() error"�"type.func() error����ֵ3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P0go.string."func() error"p4go.weak.type.*func() error�"runtime.zerovalue��"type.func() error��"type.func() error�type.error�Bgo.string."enableEventMonitoring"PLenableEventMonitoring Bgo.string."enableEventMonitoring"�*go.string."isEnabled"@4 isEnabled *go.string."isEnabled"�.go.string."func() bool"@8 func() bool .go.string."func() bool"� type.func() bool��T�x3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P.go.string."func() bool"p2go.weak.type.*func() bool�"runtime.zerovalue�� type.func() bool�� type.func() bool�type.bool�2go.string."monitorEvents"@< monitorEvents 2go.string."monitorEvents"�@go.string."func(*docker.Client)"PJfunc(*docker.Client) @go.string."func(*docker.Client)"�*type.func(*"".Client)���ӹ�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P@go.string."func(*docker.Client)"premoveListener 4go.string."removeListener"�*go.string."sendEvent"@4 sendEvent *go.string."sendEvent"�Fgo.string."func(*docker.APIEvents)"PPfunc(*docker.APIEvents) Fgo.string."func(*docker.APIEvents)"�0type.func(*"".APIEvents)�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PFgo.string."func(*docker.APIEvents)"pBgo.weak.type.*func(*"".APIEvents)�"runtime.zerovalue��0type.func(*"".APIEvents)��0type.func(*"".APIEvents)�$type.*"".APIEvents�4go.string."updateLastSeen"@>updateLastSeen 4go.string."updateLastSeen"�:type.*"".eventMonitoringState������6� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."*docker.eventMonitoringState"pLgo.weak.type.**"".eventMonitoringState�"runtime.zerovalue�8type."".eventMonitoringState`�:type.*"".eventMonitoringState��:type.*"".eventMonitoringState�go.string."Add"�type.func(int)�Ptype.func(*"".eventMonitoringState, int)�<"".(*eventMonitoringState).Add�<"".(*eventMonitoringState).Add� go.string."Done"�type.func()�Ftype.func(*"".eventMonitoringState)�>"".(*eventMonitoringState).Done�>"".(*eventMonitoringState).Done� go.string."Lock"�type.func()�Ftype.func(*"".eventMonitoringState)�>"".(*eventMonitoringState).Lock�>"".(*eventMonitoringState).Lock�"go.string."RLock"�type.func()�Ftype.func(*"".eventMonitoringState)�@"".(*eventMonitoringState).RLock�@"".(*eventMonitoringState).RLock�&go.string."RLocker"�.type.func() sync.Locker�^type.func(*"".eventMonitoringState) sync.Locker�D"".(*eventMonitoringState).RLocker�D"".(*eventMonitoringState).RLocker�&go.string."RUnlock"�type.func()�Ftype.func(*"".eventMonitoringState)�D"".(*eventMonitoringState).RUnlock�D"".(*eventMonitoringState).RUnlock�$go.string."Unlock"�type.func()�Ftype.func(*"".eventMonitoringState)�B"".(*eventMonitoringState).Unlock�B"".(*eventMonitoringState).Unlock� go.string."Wait"�type.func()�Ftype.func(*"".eventMonitoringState)�>"".(*eventMonitoringState).Wait�>"".(*eventMonitoringState).Wait�.go.string."addListener"�"go.importpath."".�Jtype.func(chan<- *"".APIEvents) error�~type.func(*"".eventMonitoringState, chan<- *"".APIEvents) error�L"".(*eventMonitoringState).addListener�L"".(*eventMonitoringState).addListener�4go.string."closeListeners"�"go.importpath."".�type.func()� Ftype.func(*"".eventMonitoringState)� R"".(*eventMonitoringState).closeListeners� R"".(*eventMonitoringState).closeListeners� 8go.string."connectWithRetry"� "go.importpath."".� 6type.func(*"".Client) error� jtype.func(*"".eventMonitoringState, *"".Client) error� V"".(*eventMonitoringState).connectWithRetry� +V"".(*eventMonitoringState).connectWithRetry� +Dgo.string."disableEventMonitoring"� +"go.importpath."".� +"type.func() error� +Rtype.func(*"".eventMonitoringState) error� +b"".(*eventMonitoringState).disableEventMonitoring� +b"".(*eventMonitoringState).disableEventMonitoring� +Bgo.string."enableEventMonitoring"� "go.importpath."".� 6type.func(*"".Client) error� jtype.func(*"".eventMonitoringState, *"".Client) error� `"".(*eventMonitoringState).enableEventMonitoring� `"".(*eventMonitoringState).enableEventMonitoring� *go.string."isEnabled"� "go.importpath."".�  type.func() bool� Ptype.func(*"".eventMonitoringState) bool� H"".(*eventMonitoringState).isEnabled� H"".(*eventMonitoringState).isEnabled� 2go.string."monitorEvents"� "go.importpath."".� *type.func(*"".Client)� ^type.func(*"".eventMonitoringState, *"".Client)� P"".(*eventMonitoringState).monitorEvents� P"".(*eventMonitoringState).monitorEvents� .go.string."noListeners"� "go.importpath."".�  type.func() bool� Ptype.func(*"".eventMonitoringState) bool� L"".(*eventMonitoringState).noListeners� L"".(*eventMonitoringState).noListeners� 4go.string."removeListener"�"go.importpath."".�Jtype.func(chan<- *"".APIEvents) error�~type.func(*"".eventMonitoringState, chan<- *"".APIEvents) error�R"".(*eventMonitoringState).removeListener�R"".(*eventMonitoringState).removeListener�*go.string."sendEvent"�"go.importpath."".�0type.func(*"".APIEvents)�dtype.func(*"".eventMonitoringState, *"".APIEvents)�H"".(*eventMonitoringState).sendEvent�H"".(*eventMonitoringState).sendEvent�4go.string."updateLastSeen"�"go.importpath."".�0type.func(*"".APIEvents)�dtype.func(*"".eventMonitoringState, *"".APIEvents)�R"".(*eventMonitoringState).updateLastSeen�R"".(*eventMonitoringState).updateLastSeen�(Q,3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."func(docker.APIVersion) bool"pLgo.weak.type.*func("".APIVersion) bool�"runtime.zerovalue��:type.func("".APIVersion) bool��:type.func("".APIVersion) bool�$type."".APIVersion�type.bool�2go.string."func() string"@< func() string 2go.string."func() string"�$type.func() string���m�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P2go.string."func() string"p6go.weak.type.*func() string�"runtime.zerovalue��$type.func() string��$type.func() string�type.string�Ngo.string."func(docker.APIVersion) int"`Xfunc(docker.APIVersion) int Ngo.string."func(docker.APIVersion) int"�8type.func("".APIVersion) int��m�q3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PNgo.string."func(docker.APIVersion) int"pJgo.weak.type.*func("".APIVersion) int�"runtime.zerovalue��8type.func("".APIVersion) int��8type.func("".APIVersion) int�$type."".APIVersion�type.int�&type.*"".APIVersion��{?׋6N � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ptype.func("".APIVersion) string��W�+3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PTgo.string."func(docker.APIVersion) string"pPgo.weak.type.*func("".APIVersion) string�"runtime.zerovalue��>type.func("".APIVersion) string��>type.func("".APIVersion) string�$type."".APIVersion�type.string�tgo.string."func(docker.APIVersion, docker.APIVersion) int"�~.func(docker.APIVersion, docker.APIVersion) int tgo.string."func(docker.APIVersion, docker.APIVersion) int"�Vtype.func("".APIVersion, "".APIVersion) int��[�@ 3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ptgo.string."func(docker.APIVersion, docker.APIVersion) int"phgo.weak.type.*func("".APIVersion, "".APIVersion) int�"runtime.zerovalue��Vtype.func("".APIVersion, "".APIVersion) int��Vtype.func("".APIVersion, "".APIVersion) int�$type."".APIVersion�$type."".APIVersion�type.int�$type."".APIVersion��D� R � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P:go.string."docker.APIVersion"p&type.*"".APIVersion�"runtime.zerovalue�type.int`�$type."".APIVersion�,go.string."APIVersion"�"go.importpath."".��$type."".APIVersion�.go.string."GreaterThan"�:type.func("".APIVersion) bool�Xtype.func("".APIVersion, "".APIVersion) bool�8"".(*APIVersion).GreaterThan�2"".APIVersion.GreaterThan�@go.string."GreaterThanOrEqualTo"�:type.func("".APIVersion) bool�Xtype.func("".APIVersion, "".APIVersion) bool�J"".(*APIVersion).GreaterThanOrEqualTo�D"".APIVersion.GreaterThanOrEqualTo�(go.string."LessThan"�:type.func("".APIVersion) bool�Xtype.func("".APIVersion, "".APIVersion) bool�2"".(*APIVersion).LessThan�,"".APIVersion.LessThan�:go.string."LessThanOrEqualTo"�:type.func("".APIVersion) bool�Xtype.func("".APIVersion, "".APIVersion) bool�D"".(*APIVersion).LessThanOrEqualTo�>"".APIVersion.LessThanOrEqualTo�$go.string."String"�$type.func() string�>type.func("".APIVersion) string�."".(*APIVersion).String�("".APIVersion.String�&go.string."compare"�"go.importpath."".�8type.func("".APIVersion) int�Vtype.func("".APIVersion, "".APIVersion) int�0"".(*APIVersion).compare�*"".APIVersion.compare�bruntime.gcbits.0x84888488444884440000000000000000 ����DH�D�2go.string."docker.Client"@< docker.Client 2go.string."docker.Client"�Dgo.string."SkipServerVersionCheck"PNSkipServerVersionCheck Dgo.string."SkipServerVersionCheck"�,go.string."HTTPClient"@6 +HTTPClient ,go.string."HTTPClient"�*go.string."TLSConfig"@4 TLSConfig *go.string."TLSConfig"�(go.string."endpoint"@2endpoint (go.string."endpoint"�.go.string."endpointURL"@8 endpointURL .go.string."endpointURL"�0go.string."eventMonitor"@: eventMonitor 0go.string."eventMonitor"�>go.string."requestedAPIVersion"PHrequestedAPIVersion >go.string."requestedAPIVersion"�8go.string."serverAPIVersion"PBserverAPIVersion 8go.string."serverAPIVersion"�go.string."requestedAPIVersion"�"go.importpath."".�$type."".APIVersion�8go.string."serverAPIVersion"�"go.importpath."".�$type."".APIVersion�*docker.Client 4go.string."*docker.Client"��go.string."func(*docker.Client, chan<- *docker.APIEvents) error"��4func(*docker.Client, chan<- *docker.APIEvents) error �go.string."func(*docker.Client, chan<- *docker.APIEvents) error"�btype.func(*"".Client, chan<- *"".APIEvents) error��fLT�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, chan<- *docker.APIEvents) error"ptgo.weak.type.*func(*"".Client, chan<- *"".APIEvents) error�"runtime.zerovalue��btype.func(*"".Client, chan<- *"".APIEvents) error��btype.func(*"".Client, chan<- *"".APIEvents) error�type.*"".Client�2type.chan<- *"".APIEvents�type.error�*go.string."struct {}"@4 struct {} *go.string."struct {}"�type.struct {}����'�  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P*go.string."struct {}"p.go.weak.type.*struct {}�"runtime.zerovalue��type.struct {}�4go.string."chan struct {}"@>chan struct {} 4go.string."chan struct {}"�&type.chan struct {}��S�^\2 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."chan struct {}"p8go.weak.type.*chan struct {}�"runtime.zerovalue�type.struct {}�Rgo.typelink.chan struct {}/chan struct {}&type.chan struct {}�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·af3107c17ee1ab6f9f33230b5c7e3062�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�Ttype..hashfunc."".AttachToContainerOptionsLtype..hash."".AttachToContainerOptions�Ptype..eqfunc."".AttachToContainerOptionsHtype..eq."".AttachToContainerOptions�Jtype..alg."".AttachToContainerOptions Ttype..hashfunc."".AttachToContainerOptionsPtype..eqfunc."".AttachToContainerOptions�Xgo.string."*docker.AttachToContainerOptions"pb *docker.AttachToContainerOptions Xgo.string."*docker.AttachToContainerOptions"�Btype.*"".AttachToContainerOptions��i���6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PXgo.string."*docker.AttachToContainerOptions"pTgo.weak.type.**"".AttachToContainerOptions�"runtime.zerovalue�@type."".AttachToContainerOptions�bruntime.gcbits.0x488c8c8c8484c4c8c848480000000000 H��������HH�Vgo.string."docker.AttachToContainerOptions"``docker.AttachToContainerOptions Vgo.string."docker.AttachToContainerOptions"�*go.string."Container"@4 Container *go.string."Container"�(go.string."qs:\"-\""0.qs:"-" (go.string."qs:\"-\""�.go.string."InputStream"@8 InputStream .go.string."InputStream"�0go.string."OutputStream"@: OutputStream 0go.string."OutputStream"�.go.string."ErrorStream"@8 ErrorStream .go.string."ErrorStream"� go.string."Logs"0*Logs go.string."Logs"�$go.string."Stream"0.Stream $go.string."Stream"�"go.string."Stdin"0,Stdin "go.string."Stdin"�$go.string."Stdout"0.Stdout $go.string."Stdout"�$go.string."Stderr"0.Stderr $go.string."Stderr"�&go.string."Success"00Success &go.string."Success"�.go.string."RawTerminal"@8 RawTerminal .go.string."RawTerminal"�Hgo.string."AttachToContainerOptions"`RAttachToContainerOptions Hgo.string."AttachToContainerOptions"�@type."".AttachToContainerOptions� � X�h�u  0@ABCDHPJ Jtype..alg."".AttachToContainerOptions0bruntime.gcbits.0x488c8c8c8484c4c8c848480000000000PVgo.string."docker.AttachToContainerOptions"pBtype.*"".AttachToContainerOptions�"runtime.zerovalue��@type."".AttachToContainerOptions�*go.string."Container"�type.string�(go.string."qs:\"-\""�.go.string."InputStream"�type.io.Reader�(go.string."qs:\"-\""�0go.string."OutputStream"�type.io.Writer�(go.string."qs:\"-\""�.go.string."ErrorStream"�type.io.Writer�(go.string."qs:\"-\""� go.string."Logs"�type.bool�$go.string."Stream"�type.bool�"go.string."Stdin"�type.bool�$go.string."Stdout"�type.bool�$go.string."Stderr"�type.bool�&go.string."Success"�&type.chan struct {}�.go.string."RawTerminal"�type.bool�(go.string."qs:\"-\""`�@type."".AttachToContainerOptions�Hgo.string."AttachToContainerOptions"�"go.importpath."".�� @type."".AttachToContainerOptions��go.string."func(*docker.Client, docker.AttachToContainerOptions) error"��;func(*docker.Client, docker.AttachToContainerOptions) error �go.string."func(*docker.Client, docker.AttachToContainerOptions) error"�ptype.func(*"".Client, "".AttachToContainerOptions) error���r߉3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, docker.AttachToContainerOptions) error"p�go.weak.type.*func(*"".Client, "".AttachToContainerOptions) error�"runtime.zerovalue��ptype.func(*"".Client, "".AttachToContainerOptions) error��ptype.func(*"".Client, "".AttachToContainerOptions) error�type.*"".Client�@type."".AttachToContainerOptions�type.error��go.string."func(*docker.Client, *docker.AuthConfiguration) error"��5func(*docker.Client, *docker.AuthConfiguration) error �go.string."func(*docker.Client, *docker.AuthConfiguration) error"�dtype.func(*"".Client, *"".AuthConfiguration) error��Q�r�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, *docker.AuthConfiguration) error"pvgo.weak.type.*func(*"".Client, *"".AuthConfiguration) error�"runtime.zerovalue��dtype.func(*"".Client, *"".AuthConfiguration) error��dtype.func(*"".Client, *"".AuthConfiguration) error�type.*"".Client�4type.*"".AuthConfiguration�type.error�Jgo.string."*docker.BuildImageOptions"`T*docker.BuildImageOptions Jgo.string."*docker.BuildImageOptions"�4type.*"".BuildImageOptions�����6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PJgo.string."*docker.BuildImageOptions"pFgo.weak.type.**"".BuildImageOptions�"runtime.zerovalue�2type."".BuildImageOptions�bruntime.gcbits.0x48484444488c8c848484848484480000 HHDDH��������H�Hgo.string."docker.BuildImageOptions"`Rdocker.BuildImageOptions Hgo.string."docker.BuildImageOptions"� go.string."Name"0*Name go.string."Name"�(go.string."qs:\"t\""0.qs:"t" (go.string."qs:\"t\""�,go.string."Dockerfile"@6 +Dockerfile ,go.string."Dockerfile"�:go.string."qs:\"dockerfile\""@@qs:"dockerfile" :go.string."qs:\"dockerfile\""�&go.string."NoCache"00NoCache &go.string."NoCache"�4go.string."qs:\"nocache\""@: qs:"nocache" 4go.string."qs:\"nocache\""�4go.string."SuppressOutput"@>SuppressOutput 4go.string."SuppressOutput"�(go.string."qs:\"q\""0.qs:"q" (go.string."qs:\"q\""� go.string."Pull"0*Pull go.string."Pull"�.go.string."qs:\"pull\""@4 qs:"pull" .go.string."qs:\"pull\""�4go.string."RmTmpContainer"@>RmTmpContainer 4go.string."RmTmpContainer"�*go.string."qs:\"rm\""00qs:"rm" *go.string."qs:\"rm\""�>go.string."ForceRmTmpContainer"PHForceRmTmpContainer >go.string."ForceRmTmpContainer"�4go.string."qs:\"forcerm\""@: qs:"forcerm" 4go.string."qs:\"forcerm\""�$go.string."Memory"0.Memory $go.string."Memory"�2go.string."qs:\"memory\""@8 qs:"memory" 2go.string."qs:\"memory\""�&go.string."Memswap"00Memswap &go.string."Memswap"�4go.string."qs:\"memswap\""@: qs:"memswap" 4go.string."qs:\"memswap\""�*go.string."CPUShares"@4 CPUShares *go.string."CPUShares"�8go.string."qs:\"cpushares\""@>qs:"cpushares" 8go.string."qs:\"cpushares\""�,go.string."CPUSetCPUs"@6 +CPUSetCPUs ,go.string."CPUSetCPUs"�:go.string."qs:\"cpusetcpus\""@@qs:"cpusetcpus" :go.string."qs:\"cpusetcpus\""�2go.string."RawJSONStream"@< RawJSONStream 2go.string."RawJSONStream"�$go.string."Remote"0.Remote $go.string."Remote"�2go.string."qs:\"remote\""@8 qs:"remote" 2go.string."qs:\"remote\""�.go.string."AuthConfigs"@8 AuthConfigs .go.string."AuthConfigs"�,go.string."ContextDir"@6 +ContextDir ,go.string."ContextDir"�:go.string."BuildImageOptions"PDBuildImageOptions :go.string."BuildImageOptions"�2type."".BuildImageOptions� � �"& W !"#$(08@P`px���� � runtime.algarray0bruntime.gcbits.0x48484444488c8c848484848484480000PHgo.string."docker.BuildImageOptions"p4type.*"".BuildImageOptions�"runtime.zerovalue��2type."".BuildImageOptions� go.string."Name"�type.string�(go.string."qs:\"t\""�,go.string."Dockerfile"�type.string�:go.string."qs:\"dockerfile\""�&go.string."NoCache"�type.bool�4go.string."qs:\"nocache\""�4go.string."SuppressOutput"�type.bool�(go.string."qs:\"q\""� go.string."Pull"�type.bool�.go.string."qs:\"pull\""�4go.string."RmTmpContainer"�type.bool�*go.string."qs:\"rm\""�>go.string."ForceRmTmpContainer"�type.bool�4go.string."qs:\"forcerm\""�$go.string."Memory"�type.int64�2go.string."qs:\"memory\""�&go.string."Memswap"�type.int64�4go.string."qs:\"memswap\""�*go.string."CPUShares"�type.int64�8go.string."qs:\"cpushares\""�,go.string."CPUSetCPUs"�type.string�:go.string."qs:\"cpusetcpus\""�.go.string."InputStream"�type.io.Reader�(go.string."qs:\"-\""� 0go.string."OutputStream"� type.io.Writer� (go.string."qs:\"-\""� 2go.string."RawJSONStream"� type.bool� +(go.string."qs:\"-\""� +$go.string."Remote"� +type.string� +2go.string."qs:\"remote\""� + go.string."Auth"� 2type."".AuthConfiguration� (go.string."qs:\"-\""� .go.string."AuthConfigs"� 4type."".AuthConfigurations� (go.string."qs:\"-\""� ,go.string."ContextDir"� type.string� (go.string."qs:\"-\""`� 2type."".BuildImageOptions� :go.string."BuildImageOptions"� "go.importpath."".� � 2type."".BuildImageOptions��go.string."func(*docker.Client, docker.BuildImageOptions) error"��4func(*docker.Client, docker.BuildImageOptions) error �go.string."func(*docker.Client, docker.BuildImageOptions) error"�btype.func(*"".Client, "".BuildImageOptions) error��`�c�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, docker.BuildImageOptions) error"ptgo.weak.type.*func(*"".Client, "".BuildImageOptions) error�"runtime.zerovalue��btype.func(*"".Client, "".BuildImageOptions) error��btype.func(*"".Client, "".BuildImageOptions) error�type.*"".Client�2type."".BuildImageOptions�type.error�0go.string."*docker.Port"@: *docker.Port 0go.string."*docker.Port"� go.string."Port"0*Port go.string."Port"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�"go.string."Proto"0,Proto "go.string."Proto"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Jgo.string."func(*docker.Port) string"`Tfunc(*docker.Port) string Jgo.string."func(*docker.Port) string"�4type.func(*"".Port) string��b)�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PJgo.string."func(*docker.Port) string"pFgo.weak.type.*func(*"".Port) string�"runtime.zerovalue��4type.func(*"".Port) string��4type.func(*"".Port) string�type.*"".Port�type.string�type.*"".Port��/hߠ6$ � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P0go.string."*docker.Port"p,go.weak.type.**"".Port�"runtime.zerovalue�type."".Port`�type.*"".Port��type.*"".Port� go.string."Port"�$type.func() string�4type.func(*"".Port) string�"".(*Port).Port�"".(*Port).Port�"go.string."Proto"�$type.func() string�4type.func(*"".Port) string� "".(*Port).Proto� "".(*Port).Proto�bruntime.gcbits.0x48000000000000000000000000000000 H�.go.string."docker.Port"@8 docker.Port .go.string."docker.Port"�Hgo.string."func(docker.Port) string"`Rfunc(docker.Port) string Hgo.string."func(docker.Port) string"�2type.func("".Port) string��^uDž3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PHgo.string."func(docker.Port) string"pDgo.weak.type.*func("".Port) string�"runtime.zerovalue��2type.func("".Port) string��2type.func("".Port) string�type."".Port�type.string�type."".Port���?M& � runtime.algarray0bruntime.gcbits.0x48000000000000000000000000000000P.go.string."docker.Port"ptype.*"".Port�"runtime.zerovalue`�type."".Port� go.string."Port"�"go.importpath."".��type."".Port� go.string."Port"�$type.func() string�2type.func("".Port) string�"".(*Port).Port�"".Port.Port�"go.string."Proto"�$type.func() string�2type.func("".Port) string� "".(*Port).Proto�"".Port.Proto�2go.string."[]docker.Port"@< []docker.Port 2go.string."[]docker.Port"�type.[]"".Port��?�# � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P2go.string."[]docker.Port"p.go.weak.type.*[]"".Port�"runtime.zerovalue�type."".Port�Fgo.typelink.[]docker.Port/[]"".Porttype.[]"".Port�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�2type..hashfunc.[8]"".Port*type..hash.[8]"".Port�.type..eqfunc.[8]"".Port&type..eq.[8]"".Port�(type..alg.[8]"".Port 2type..hashfunc.[8]"".Port.type..eqfunc.[8]"".Port�4go.string."[8]docker.Port"@>[8]docker.Port 4go.string."[8]docker.Port"�type.[8]"".Port���~/(� (type..alg.[8]"".Port0bruntime.gcbits.0x48484848484848480000000000000000P4go.string."[8]docker.Port"p0go.weak.type.*[8]"".Port�"runtime.zerovalue�type."".Port�type.[]"".Port�Jgo.typelink.[8]docker.Port/[8]"".Porttype.[8]"".Port�.go.string."[]struct {}"@8 []struct {} .go.string."[]struct {}"� type.[]struct {}���̥� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P.go.string."[]struct {}"p2go.weak.type.*[]struct {}�"runtime.zerovalue�type.struct {}�Fgo.typelink.[]struct {}/[]struct {} type.[]struct {}�0go.string."[8]struct {}"@: [8]struct {} 0go.string."[8]struct {}"�"type.[8]struct {}��>�y �  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P0go.string."[8]struct {}"p4go.weak.type.*[8]struct {}�"runtime.zerovalue�type.struct {}� type.[]struct {}�Jgo.typelink.[8]struct {}/[8]struct {}"type.[8]struct {}�Zgo.string."*map.bucket[docker.Port]struct {}"pd!*map.bucket[docker.Port]struct {} Zgo.string."*map.bucket[docker.Port]struct {}"�Dtype.*map.bucket["".Port]struct {}��1S6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PZgo.string."*map.bucket[docker.Port]struct {}"pVgo.weak.type.**map.bucket["".Port]struct {}�"runtime.zerovalue�Btype.map.bucket["".Port]struct {}�bruntime.gcbits.0x84848484848484848400000000000000 ����������Xgo.string."map.bucket[docker.Port]struct {}"pb map.bucket[docker.Port]struct {} Xgo.string."map.bucket[docker.Port]struct {}"�Btype.map.bucket["".Port]struct {}����(��� � runtime.algarray0bruntime.gcbits.0x84848484848484848400000000000000PXgo.string."map.bucket[docker.Port]struct {}"pTgo.weak.type.*map.bucket["".Port]struct {}�"runtime.zerovalue��Btype.map.bucket["".Port]struct {}� go.string."keys"�type.[8]"".Port�$go.string."values"�"type.[8]struct {}�(go.string."overflow"�Dtype.*map.bucket["".Port]struct {}�Rgo.string."map.hdr[docker.Port]struct {}"`\map.hdr[docker.Port]struct {} Rgo.string."map.hdr[docker.Port]struct {}"�Y� � runtime.algarray0Btype..gc.map.bucket[string]string@Jtype..gcprog.map.bucket[string]stringPHgo.string."map.bucket[string]string"pLgo.weak.type.*map.bucket[string]string�"runtime.zerovalue��:type.map.bucket[string]string� go.string."keys"�type.[8]string�$go.string."values"�type.[8]string�(go.string."overflow"�go.weak.type.*map[string]string�"runtime.zerovalue�type.string�type.string�:type.map.bucket[string]string�4type.map.hdr[string]string�^go.typelink.map[string]string/map[string]string,type.map[string]string�,$type..gc."".Config0�,type..gcprog."".Config.fVY�Y�e���e �2go.string."docker.Config"@< docker.Config 2go.string."docker.Config"�(go.string."Hostname"@2Hostname (go.string."Hostname"��go.string."json:\"Hostname,omitempty\" yaml:\"Hostname,omitempty\""��3json:"Hostname,omitempty" yaml:"Hostname,omitempty" �go.string."json:\"Hostname,omitempty\" yaml:\"Hostname,omitempty\""�,go.string."Domainname"@6 +Domainname ,go.string."Domainname"��go.string."json:\"Domainname,omitempty\" yaml:\"Domainname,omitempty\""��7json:"Domainname,omitempty" yaml:"Domainname,omitempty" �go.string."json:\"Domainname,omitempty\" yaml:\"Domainname,omitempty\""� go.string."User"0*User go.string."User"�vgo.string."json:\"User,omitempty\" yaml:\"User,omitempty\""�x+json:"User,omitempty" yaml:"User,omitempty" vgo.string."json:\"User,omitempty\" yaml:\"User,omitempty\""�~go.string."json:\"Memory,omitempty\" yaml:\"Memory,omitempty\""��/json:"Memory,omitempty" yaml:"Memory,omitempty" ~go.string."json:\"Memory,omitempty\" yaml:\"Memory,omitempty\""�,go.string."MemorySwap"@6 +MemorySwap ,go.string."MemorySwap"��go.string."json:\"MemorySwap,omitempty\" yaml:\"MemorySwap,omitempty\""��7json:"MemorySwap,omitempty" yaml:"MemorySwap,omitempty" �go.string."json:\"MemorySwap,omitempty\" yaml:\"MemorySwap,omitempty\""��go.string."json:\"CpuShares,omitempty\" yaml:\"CpuShares,omitempty\""��5json:"CpuShares,omitempty" yaml:"CpuShares,omitempty" �go.string."json:\"CpuShares,omitempty\" yaml:\"CpuShares,omitempty\""�$go.string."CPUSet"0.CPUSet $go.string."CPUSet"�~go.string."json:\"Cpuset,omitempty\" yaml:\"Cpuset,omitempty\""��/json:"Cpuset,omitempty" yaml:"Cpuset,omitempty" ~go.string."json:\"Cpuset,omitempty\" yaml:\"Cpuset,omitempty\""�.go.string."AttachStdin"@8 AttachStdin .go.string."AttachStdin"��go.string."json:\"AttachStdin,omitempty\" yaml:\"AttachStdin,omitempty\""��9json:"AttachStdin,omitempty" yaml:"AttachStdin,omitempty" �go.string."json:\"AttachStdin,omitempty\" yaml:\"AttachStdin,omitempty\""�0go.string."AttachStdout"@: AttachStdout 0go.string."AttachStdout"��go.string."json:\"AttachStdout,omitempty\" yaml:\"AttachStdout,omitempty\""��;json:"AttachStdout,omitempty" yaml:"AttachStdout,omitempty" �go.string."json:\"AttachStdout,omitempty\" yaml:\"AttachStdout,omitempty\""�0go.string."AttachStderr"@: AttachStderr 0go.string."AttachStderr"��go.string."json:\"AttachStderr,omitempty\" yaml:\"AttachStderr,omitempty\""��;json:"AttachStderr,omitempty" yaml:"AttachStderr,omitempty" �go.string."json:\"AttachStderr,omitempty\" yaml:\"AttachStderr,omitempty\""�*go.string."PortSpecs"@4 PortSpecs *go.string."PortSpecs"��go.string."json:\"PortSpecs,omitempty\" yaml:\"PortSpecs,omitempty\""��5json:"PortSpecs,omitempty" yaml:"PortSpecs,omitempty" �go.string."json:\"PortSpecs,omitempty\" yaml:\"PortSpecs,omitempty\""�0go.string."ExposedPorts"@: ExposedPorts 0go.string."ExposedPorts"��go.string."json:\"ExposedPorts,omitempty\" yaml:\"ExposedPorts,omitempty\""��;json:"ExposedPorts,omitempty" yaml:"ExposedPorts,omitempty" �go.string."json:\"ExposedPorts,omitempty\" yaml:\"ExposedPorts,omitempty\""�go.string."Tty"0(Tty go.string."Tty"�rgo.string."json:\"Tty,omitempty\" yaml:\"Tty,omitempty\""�t)json:"Tty,omitempty" yaml:"Tty,omitempty" rgo.string."json:\"Tty,omitempty\" yaml:\"Tty,omitempty\""�*go.string."OpenStdin"@4 OpenStdin *go.string."OpenStdin"��go.string."json:\"OpenStdin,omitempty\" yaml:\"OpenStdin,omitempty\""��5json:"OpenStdin,omitempty" yaml:"OpenStdin,omitempty" �go.string."json:\"OpenStdin,omitempty\" yaml:\"OpenStdin,omitempty\""�*go.string."StdinOnce"@4 StdinOnce *go.string."StdinOnce"��go.string."json:\"StdinOnce,omitempty\" yaml:\"StdinOnce,omitempty\""��5json:"StdinOnce,omitempty" yaml:"StdinOnce,omitempty" �go.string."json:\"StdinOnce,omitempty\" yaml:\"StdinOnce,omitempty\""�go.string."Env"0(Env go.string."Env"�rgo.string."json:\"Env,omitempty\" yaml:\"Env,omitempty\""�t)json:"Env,omitempty" yaml:"Env,omitempty" rgo.string."json:\"Env,omitempty\" yaml:\"Env,omitempty\""�go.string."Cmd"0(Cmd go.string."Cmd"�Jgo.string."json:\"Cmd\" yaml:\"Cmd\""PLjson:"Cmd" yaml:"Cmd" Jgo.string."json:\"Cmd\" yaml:\"Cmd\""�go.string."DNS"0(DNS go.string."DNS"�rgo.string."json:\"Dns,omitempty\" yaml:\"Dns,omitempty\""�t)json:"Dns,omitempty" yaml:"Dns,omitempty" rgo.string."json:\"Dns,omitempty\" yaml:\"Dns,omitempty\""�"go.string."Image"0,Image "go.string."Image"�zgo.string."json:\"Image,omitempty\" yaml:\"Image,omitempty\""�|-json:"Image,omitempty" yaml:"Image,omitempty" zgo.string."json:\"Image,omitempty\" yaml:\"Image,omitempty\""�&go.string."Volumes"00Volumes &go.string."Volumes"��go.string."json:\"Volumes,omitempty\" yaml:\"Volumes,omitempty\""��1json:"Volumes,omitempty" yaml:"Volumes,omitempty" �go.string."json:\"Volumes,omitempty\" yaml:\"Volumes,omitempty\""�.go.string."VolumesFrom"@8 VolumesFrom .go.string."VolumesFrom"��go.string."json:\"VolumesFrom,omitempty\" yaml:\"VolumesFrom,omitempty\""��9json:"VolumesFrom,omitempty" yaml:"VolumesFrom,omitempty" �go.string."json:\"VolumesFrom,omitempty\" yaml:\"VolumesFrom,omitempty\""�,go.string."WorkingDir"@6 +WorkingDir ,go.string."WorkingDir"��go.string."json:\"WorkingDir,omitempty\" yaml:\"WorkingDir,omitempty\""��7json:"WorkingDir,omitempty" yaml:"WorkingDir,omitempty" �go.string."json:\"WorkingDir,omitempty\" yaml:\"WorkingDir,omitempty\""�,go.string."MacAddress"@6 +MacAddress ,go.string."MacAddress"��go.string."json:\"MacAddress,omitempty\" yaml:\"MacAddress,omitempty\""��7json:"MacAddress,omitempty" yaml:"MacAddress,omitempty" �go.string."json:\"MacAddress,omitempty\" yaml:\"MacAddress,omitempty\""�,go.string."Entrypoint"@6 +Entrypoint ,go.string."Entrypoint"�fgo.string."json:\"Entrypoint\" yaml:\"Entrypoint\""ph#json:"Entrypoint" yaml:"Entrypoint" fgo.string."json:\"Entrypoint\" yaml:\"Entrypoint\""�6go.string."NetworkDisabled"@@NetworkDisabled 6go.string."NetworkDisabled"��go.string."json:\"NetworkDisabled,omitempty\" yaml:\"NetworkDisabled,omitempty\""��Ajson:"NetworkDisabled,omitempty" yaml:"NetworkDisabled,omitempty" �go.string."json:\"NetworkDisabled,omitempty\" yaml:\"NetworkDisabled,omitempty\""�0go.string."SecurityOpts"@: SecurityOpts 0go.string."SecurityOpts"��go.string."json:\"SecurityOpts,omitempty\" yaml:\"SecurityOpts,omitempty\""��;json:"SecurityOpts,omitempty" yaml:"SecurityOpts,omitempty" �go.string."json:\"SecurityOpts,omitempty\" yaml:\"SecurityOpts,omitempty\""�&go.string."OnBuild"00OnBuild &go.string."OnBuild"��go.string."json:\"OnBuild,omitempty\" yaml:\"OnBuild,omitempty\""��1json:"OnBuild,omitempty" yaml:"OnBuild,omitempty" �go.string."json:\"OnBuild,omitempty\" yaml:\"OnBuild,omitempty\""�$go.string."Labels"0.Labels $go.string."Labels"�~go.string."json:\"Labels,omitempty\" yaml:\"Labels,omitempty\""��/json:"Labels,omitempty" yaml:"Labels,omitempty" ~go.string."json:\"Labels,omitempty\" yaml:\"Labels,omitempty\""�$go.string."Config"0.Config $go.string."Config"�type."".Config��p�a0zY 08@HXYZ`x����������08Ph� � runtime.algarray0$type..gc."".Config@,type..gcprog."".ConfigP2go.string."docker.Config"ptype.*"".Config�"runtime.zerovalue��type."".Config�(go.string."Hostname"�type.string��go.string."json:\"Hostname,omitempty\" yaml:\"Hostname,omitempty\""�,go.string."Domainname"�type.string��go.string."json:\"Domainname,omitempty\" yaml:\"Domainname,omitempty\""� go.string."User"�type.string�vgo.string."json:\"User,omitempty\" yaml:\"User,omitempty\""�$go.string."Memory"�type.int64�~go.string."json:\"Memory,omitempty\" yaml:\"Memory,omitempty\""�,go.string."MemorySwap"�type.int64��go.string."json:\"MemorySwap,omitempty\" yaml:\"MemorySwap,omitempty\""�*go.string."CPUShares"�type.int64��go.string."json:\"CpuShares,omitempty\" yaml:\"CpuShares,omitempty\""�$go.string."CPUSet"�type.string�~go.string."json:\"Cpuset,omitempty\" yaml:\"Cpuset,omitempty\""�.go.string."AttachStdin"�type.bool��go.string."json:\"AttachStdin,omitempty\" yaml:\"AttachStdin,omitempty\""�0go.string."AttachStdout"�type.bool��go.string."json:\"AttachStdout,omitempty\" yaml:\"AttachStdout,omitempty\""�0go.string."AttachStderr"�type.bool��go.string."json:\"AttachStderr,omitempty\" yaml:\"AttachStderr,omitempty\""�*go.string."PortSpecs"�type.[]string��go.string."json:\"PortSpecs,omitempty\" yaml:\"PortSpecs,omitempty\""�0go.string."ExposedPorts"�4type.map["".Port]struct {}��go.string."json:\"ExposedPorts,omitempty\" yaml:\"ExposedPorts,omitempty\""� go.string."Tty"� type.bool� rgo.string."json:\"Tty,omitempty\" yaml:\"Tty,omitempty\""� *go.string."OpenStdin"� type.bool� +�go.string."json:\"OpenStdin,omitempty\" yaml:\"OpenStdin,omitempty\""� +*go.string."StdinOnce"� +type.bool� +�go.string."json:\"StdinOnce,omitempty\" yaml:\"StdinOnce,omitempty\""� +go.string."Env"� type.[]string� rgo.string."json:\"Env,omitempty\" yaml:\"Env,omitempty\""� go.string."Cmd"� type.[]string� Jgo.string."json:\"Cmd\" yaml:\"Cmd\""� go.string."DNS"� type.[]string� rgo.string."json:\"Dns,omitempty\" yaml:\"Dns,omitempty\""� "go.string."Image"� type.string� zgo.string."json:\"Image,omitempty\" yaml:\"Image,omitempty\""� &go.string."Volumes"� 2type.map[string]struct {}� �go.string."json:\"Volumes,omitempty\" yaml:\"Volumes,omitempty\""�.go.string."VolumesFrom"�type.string��go.string."json:\"VolumesFrom,omitempty\" yaml:\"VolumesFrom,omitempty\""�,go.string."WorkingDir"�type.string��go.string."json:\"WorkingDir,omitempty\" yaml:\"WorkingDir,omitempty\""�,go.string."MacAddress"�type.string��go.string."json:\"MacAddress,omitempty\" yaml:\"MacAddress,omitempty\""�,go.string."Entrypoint"�type.[]string�fgo.string."json:\"Entrypoint\" yaml:\"Entrypoint\""�6go.string."NetworkDisabled"�type.bool��go.string."json:\"NetworkDisabled,omitempty\" yaml:\"NetworkDisabled,omitempty\""�0go.string."SecurityOpts"�type.[]string��go.string."json:\"SecurityOpts,omitempty\" yaml:\"SecurityOpts,omitempty\""�&go.string."OnBuild"�type.[]string��go.string."json:\"OnBuild,omitempty\" yaml:\"OnBuild,omitempty\""�$go.string."Labels"�,type.map[string]string�~go.string."json:\"Labels,omitempty\" yaml:\"Labels,omitempty\""`�type."".Config�$go.string."Config"�"go.importpath."".��type."".Config�4go.string."*docker.Config"@>*docker.Config 4go.string."*docker.Config"�type.*"".Config���*�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."*docker.Config"p0go.weak.type.**"".Config�"runtime.zerovalue�type."".Config�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�Ptype..hashfunc."".CommitContainerOptionsHtype..hash."".CommitContainerOptions�Ltype..eqfunc."".CommitContainerOptionsDtype..eq."".CommitContainerOptions�Ftype..alg."".CommitContainerOptions Ptype..hashfunc."".CommitContainerOptionsLtype..eqfunc."".CommitContainerOptions�Tgo.string."*docker.CommitContainerOptions"`^*docker.CommitContainerOptions Tgo.string."*docker.CommitContainerOptions"�>type.*"".CommitContainerOptions����Jn6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PTgo.string."*docker.CommitContainerOptions"pPgo.weak.type.**"".CommitContainerOptions�"runtime.zerovalue�type.*"".CommitContainerOptions�"runtime.zerovalue��*docker.Change 4go.string."*docker.Change"�Ngo.string."func(*docker.Change) string"`Xfunc(*docker.Change) string Ngo.string."func(*docker.Change) string"�8type.func(*"".Change) string��.+e�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PNgo.string."func(*docker.Change) string"pJgo.weak.type.*func(*"".Change) string�"runtime.zerovalue��8type.func(*"".Change) string��8type.func(*"".Change) string�type.*"".Change�type.string�type.*"".Change��~IB�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."*docker.Change"p0go.weak.type.**"".Change�"runtime.zerovalue�type."".Change`�type.*"".Change��type.*"".Change�$go.string."String"�$type.func() string�8type.func(*"".Change) string�&"".(*Change).String�&"".(*Change).String�2go.string."docker.Change"@< docker.Change 2go.string."docker.Change"� go.string."Path"0*Path go.string."Path"� go.string."Kind"0*Kind go.string."Kind"�$go.string."Change"0.Change $go.string."Change"�type."".Change���뷲 &type..alg."".Change0bruntime.gcbits.0x48844400000000000000000000000000P2go.string."docker.Change"ptype.*"".Change�"runtime.zerovalue��type."".Change� go.string."Path"�type.string� go.string."Kind"�$type."".ChangeType`�type."".Change�$go.string."Change"�"go.importpath."".��type."".Change�6go.string."[]docker.Change"@@[]docker.Change 6go.string."[]docker.Change"� type.[]"".Change��f~|; � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P6go.string."[]docker.Change"p2go.weak.type.*[]"".Change�"runtime.zerovalue�type."".Change�Ngo.typelink.[]docker.Change/[]"".Change type.[]"".Change��go.string."func(*docker.Client, string) ([]docker.Change, error)"��5func(*docker.Client, string) ([]docker.Change, error) �go.string."func(*docker.Client, string) ([]docker.Change, error)"�dtype.func(*"".Client, string) ([]"".Change, error)��ˆ��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, string) ([]docker.Change, error)"pvgo.weak.type.*func(*"".Client, string) ([]"".Change, error)�"runtime.zerovalue��dtype.func(*"".Client, string) ([]"".Change, error)��dtype.func(*"".Client, string) ([]"".Change, error)�type.*"".Client�type.string� type.[]"".Change�type.error�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·e13351f28add7c60853cb3aac0a0e34e�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�Ttype..hashfunc."".CopyFromContainerOptionsLtype..hash."".CopyFromContainerOptions�Ptype..eqfunc."".CopyFromContainerOptionsHtype..eq."".CopyFromContainerOptions�Jtype..alg."".CopyFromContainerOptions Ttype..hashfunc."".CopyFromContainerOptionsPtype..eqfunc."".CopyFromContainerOptions�Xgo.string."*docker.CopyFromContainerOptions"pb *docker.CopyFromContainerOptions Xgo.string."*docker.CopyFromContainerOptions"�Btype.*"".CopyFromContainerOptions�����26 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PXgo.string."*docker.CopyFromContainerOptions"pTgo.weak.type.**"".CopyFromContainerOptions�"runtime.zerovalue�@type."".CopyFromContainerOptions�bruntime.gcbits.0x8c484800000000000000000000000000 �HH�Vgo.string."docker.CopyFromContainerOptions"``docker.CopyFromContainerOptions Vgo.string."docker.CopyFromContainerOptions"�,go.string."json:\"-\""@2json:"-" ,go.string."json:\"-\""�(go.string."Resource"@2Resource (go.string."Resource"�Hgo.string."CopyFromContainerOptions"`RCopyFromContainerOptions Hgo.string."CopyFromContainerOptions"�@type."".CopyFromContainerOptions��0��� $ Jtype..alg."".CopyFromContainerOptions0bruntime.gcbits.0x8c484800000000000000000000000000PVgo.string."docker.CopyFromContainerOptions"pBtype.*"".CopyFromContainerOptions�"runtime.zerovalue��@type."".CopyFromContainerOptions�0go.string."OutputStream"�type.io.Writer�,go.string."json:\"-\""�*go.string."Container"�type.string�,go.string."json:\"-\""�(go.string."Resource"�type.string`�@type."".CopyFromContainerOptions�Hgo.string."CopyFromContainerOptions"�"go.importpath."".��@type."".CopyFromContainerOptions��go.string."func(*docker.Client, docker.CopyFromContainerOptions) error"��;func(*docker.Client, docker.CopyFromContainerOptions) error �go.string."func(*docker.Client, docker.CopyFromContainerOptions) error"�ptype.func(*"".Client, "".CopyFromContainerOptions) error����ޭ3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, docker.CopyFromContainerOptions) error"p�go.weak.type.*func(*"".Client, "".CopyFromContainerOptions) error�"runtime.zerovalue��ptype.func(*"".Client, "".CopyFromContainerOptions) error��ptype.func(*"".Client, "".CopyFromContainerOptions) error�type.*"".Client�@type."".CopyFromContainerOptions�type.error�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�go.string."docker.KeyValuePair"PHdocker.KeyValuePair >go.string."docker.KeyValuePair"�go.string."Key"0(Key go.string."Key"�rgo.string."json:\"Key,omitempty\" yaml:\"Key,omitempty\""�t)json:"Key,omitempty" yaml:"Key,omitempty" rgo.string."json:\"Key,omitempty\" yaml:\"Key,omitempty\""�"go.string."Value"0,Value "go.string."Value"�zgo.string."json:\"Value,omitempty\" yaml:\"Value,omitempty\""�|-json:"Value,omitempty" yaml:"Value,omitempty" zgo.string."json:\"Value,omitempty\" yaml:\"Value,omitempty\""�0go.string."KeyValuePair"@: KeyValuePair 0go.string."KeyValuePair"�(type."".KeyValuePair�� ��� 2type..alg."".KeyValuePair0bruntime.gcbits.0x48480000000000000000000000000000P>go.string."docker.KeyValuePair"p*type.*"".KeyValuePair�"runtime.zerovalue��(type."".KeyValuePair�go.string."Key"�type.string�rgo.string."json:\"Key,omitempty\" yaml:\"Key,omitempty\""�"go.string."Value"�type.string�zgo.string."json:\"Value,omitempty\" yaml:\"Value,omitempty\""`�(type."".KeyValuePair�0go.string."KeyValuePair"�"go.importpath."".��(type."".KeyValuePair�Bgo.string."[]docker.KeyValuePair"PL[]docker.KeyValuePair Bgo.string."[]docker.KeyValuePair"�,type.[]"".KeyValuePair��_9�� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000PBgo.string."[]docker.KeyValuePair"p>go.weak.type.*[]"".KeyValuePair�"runtime.zerovalue�(type."".KeyValuePair�fgo.typelink.[]docker.KeyValuePair/[]"".KeyValuePair,type.[]"".KeyValuePair�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�:type..hashfunc."".PortBinding2type..hash."".PortBinding�6type..eqfunc."".PortBinding.type..eq."".PortBinding�0type..alg."".PortBinding :type..hashfunc."".PortBinding6type..eqfunc."".PortBinding�>go.string."*docker.PortBinding"PH*docker.PortBinding >go.string."*docker.PortBinding"�(type.*"".PortBinding��>!�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."*docker.PortBinding"p:go.weak.type.**"".PortBinding�"runtime.zerovalue�&type."".PortBinding�type..hashfunc."".RestartPolicy6type..hash."".RestartPolicy�:type..eqfunc."".RestartPolicy2type..eq."".RestartPolicy�4type..alg."".RestartPolicy >type..hashfunc."".RestartPolicy:type..eqfunc."".RestartPolicy�Bgo.string."*docker.RestartPolicy"PL*docker.RestartPolicy Bgo.string."*docker.RestartPolicy"�,type.*"".RestartPolicy��۰�J6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."*docker.RestartPolicy"p>go.weak.type.**"".RestartPolicy�"runtime.zerovalue�*type."".RestartPolicy�@go.string."docker.RestartPolicy"PJdocker.RestartPolicy @go.string."docker.RestartPolicy"�vgo.string."json:\"Name,omitempty\" yaml:\"Name,omitempty\""�x+json:"Name,omitempty" yaml:"Name,omitempty" vgo.string."json:\"Name,omitempty\" yaml:\"Name,omitempty\""�:go.string."MaximumRetryCount"PDMaximumRetryCount :go.string."MaximumRetryCount"��go.string."json:\"MaximumRetryCount,omitempty\" yaml:\"MaximumRetryCount,omitempty\""��Ejson:"MaximumRetryCount,omitempty" yaml:"MaximumRetryCount,omitempty" �go.string."json:\"MaximumRetryCount,omitempty\" yaml:\"MaximumRetryCount,omitempty\""�2go.string."RestartPolicy"@< RestartPolicy 2go.string."RestartPolicy"�*type."".RestartPolicy���)6� 4type..alg."".RestartPolicy0bruntime.gcbits.0x48844400000000000000000000000000P@go.string."docker.RestartPolicy"p,type.*"".RestartPolicy�"runtime.zerovalue��*type."".RestartPolicy� go.string."Name"�type.string�vgo.string."json:\"Name,omitempty\" yaml:\"Name,omitempty\""�:go.string."MaximumRetryCount"�type.int��go.string."json:\"MaximumRetryCount,omitempty\" yaml:\"MaximumRetryCount,omitempty\""`�*type."".RestartPolicy�2go.string."RestartPolicy"�"go.importpath."".��*type."".RestartPolicy�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�0type..hashfunc."".Device(type..hash."".Device�,type..eqfunc."".Device$type..eq."".Device�&type..alg."".Device 0type..hashfunc."".Device,type..eqfunc."".Device�4go.string."*docker.Device"@>*docker.Device 4go.string."*docker.Device"�type.*"".Device��un�|6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."*docker.Device"p0go.weak.type.**"".Device�"runtime.zerovalue�type."".Device�2go.string."docker.Device"@< docker.Device 2go.string."docker.Device"�,go.string."PathOnHost"@6 +PathOnHost ,go.string."PathOnHost"��go.string."json:\"PathOnHost,omitempty\" yaml:\"PathOnHost,omitempty\""��7json:"PathOnHost,omitempty" yaml:"PathOnHost,omitempty" �go.string."json:\"PathOnHost,omitempty\" yaml:\"PathOnHost,omitempty\""�6go.string."PathInContainer"@@PathInContainer 6go.string."PathInContainer"��go.string."json:\"PathInContainer,omitempty\" yaml:\"PathInContainer,omitempty\""��Ajson:"PathInContainer,omitempty" yaml:"PathInContainer,omitempty" �go.string."json:\"PathInContainer,omitempty\" yaml:\"PathInContainer,omitempty\""�:go.string."CgroupPermissions"PDCgroupPermissions :go.string."CgroupPermissions"��go.string."json:\"CgroupPermissions,omitempty\" yaml:\"CgroupPermissions,omitempty\""��Ejson:"CgroupPermissions,omitempty" yaml:"CgroupPermissions,omitempty" �go.string."json:\"CgroupPermissions,omitempty\" yaml:\"CgroupPermissions,omitempty\""�$go.string."Device"0.Device $go.string."Device"�type."".Device��0 ��� & &type..alg."".Device0bruntime.gcbits.0x48484800000000000000000000000000P2go.string."docker.Device"ptype.*"".Device�"runtime.zerovalue��type."".Device�,go.string."PathOnHost"�type.string��go.string."json:\"PathOnHost,omitempty\" yaml:\"PathOnHost,omitempty\""�6go.string."PathInContainer"�type.string��go.string."json:\"PathInContainer,omitempty\" yaml:\"PathInContainer,omitempty\""�:go.string."CgroupPermissions"�type.string��go.string."json:\"CgroupPermissions,omitempty\" yaml:\"CgroupPermissions,omitempty\""`�type."".Device�$go.string."Device"�"go.importpath."".��type."".Device�6go.string."[]docker.Device"@@[]docker.Device 6go.string."[]docker.Device"� type.[]"".Device��jjW� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P6go.string."[]docker.Device"p2go.weak.type.*[]"".Device�"runtime.zerovalue�type."".Device�Ngo.typelink.[]docker.Device/[]"".Device type.[]"".Device�:go.string."*docker.LogConfig"PD*docker.LogConfig :go.string."*docker.LogConfig"�$type.*"".LogConfig���(��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."*docker.LogConfig"p6go.weak.type.**"".LogConfig�"runtime.zerovalue�"type."".LogConfig�bruntime.gcbits.0x48888400000000000000000000000000 H���8go.string."docker.LogConfig"PBdocker.LogConfig 8go.string."docker.LogConfig"� go.string."Type"0*Type go.string."Type"�vgo.string."json:\"Type,omitempty\" yaml:\"Type,omitempty\""�x+json:"Type,omitempty" yaml:"Type,omitempty" vgo.string."json:\"Type,omitempty\" yaml:\"Type,omitempty\""�*go.string."LogConfig"@4 LogConfig *go.string."LogConfig"�"type."".LogConfig���z� � runtime.algarray0bruntime.gcbits.0x48888400000000000000000000000000P8go.string."docker.LogConfig"p$type.*"".LogConfig�"runtime.zerovalue��"type."".LogConfig� go.string."Type"�type.string�vgo.string."json:\"Type,omitempty\" yaml:\"Type,omitempty\""�$go.string."Config"�,type.map[string]string�~go.string."json:\"Config,omitempty\" yaml:\"Config,omitempty\""`�"type."".LogConfig�*go.string."LogConfig"�"go.importpath."".��"type."".LogConfig�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�0type..hashfunc."".ULimit(type..hash."".ULimit�,type..eqfunc."".ULimit$type..eq."".ULimit�&type..alg."".ULimit 0type..hashfunc."".ULimit,type..eqfunc."".ULimit�4go.string."*docker.ULimit"@>*docker.ULimit 4go.string."*docker.ULimit"�type.*"".ULimit��և�<6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."*docker.ULimit"p0go.weak.type.**"".ULimit�"runtime.zerovalue�type."".ULimit�bruntime.gcbits.0x48440000000000000000000000000000 HD�2go.string."docker.ULimit"@< docker.ULimit 2go.string."docker.ULimit"� go.string."Soft"0*Soft go.string."Soft"�vgo.string."json:\"Soft,omitempty\" yaml:\"Soft,omitempty\""�x+json:"Soft,omitempty" yaml:"Soft,omitempty" vgo.string."json:\"Soft,omitempty\" yaml:\"Soft,omitempty\""� go.string."Hard"0*Hard go.string."Hard"�vgo.string."json:\"Hard,omitempty\" yaml:\"Hard,omitempty\""�x+json:"Hard,omitempty" yaml:"Hard,omitempty" vgo.string."json:\"Hard,omitempty\" yaml:\"Hard,omitempty\""�$go.string."ULimit"0.ULimit $go.string."ULimit"�type."".ULimit�� ��G& &type..alg."".ULimit0bruntime.gcbits.0x48440000000000000000000000000000P2go.string."docker.ULimit"ptype.*"".ULimit�"runtime.zerovalue��type."".ULimit� go.string."Name"�type.string�vgo.string."json:\"Name,omitempty\" yaml:\"Name,omitempty\""� go.string."Soft"�type.int64�vgo.string."json:\"Soft,omitempty\" yaml:\"Soft,omitempty\""� go.string."Hard"�type.int64�vgo.string."json:\"Hard,omitempty\" yaml:\"Hard,omitempty\""`�type."".ULimit�$go.string."ULimit"�"go.importpath."".��type."".ULimit�6go.string."[]docker.ULimit"@@[]docker.ULimit 6go.string."[]docker.ULimit"� type.[]"".ULimit��ReadonlyRootfs 4go.string."ReadonlyRootfs"��go.string."json:\"ReadonlyRootfs,omitempty\" yaml:\"ReadonlyRootfs,omitempty\""��?json:"ReadonlyRootfs,omitempty" yaml:"ReadonlyRootfs,omitempty" �go.string."json:\"ReadonlyRootfs,omitempty\" yaml:\"ReadonlyRootfs,omitempty\""�.go.string."SecurityOpt"@8 SecurityOpt .go.string."SecurityOpt"��go.string."json:\"SecurityOpt,omitempty\" yaml:\"SecurityOpt,omitempty\""��9json:"SecurityOpt,omitempty" yaml:"SecurityOpt,omitempty" �go.string."json:\"SecurityOpt,omitempty\" yaml:\"SecurityOpt,omitempty\""�0go.string."CgroupParent"@: CgroupParent 0go.string."CgroupParent"��go.string."json:\"CgroupParent,omitempty\" yaml:\"CgroupParent,omitempty\""��;json:"CgroupParent,omitempty" yaml:"CgroupParent,omitempty" �go.string."json:\"CgroupParent,omitempty\" yaml:\"CgroupParent,omitempty\""�(go.string."CPUQuota"@2CPUQuota (go.string."CPUQuota"��go.string."json:\"CpuQuota,omitempty\" yaml:\"CpuQuota,omitempty\""��3json:"CpuQuota,omitempty" yaml:"CpuQuota,omitempty" �go.string."json:\"CpuQuota,omitempty\" yaml:\"CpuQuota,omitempty\""�*go.string."CPUPeriod"@4 CPUPeriod *go.string."CPUPeriod"��go.string."json:\"CpuPeriod,omitempty\" yaml:\"CpuPeriod,omitempty\""��5json:"CpuPeriod,omitempty" yaml:"CpuPeriod,omitempty" �go.string."json:\"CpuPeriod,omitempty\" yaml:\"CpuPeriod,omitempty\""�&go.string."Ulimits"00Ulimits &go.string."Ulimits"��go.string."json:\"Ulimits,omitempty\" yaml:\"Ulimits,omitempty\""��1json:"Ulimits,omitempty" yaml:"Ulimits,omitempty" �go.string."json:\"Ulimits,omitempty\" yaml:\"Ulimits,omitempty\""�,go.string."HostConfig"@6 +HostConfig ,go.string."HostConfig"�$type."".HostConfig����,Y0HXpx������ 0@Xp����������� � runtime.algarray@4type..gcprog."".HostConfigP:go.string."docker.HostConfig"p&type.*"".HostConfig�"runtime.zerovalue��$type."".HostConfig�"go.string."Binds"�type.[]string�zgo.string."json:\"Binds,omitempty\" yaml:\"Binds,omitempty\""�$go.string."CapAdd"�type.[]string�~go.string."json:\"CapAdd,omitempty\" yaml:\"CapAdd,omitempty\""�&go.string."CapDrop"�type.[]string��go.string."json:\"CapDrop,omitempty\" yaml:\"CapDrop,omitempty\""�6go.string."ContainerIDFile"�type.string��go.string."json:\"ContainerIDFile,omitempty\" yaml:\"ContainerIDFile,omitempty\""�&go.string."LxcConf"�,type.[]"".KeyValuePair��go.string."json:\"LxcConf,omitempty\" yaml:\"LxcConf,omitempty\""�,go.string."Privileged"�type.bool��go.string."json:\"Privileged,omitempty\" yaml:\"Privileged,omitempty\""�0go.string."PortBindings"�Btype.map["".Port][]"".PortBinding��go.string."json:\"PortBindings,omitempty\" yaml:\"PortBindings,omitempty\""�"go.string."Links"�type.[]string�zgo.string."json:\"Links,omitempty\" yaml:\"Links,omitempty\""�6go.string."PublishAllPorts"�type.bool��go.string."json:\"PublishAllPorts,omitempty\" yaml:\"PublishAllPorts,omitempty\""�go.string."DNS"�type.[]string�rgo.string."json:\"Dns,omitempty\" yaml:\"Dns,omitempty\""�*go.string."DNSSearch"�type.[]string��go.string."json:\"DnsSearch,omitempty\" yaml:\"DnsSearch,omitempty\""�,go.string."ExtraHosts"�type.[]string��go.string."json:\"ExtraHosts,omitempty\" yaml:\"ExtraHosts,omitempty\""� .go.string."VolumesFrom"� type.[]string� �go.string."json:\"VolumesFrom,omitempty\" yaml:\"VolumesFrom,omitempty\""� .go.string."NetworkMode"� type.string� +�go.string."json:\"NetworkMode,omitempty\" yaml:\"NetworkMode,omitempty\""� +&go.string."IpcMode"� +type.string� +�go.string."json:\"IpcMode,omitempty\" yaml:\"IpcMode,omitempty\""� +&go.string."PidMode"� type.string� �go.string."json:\"PidMode,omitempty\" yaml:\"PidMode,omitempty\""� &go.string."UTSMode"� type.string� �go.string."json:\"UTSMode,omitempty\" yaml:\"UTSMode,omitempty\""� 2go.string."RestartPolicy"� *type."".RestartPolicy� �go.string."json:\"RestartPolicy,omitempty\" yaml:\"RestartPolicy,omitempty\""� &go.string."Devices"�  type.[]"".Device� �go.string."json:\"Devices,omitempty\" yaml:\"Devices,omitempty\""� *go.string."LogConfig"� "type."".LogConfig� �go.string."json:\"LogConfig,omitempty\" yaml:\"LogConfig,omitempty\""�4go.string."ReadonlyRootfs"�type.bool��go.string."json:\"ReadonlyRootfs,omitempty\" yaml:\"ReadonlyRootfs,omitempty\""�.go.string."SecurityOpt"�type.[]string��go.string."json:\"SecurityOpt,omitempty\" yaml:\"SecurityOpt,omitempty\""�0go.string."CgroupParent"�type.string��go.string."json:\"CgroupParent,omitempty\" yaml:\"CgroupParent,omitempty\""�$go.string."Memory"�type.int64�~go.string."json:\"Memory,omitempty\" yaml:\"Memory,omitempty\""�,go.string."MemorySwap"�type.int64��go.string."json:\"MemorySwap,omitempty\" yaml:\"MemorySwap,omitempty\""�*go.string."CPUShares"�type.int64��go.string."json:\"CpuShares,omitempty\" yaml:\"CpuShares,omitempty\""�$go.string."CPUSet"�type.string�~go.string."json:\"Cpuset,omitempty\" yaml:\"Cpuset,omitempty\""�(go.string."CPUQuota"�type.int64��go.string."json:\"CpuQuota,omitempty\" yaml:\"CpuQuota,omitempty\""�*go.string."CPUPeriod"�type.int64��go.string."json:\"CpuPeriod,omitempty\" yaml:\"CpuPeriod,omitempty\""�&go.string."Ulimits"� type.[]"".ULimit��go.string."json:\"Ulimits,omitempty\" yaml:\"Ulimits,omitempty\""`�$type."".HostConfig�,go.string."HostConfig"�"go.importpath."".��$type."".HostConfig��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ptype.*"".CreateContainerOptions��7t�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PTgo.string."*docker.CreateContainerOptions"pPgo.weak.type.**"".CreateContainerOptions�"runtime.zerovalue�type.*"".CreateContainerOptions�"runtime.zerovalue�� � runtime.algarray0bruntime.gcbits.0x48484848448884848444840000000000P8go.string."docker.SwarmNode"p$type.*"".SwarmNode�"runtime.zerovalue��"type."".SwarmNode�go.string."ID"�type.string�ngo.string."json:\"ID,omitempty\" yaml:\"ID,omitempty\""�go.string."IP"�type.string�ngo.string."json:\"IP,omitempty\" yaml:\"IP,omitempty\""� go.string."Addr"�type.string�vgo.string."json:\"Addr,omitempty\" yaml:\"Addr,omitempty\""� go.string."Name"�type.string�vgo.string."json:\"Name,omitempty\" yaml:\"Name,omitempty\""� go.string."CPUs"�type.int64�vgo.string."json:\"CPUs,omitempty\" yaml:\"CPUs,omitempty\""�$go.string."Memory"�type.int64�~go.string."json:\"Memory,omitempty\" yaml:\"Memory,omitempty\""�$go.string."Labels"�,type.map[string]string�~go.string."json:\"Labels,omitempty\" yaml:\"Labels,omitempty\""`�"type."".SwarmNode�*go.string."SwarmNode"�"go.importpath."".��"type."".SwarmNode�:go.string."*docker.SwarmNode"PD*docker.SwarmNode :go.string."*docker.SwarmNode"�$type.*"".SwarmNode��`L�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."*docker.SwarmNode"p6go.weak.type.**"".SwarmNode�"runtime.zerovalue�"type."".SwarmNode�>go.string."*docker.PortMapping"PH*docker.PortMapping >go.string."*docker.PortMapping"�(type.*"".PortMapping��i�ޑ6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."*docker.PortMapping"p:go.weak.type.**"".PortMapping�"runtime.zerovalue�&type."".PortMapping�go.weak.type.*[8]"".PortMapping�"runtime.zerovalue�&type."".PortMapping�*type.[]"".PortMapping�fgo.typelink.[8]docker.PortMapping/[8]"".PortMapping,type.[8]"".PortMapping�bgo.string."*map.bucket[string]docker.PortMapping"pl%*map.bucket[string]docker.PortMapping bgo.string."*map.bucket[string]docker.PortMapping"�Ltype.*map.bucket[string]"".PortMapping����16 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pbgo.string."*map.bucket[string]docker.PortMapping"p^go.weak.type.**map.bucket[string]"".PortMapping�"runtime.zerovalue�Jtype.map.bucket[string]"".PortMapping�`go.string."map.bucket[string]docker.PortMapping"pj$map.bucket[string]docker.PortMapping `go.string."map.bucket[string]docker.PortMapping"�Jtype.map.bucket[string]"".PortMapping�����~��� � runtime.algarray0bruntime.gcbits.0x84848484848484848488888888000000P`go.string."map.bucket[string]docker.PortMapping"p\go.weak.type.*map.bucket[string]"".PortMapping�"runtime.zerovalue��Jtype.map.bucket[string]"".PortMapping� go.string."keys"�type.[8]string�$go.string."values"�,type.[8]"".PortMapping�(go.string."overflow"�Ltype.*map.bucket[string]"".PortMapping�Zgo.string."map.hdr[string]docker.PortMapping"pd!map.hdr[string]docker.PortMapping Zgo.string."map.hdr[string]docker.PortMapping"�Dtype.map.hdr[string]"".PortMapping��0�Ag�  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000PZgo.string."map.hdr[string]docker.PortMapping"pVgo.weak.type.*map.hdr[string]"".PortMapping�"runtime.zerovalue��Dtype.map.hdr[string]"".PortMapping�&go.string."buckets"�Ltype.*map.bucket[string]"".PortMapping�,go.string."oldbuckets"�Ltype.*map.bucket[string]"".PortMapping�Rgo.string."map[string]docker.PortMapping"`\map[string]docker.PortMapping Rgo.string."map[string]docker.PortMapping"�type..gcprog."".NetworkSettings����YfY�Dgo.string."docker.NetworkSettings"PNdocker.NetworkSettings Dgo.string."docker.NetworkSettings"�*go.string."IPAddress"@4 IPAddress *go.string."IPAddress"��go.string."json:\"IPAddress,omitempty\" yaml:\"IPAddress,omitempty\""��5json:"IPAddress,omitempty" yaml:"IPAddress,omitempty" �go.string."json:\"IPAddress,omitempty\" yaml:\"IPAddress,omitempty\""�.go.string."IPPrefixLen"@8 IPPrefixLen .go.string."IPPrefixLen"��go.string."json:\"IPPrefixLen,omitempty\" yaml:\"IPPrefixLen,omitempty\""��9json:"IPPrefixLen,omitempty" yaml:"IPPrefixLen,omitempty" �go.string."json:\"IPPrefixLen,omitempty\" yaml:\"IPPrefixLen,omitempty\""�&go.string."Gateway"00Gateway &go.string."Gateway"��go.string."json:\"Gateway,omitempty\" yaml:\"Gateway,omitempty\""��1json:"Gateway,omitempty" yaml:"Gateway,omitempty" �go.string."json:\"Gateway,omitempty\" yaml:\"Gateway,omitempty\""�$go.string."Bridge"0.Bridge $go.string."Bridge"�~go.string."json:\"Bridge,omitempty\" yaml:\"Bridge,omitempty\""��/json:"Bridge,omitempty" yaml:"Bridge,omitempty" ~go.string."json:\"Bridge,omitempty\" yaml:\"Bridge,omitempty\""��go.string."json:\"PortMapping,omitempty\" yaml:\"PortMapping,omitempty\""��9json:"PortMapping,omitempty" yaml:"PortMapping,omitempty" �go.string."json:\"PortMapping,omitempty\" yaml:\"PortMapping,omitempty\""�"go.string."Ports"0,Ports "go.string."Ports"�zgo.string."json:\"Ports,omitempty\" yaml:\"Ports,omitempty\""�|-json:"Ports,omitempty" yaml:"Ports,omitempty" zgo.string."json:\"Ports,omitempty\" yaml:\"Ports,omitempty\""�*go.string."NetworkID"@4 NetworkID *go.string."NetworkID"��go.string."json:\"NetworkID,omitempty\" yaml:\"NetworkID,omitempty\""��5json:"NetworkID,omitempty" yaml:"NetworkID,omitempty" �go.string."json:\"NetworkID,omitempty\" yaml:\"NetworkID,omitempty\""�,go.string."EndpointID"@6 +EndpointID ,go.string."EndpointID"��go.string."json:\"EndpointID,omitempty\" yaml:\"EndpointID,omitempty\""��7json:"EndpointID,omitempty" yaml:"EndpointID,omitempty" �go.string."json:\"EndpointID,omitempty\" yaml:\"EndpointID,omitempty\""�,go.string."SandboxKey"@6 +SandboxKey ,go.string."SandboxKey"��go.string."json:\"SandboxKey,omitempty\" yaml:\"SandboxKey,omitempty\""��7json:"SandboxKey,omitempty" yaml:"SandboxKey,omitempty" �go.string."json:\"SandboxKey,omitempty\" yaml:\"SandboxKey,omitempty\""�:go.string."GlobalIPv6Address"PDGlobalIPv6Address :go.string."GlobalIPv6Address"��go.string."json:\"GlobalIPv6Address,omitempty\" yaml:\"GlobalIPv6Address,omitempty\""��Ejson:"GlobalIPv6Address,omitempty" yaml:"GlobalIPv6Address,omitempty" �go.string."json:\"GlobalIPv6Address,omitempty\" yaml:\"GlobalIPv6Address,omitempty\""�>go.string."GlobalIPv6PrefixLen"PHGlobalIPv6PrefixLen >go.string."GlobalIPv6PrefixLen"��go.string."json:\"GlobalIPv6PrefixLen,omitempty\" yaml:\"GlobalIPv6PrefixLen,omitempty\""��Ijson:"GlobalIPv6PrefixLen,omitempty" yaml:"GlobalIPv6PrefixLen,omitempty" �go.string."json:\"GlobalIPv6PrefixLen,omitempty\" yaml:\"GlobalIPv6PrefixLen,omitempty\""�.go.string."IPv6Gateway"@8 IPv6Gateway .go.string."IPv6Gateway"��go.string."json:\"IPv6Gateway,omitempty\" yaml:\"IPv6Gateway,omitempty\""��9json:"IPv6Gateway,omitempty" yaml:"IPv6Gateway,omitempty" �go.string."json:\"IPv6Gateway,omitempty\" yaml:\"IPv6Gateway,omitempty\""�@go.string."LinkLocalIPv6Address"PJLinkLocalIPv6Address @go.string."LinkLocalIPv6Address"��go.string."json:\"LinkLocalIPv6Address,omitempty\" yaml:\"LinkLocalIPv6Address,omitempty\""��Kjson:"LinkLocalIPv6Address,omitempty" yaml:"LinkLocalIPv6Address,omitempty" �go.string."json:\"LinkLocalIPv6Address,omitempty\" yaml:\"LinkLocalIPv6Address,omitempty\""�Dgo.string."LinkLocalIPv6PrefixLen"PNLinkLocalIPv6PrefixLen Dgo.string."LinkLocalIPv6PrefixLen"��go.string."json:\"LinkLocalIPv6PrefixLen,omitempty\" yaml:\"LinkLocalIPv6PrefixLen,omitempty\""��Ojson:"LinkLocalIPv6PrefixLen,omitempty" yaml:"LinkLocalIPv6PrefixLen,omitempty" �go.string."json:\"LinkLocalIPv6PrefixLen,omitempty\" yaml:\"LinkLocalIPv6PrefixLen,omitempty\""�@go.string."SecondaryIPAddresses"PJSecondaryIPAddresses @go.string."SecondaryIPAddresses"��go.string."json:\"SecondaryIPAddresses,omitempty\" yaml:\"SecondaryIPAddresses,omitempty\""��Kjson:"SecondaryIPAddresses,omitempty" yaml:"SecondaryIPAddresses,omitempty" �go.string."json:\"SecondaryIPAddresses,omitempty\" yaml:\"SecondaryIPAddresses,omitempty\""�Dgo.string."SecondaryIPv6Addresses"PNSecondaryIPv6Addresses Dgo.string."SecondaryIPv6Addresses"��go.string."json:\"SecondaryIPv6Addresses,omitempty\" yaml:\"SecondaryIPv6Addresses,omitempty\""��Ojson:"SecondaryIPv6Addresses,omitempty" yaml:"SecondaryIPv6Addresses,omitempty" �go.string."json:\"SecondaryIPv6Addresses,omitempty\" yaml:\"SecondaryIPv6Addresses,omitempty\""�6go.string."NetworkSettings"@@NetworkSettings 6go.string."NetworkSettings"�.type."".NetworkSettings� � ����Y(8HPXhx�������| � runtime.algarray06type..gc."".NetworkSettings@>type..gcprog."".NetworkSettingsPDgo.string."docker.NetworkSettings"p0type.*"".NetworkSettings�"runtime.zerovalue��.type."".NetworkSettings�*go.string."IPAddress"�type.string��go.string."json:\"IPAddress,omitempty\" yaml:\"IPAddress,omitempty\""�.go.string."IPPrefixLen"�type.int��go.string."json:\"IPPrefixLen,omitempty\" yaml:\"IPPrefixLen,omitempty\""�,go.string."MacAddress"�type.string��go.string."json:\"MacAddress,omitempty\" yaml:\"MacAddress,omitempty\""�&go.string."Gateway"�type.string��go.string."json:\"Gateway,omitempty\" yaml:\"Gateway,omitempty\""�$go.string."Bridge"�type.string�~go.string."json:\"Bridge,omitempty\" yaml:\"Bridge,omitempty\""�.go.string."PortMapping"�go.string."GlobalIPv6PrefixLen"�type.int��go.string."json:\"GlobalIPv6PrefixLen,omitempty\" yaml:\"GlobalIPv6PrefixLen,omitempty\""� .go.string."IPv6Gateway"� type.string� �go.string."json:\"IPv6Gateway,omitempty\" yaml:\"IPv6Gateway,omitempty\""� @go.string."LinkLocalIPv6Address"� type.string� +�go.string."json:\"LinkLocalIPv6Address,omitempty\" yaml:\"LinkLocalIPv6Address,omitempty\""� +Dgo.string."LinkLocalIPv6PrefixLen"� +type.int� +�go.string."json:\"LinkLocalIPv6PrefixLen,omitempty\" yaml:\"LinkLocalIPv6PrefixLen,omitempty\""� +@go.string."SecondaryIPAddresses"� type.[]string� �go.string."json:\"SecondaryIPAddresses,omitempty\" yaml:\"SecondaryIPAddresses,omitempty\""� Dgo.string."SecondaryIPv6Addresses"� type.[]string� �go.string."json:\"SecondaryIPv6Addresses,omitempty\" yaml:\"SecondaryIPv6Addresses,omitempty\""`� .type."".NetworkSettings� 6go.string."NetworkSettings"� "go.importpath."".� � .type."".NetworkSettings�Fgo.string."*docker.NetworkSettings"PP*docker.NetworkSettings Fgo.string."*docker.NetworkSettings"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�2type..hashfunc."".APIPort*type..hash."".APIPort�.type..eqfunc."".APIPort&type..eq."".APIPort�(type..alg."".APIPort 2type..hashfunc."".APIPort.type..eqfunc."".APIPort�6go.string."*docker.APIPort"@@*docker.APIPort 6go.string."*docker.APIPort"� type.*"".APIPort���� Q6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."*docker.APIPort"p2go.weak.type.**"".APIPort�"runtime.zerovalue�type."".APIPort�bruntime.gcbits.0x44484800000000000000000000000000 DHH�4go.string."docker.APIPort"@>docker.APIPort 4go.string."docker.APIPort"�.go.string."PrivatePort"@8 PrivatePort .go.string."PrivatePort"��go.string."json:\"PrivatePort,omitempty\" yaml:\"PrivatePort,omitempty\""��9json:"PrivatePort,omitempty" yaml:"PrivatePort,omitempty" �go.string."json:\"PrivatePort,omitempty\" yaml:\"PrivatePort,omitempty\""�,go.string."PublicPort"@6 +PublicPort ,go.string."PublicPort"��go.string."json:\"PublicPort,omitempty\" yaml:\"PublicPort,omitempty\""��7json:"PublicPort,omitempty" yaml:"PublicPort,omitempty" �go.string."json:\"PublicPort,omitempty\" yaml:\"PublicPort,omitempty\""�&go.string."APIPort"00APIPort &go.string."APIPort"�type."".APIPort��0�oW� , (type..alg."".APIPort0bruntime.gcbits.0x44484800000000000000000000000000P4go.string."docker.APIPort"p type.*"".APIPort�"runtime.zerovalue��type."".APIPort�.go.string."PrivatePort"�type.int64��go.string."json:\"PrivatePort,omitempty\" yaml:\"PrivatePort,omitempty\""�,go.string."PublicPort"�type.int64��go.string."json:\"PublicPort,omitempty\" yaml:\"PublicPort,omitempty\""� go.string."Type"�type.string�vgo.string."json:\"Type,omitempty\" yaml:\"Type,omitempty\""�go.string."IP"�type.string�ngo.string."json:\"IP,omitempty\" yaml:\"IP,omitempty\""`�type."".APIPort�&go.string."APIPort"�"go.importpath."".��type."".APIPort�8go.string."[]docker.APIPort"PB[]docker.APIPort 8go.string."[]docker.APIPort"�"type.[]"".APIPort����$ � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P8go.string."[]docker.APIPort"p4go.weak.type.*[]"".APIPort�"runtime.zerovalue�type."".APIPort�Rgo.typelink.[]docker.APIPort/[]"".APIPort"type.[]"".APIPort�tgo.string."func(*docker.NetworkSettings) []docker.APIPort"�~.func(*docker.NetworkSettings) []docker.APIPort tgo.string."func(*docker.NetworkSettings) []docker.APIPort"�Vtype.func(*"".NetworkSettings) []"".APIPort�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ptgo.string."func(*docker.NetworkSettings) []docker.APIPort"phgo.weak.type.*func(*"".NetworkSettings) []"".APIPort�"runtime.zerovalue��Vtype.func(*"".NetworkSettings) []"".APIPort��Vtype.func(*"".NetworkSettings) []"".APIPort�0type.*"".NetworkSettings�"type.[]"".APIPort�4go.string."PortMappingAPI"@>PortMappingAPI 4go.string."PortMappingAPI"�Fgo.string."func() []docker.APIPort"PPfunc() []docker.APIPort Fgo.string."func() []docker.APIPort"�0type.func() []"".APIPort���:�x3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PFgo.string."func() []docker.APIPort"pBgo.weak.type.*func() []"".APIPort�"runtime.zerovalue��0type.func() []"".APIPort��0type.func() []"".APIPort�"type.[]"".APIPort�0type.*"".NetworkSettings���VB6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PFgo.string."*docker.NetworkSettings"pBgo.weak.type.**"".NetworkSettings�"runtime.zerovalue�.type."".NetworkSettings`�0type.*"".NetworkSettings��0type.*"".NetworkSettings�4go.string."PortMappingAPI"�0type.func() []"".APIPort�Vtype.func(*"".NetworkSettings) []"".APIPort�H"".(*NetworkSettings).PortMappingAPI�H"".(*NetworkSettings).PortMappingAPI�$go.string."[]bool"0.[]bool $go.string."[]bool"�type.[]bool����� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P$go.string."[]bool"p(go.weak.type.*[]bool�"runtime.zerovalue�type.bool�2go.typelink.[]bool/[]booltype.[]bool�&go.string."[8]bool"00[8]bool &go.string."[8]bool"�type.[8]bool��s�5� � runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P&go.string."[8]bool"p*go.weak.type.*[8]bool�"runtime.zerovalue�type.bool�type.[]bool�6go.typelink.[8]bool/[8]booltype.[8]bool�Fgo.string."*map.bucket[string]bool"PP*map.bucket[string]bool Fgo.string."*map.bucket[string]bool"�8type.*map.bucket[string]bool���[�E6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PFgo.string."*map.bucket[string]bool"pJgo.weak.type.**map.bucket[string]bool�"runtime.zerovalue�6type.map.bucket[string]bool�,>type..gc.map.bucket[string]bool(�Ftype..gcprog.map.bucket[string]bool����%�Dgo.string."map.bucket[string]bool"PNmap.bucket[string]bool Dgo.string."map.bucket[string]bool"�6type.map.bucket[string]bool���2aB�Y�� � runtime.algarray0>type..gc.map.bucket[string]bool@Ftype..gcprog.map.bucket[string]boolPDgo.string."map.bucket[string]bool"pHgo.weak.type.*map.bucket[string]bool�"runtime.zerovalue��6type.map.bucket[string]bool� go.string."keys"�type.[8]string�$go.string."values"�type.[8]bool�(go.string."overflow"�8type.*map.bucket[string]bool�>go.string."map.hdr[string]bool"PHmap.hdr[string]bool >go.string."map.hdr[string]bool"�0type.map.hdr[string]bool��03�(  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000P>go.string."map.hdr[string]bool"pBgo.weak.type.*map.hdr[string]bool�"runtime.zerovalue��0type.map.hdr[string]bool�&go.string."buckets"�8type.*map.bucket[string]bool�,go.string."oldbuckets"�8type.*map.bucket[string]bool�6go.string."map[string]bool"@@map[string]bool 6go.string."map[string]bool"�(type.map[string]bool����5� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."map[string]bool"p:go.weak.type.*map[string]bool�"runtime.zerovalue�type.string�type.bool�6type.map.bucket[string]bool�0type.map.hdr[string]bool�Vgo.typelink.map[string]bool/map[string]bool(type.map[string]bool�,*type..gc."".Containerd�2type..gcprog."".Container 1V�eeeijfff���8go.string."docker.Container"PBdocker.Container 8go.string."docker.Container"�vgo.string."json:\"Path,omitempty\" yaml:\"Path,omitempty\""�x+json:"Path,omitempty" yaml:"Path,omitempty" vgo.string."json:\"Path,omitempty\" yaml:\"Path,omitempty\""� go.string."Args"0*Args go.string."Args"�vgo.string."json:\"Args,omitempty\" yaml:\"Args,omitempty\""�x+json:"Args,omitempty" yaml:"Args,omitempty" vgo.string."json:\"Args,omitempty\" yaml:\"Args,omitempty\""�zgo.string."json:\"State,omitempty\" yaml:\"State,omitempty\""�|-json:"State,omitempty" yaml:"State,omitempty" zgo.string."json:\"State,omitempty\" yaml:\"State,omitempty\""� go.string."Node"0*Node go.string."Node"�vgo.string."json:\"Node,omitempty\" yaml:\"Node,omitempty\""�x+json:"Node,omitempty" yaml:"Node,omitempty" vgo.string."json:\"Node,omitempty\" yaml:\"Node,omitempty\""��go.string."json:\"NetworkSettings,omitempty\" yaml:\"NetworkSettings,omitempty\""��Ajson:"NetworkSettings,omitempty" yaml:"NetworkSettings,omitempty" �go.string."json:\"NetworkSettings,omitempty\" yaml:\"NetworkSettings,omitempty\""�.go.string."SysInitPath"@8 SysInitPath .go.string."SysInitPath"��go.string."json:\"SysInitPath,omitempty\" yaml:\"SysInitPath,omitempty\""��9json:"SysInitPath,omitempty" yaml:"SysInitPath,omitempty" �go.string."json:\"SysInitPath,omitempty\" yaml:\"SysInitPath,omitempty\""�4go.string."ResolvConfPath"@>ResolvConfPath 4go.string."ResolvConfPath"��go.string."json:\"ResolvConfPath,omitempty\" yaml:\"ResolvConfPath,omitempty\""��?json:"ResolvConfPath,omitempty" yaml:"ResolvConfPath,omitempty" �go.string."json:\"ResolvConfPath,omitempty\" yaml:\"ResolvConfPath,omitempty\""�0go.string."HostnamePath"@: HostnamePath 0go.string."HostnamePath"��go.string."json:\"HostnamePath,omitempty\" yaml:\"HostnamePath,omitempty\""��;json:"HostnamePath,omitempty" yaml:"HostnamePath,omitempty" �go.string."json:\"HostnamePath,omitempty\" yaml:\"HostnamePath,omitempty\""�*go.string."HostsPath"@4 HostsPath *go.string."HostsPath"��go.string."json:\"HostsPath,omitempty\" yaml:\"HostsPath,omitempty\""��5json:"HostsPath,omitempty" yaml:"HostsPath,omitempty" �go.string."json:\"HostsPath,omitempty\" yaml:\"HostsPath,omitempty\""�&go.string."LogPath"00LogPath &go.string."LogPath"��go.string."json:\"LogPath,omitempty\" yaml:\"LogPath,omitempty\""��1json:"LogPath,omitempty" yaml:"LogPath,omitempty" �go.string."json:\"LogPath,omitempty\" yaml:\"LogPath,omitempty\""�$go.string."Driver"0.Driver $go.string."Driver"�~go.string."json:\"Driver,omitempty\" yaml:\"Driver,omitempty\""��/json:"Driver,omitempty" yaml:"Driver,omitempty" ~go.string."json:\"Driver,omitempty\" yaml:\"Driver,omitempty\""�*go.string."VolumesRW"@4 VolumesRW *go.string."VolumesRW"��go.string."json:\"VolumesRW,omitempty\" yaml:\"VolumesRW,omitempty\""��5json:"VolumesRW,omitempty" yaml:"VolumesRW,omitempty" �go.string."json:\"VolumesRW,omitempty\" yaml:\"VolumesRW,omitempty\""��go.string."json:\"HostConfig,omitempty\" yaml:\"HostConfig,omitempty\""��7json:"HostConfig,omitempty" yaml:"HostConfig,omitempty" �go.string."json:\"HostConfig,omitempty\" yaml:\"HostConfig,omitempty\""�&go.string."ExecIDs"00ExecIDs &go.string."ExecIDs"��go.string."json:\"ExecIDs,omitempty\" yaml:\"ExecIDs,omitempty\""��1json:"ExecIDs,omitempty" yaml:"ExecIDs,omitempty" �go.string."json:\"ExecIDs,omitempty\" yaml:\"ExecIDs,omitempty\""�0go.string."RestartCount"@: RestartCount 0go.string."RestartCount"��go.string."json:\"RestartCount,omitempty\" yaml:\"RestartCount,omitempty\""��;json:"RestartCount,omitempty" yaml:"RestartCount,omitempty" �go.string."json:\"RestartCount,omitempty\" yaml:\"RestartCount,omitempty\""�6go.string."AppArmorProfile"@@AppArmorProfile 6go.string."AppArmorProfile"��go.string."json:\"AppArmorProfile,omitempty\" yaml:\"AppArmorProfile,omitempty\""��Ajson:"AppArmorProfile,omitempty" yaml:"AppArmorProfile,omitempty" �go.string."json:\"AppArmorProfile,omitempty\" yaml:\"AppArmorProfile,omitempty\""�"type."".Container����5SdY(8PX������ 0@HPXpx� � runtime.algarray0*type..gc."".Container@2type..gcprog."".ContainerP8go.string."docker.Container"p$type.*"".Container�"runtime.zerovalue��"type."".Container�go.string."ID"�type.string�Fgo.string."json:\"Id\" yaml:\"Id\""�&go.string."Created"�type.time.Time��go.string."json:\"Created,omitempty\" yaml:\"Created,omitempty\""� go.string."Path"�type.string�vgo.string."json:\"Path,omitempty\" yaml:\"Path,omitempty\""� go.string."Args"�type.[]string�vgo.string."json:\"Args,omitempty\" yaml:\"Args,omitempty\""�$go.string."Config"�type.*"".Config�~go.string."json:\"Config,omitempty\" yaml:\"Config,omitempty\""�"go.string."State"�type."".State�zgo.string."json:\"State,omitempty\" yaml:\"State,omitempty\""�"go.string."Image"�type.string�zgo.string."json:\"Image,omitempty\" yaml:\"Image,omitempty\""� go.string."Node"�$type.*"".SwarmNode�vgo.string."json:\"Node,omitempty\" yaml:\"Node,omitempty\""�6go.string."NetworkSettings"�0type.*"".NetworkSettings��go.string."json:\"NetworkSettings,omitempty\" yaml:\"NetworkSettings,omitempty\""�.go.string."SysInitPath"�type.string��go.string."json:\"SysInitPath,omitempty\" yaml:\"SysInitPath,omitempty\""�4go.string."ResolvConfPath"�type.string��go.string."json:\"ResolvConfPath,omitempty\" yaml:\"ResolvConfPath,omitempty\""�0go.string."HostnamePath"�type.string��go.string."json:\"HostnamePath,omitempty\" yaml:\"HostnamePath,omitempty\""� *go.string."HostsPath"� type.string� �go.string."json:\"HostsPath,omitempty\" yaml:\"HostsPath,omitempty\""� &go.string."LogPath"� type.string� +�go.string."json:\"LogPath,omitempty\" yaml:\"LogPath,omitempty\""� + go.string."Name"� +type.string� +vgo.string."json:\"Name,omitempty\" yaml:\"Name,omitempty\""� +$go.string."Driver"� type.string� ~go.string."json:\"Driver,omitempty\" yaml:\"Driver,omitempty\""� &go.string."Volumes"� ,type.map[string]string� �go.string."json:\"Volumes,omitempty\" yaml:\"Volumes,omitempty\""� *go.string."VolumesRW"� (type.map[string]bool� �go.string."json:\"VolumesRW,omitempty\" yaml:\"VolumesRW,omitempty\""� ,go.string."HostConfig"� &type.*"".HostConfig� �go.string."json:\"HostConfig,omitempty\" yaml:\"HostConfig,omitempty\""� &go.string."ExecIDs"� type.[]string� �go.string."json:\"ExecIDs,omitempty\" yaml:\"ExecIDs,omitempty\""�0go.string."RestartCount"�type.int��go.string."json:\"RestartCount,omitempty\" yaml:\"RestartCount,omitempty\""�6go.string."AppArmorProfile"�type.string��go.string."json:\"AppArmorProfile,omitempty\" yaml:\"AppArmorProfile,omitempty\""`�"type."".Container�*go.string."Container"�"go.importpath."".��"type."".Container�:go.string."*docker.Container"PD*docker.Container :go.string."*docker.Container"�$type.*"".Container������6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."*docker.Container"p6go.weak.type.**"".Container�"runtime.zerovalue�"type."".Container��go.string."func(*docker.Client, docker.CreateContainerOptions) (*docker.Container, error)"��Nfunc(*docker.Client, docker.CreateContainerOptions) (*docker.Container, error) �go.string."func(*docker.Client, docker.CreateContainerOptions) (*docker.Container, error)"��type.func(*"".Client, "".CreateContainerOptions) (*"".Container, error)��k73 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, docker.CreateContainerOptions) (*docker.Container, error)"p�go.weak.type.*func(*"".Client, "".CreateContainerOptions) (*"".Container, error)�"runtime.zerovalue���type.func(*"".Client, "".CreateContainerOptions) (*"".Container, error)���type.func(*"".Client, "".CreateContainerOptions) (*"".Container, error)�type.*"".Client� � runtime.algarray0bruntime.gcbits.0x84444848000000000000000000000000PHgo.string."docker.CreateExecOptions"p4type.*"".CreateExecOptions�"runtime.zerovalue��2type."".CreateExecOptions�.go.string."AttachStdin"�type.bool��go.string."json:\"AttachStdin,omitempty\" yaml:\"AttachStdin,omitempty\""�0go.string."AttachStdout"�type.bool��go.string."json:\"AttachStdout,omitempty\" yaml:\"AttachStdout,omitempty\""�0go.string."AttachStderr"�type.bool��go.string."json:\"AttachStderr,omitempty\" yaml:\"AttachStderr,omitempty\""�go.string."Tty"�type.bool�rgo.string."json:\"Tty,omitempty\" yaml:\"Tty,omitempty\""�go.string."Cmd"�type.[]string�rgo.string."json:\"Cmd,omitempty\" yaml:\"Cmd,omitempty\""�*go.string."Container"�type.string��go.string."json:\"Container,omitempty\" yaml:\"Container,omitempty\""� go.string."User"�type.string�vgo.string."json:\"User,omitempty\" yaml:\"User,omitempty\""`�2type."".CreateExecOptions�:go.string."CreateExecOptions"�"go.importpath."".��2type."".CreateExecOptions�.go.string."docker.Exec"@8 docker.Exec .go.string."docker.Exec"�ngo.string."json:\"Id,omitempty\" yaml:\"Id,omitempty\""pp'json:"Id,omitempty" yaml:"Id,omitempty" ngo.string."json:\"Id,omitempty\" yaml:\"Id,omitempty\""� go.string."Exec"0*Exec go.string."Exec"�type."".Exec��C̀� � runtime.algarray0bruntime.gcbits.0x48000000000000000000000000000000P.go.string."docker.Exec"ptype.*"".Exec�"runtime.zerovalue��type."".Exec�go.string."ID"�type.string�ngo.string."json:\"Id,omitempty\" yaml:\"Id,omitempty\""`�type."".Exec� go.string."Exec"�"go.importpath."".��type."".Exec�0go.string."*docker.Exec"@: *docker.Exec 0go.string."*docker.Exec"�type.*"".Exec��Ά� 6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P0go.string."*docker.Exec"p,go.weak.type.**"".Exec�"runtime.zerovalue�type."".Exec��go.string."func(*docker.Client, docker.CreateExecOptions) (*docker.Exec, error)"��Dfunc(*docker.Client, docker.CreateExecOptions) (*docker.Exec, error) �go.string."func(*docker.Client, docker.CreateExecOptions) (*docker.Exec, error)"�ztype.func(*"".Client, "".CreateExecOptions) (*"".Exec, error)��T!�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, docker.CreateExecOptions) (*docker.Exec, error)"p�go.weak.type.*func(*"".Client, "".CreateExecOptions) (*"".Exec, error)�"runtime.zerovalue��ztype.func(*"".Client, "".CreateExecOptions) (*"".Exec, error)��ztype.func(*"".Client, "".CreateExecOptions) (*"".Exec, error)�type.*"".Client�2type."".CreateExecOptions�type.*"".Exec�type.error�bruntime.gcbits.0xcc000000000000000000000000000000 ��0go.string."interface {}"@: interface {} 0go.string."interface {}"�"type.interface {}���W� � runtime.algarray0bruntime.gcbits.0xcc000000000000000000000000000000P0go.string."interface {}"p4go.weak.type.*interface {}�"runtime.zerovalue��"type.interface {}�4go.string."[]interface {}"@>[]interface {} 4go.string."[]interface {}"�&type.[]interface {}��p��/ � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P4go.string."[]interface {}"p8go.weak.type.*[]interface {}�"runtime.zerovalue�"type.interface {}�Rgo.typelink.[]interface {}/[]interface {}&type.[]interface {}�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�json:"options" 8go.string."json:\"options\""�@go.string."CreateNetworkOptions"PJCreateNetworkOptions @go.string."CreateNetworkOptions"�8type."".CreateNetworkOptions��(bH>? & � runtime.algarray0bruntime.gcbits.0x48488884840000000000000000000000PNgo.string."docker.CreateNetworkOptions"p:type.*"".CreateNetworkOptions�"runtime.zerovalue��8type."".CreateNetworkOptions� go.string."Name"�type.string�2go.string."json:\"name\""�.go.string."NetworkType"�type.string�Bgo.string."json:\"network_type\""�&go.string."Options"�8type.map[string]interface {}�8go.string."json:\"options\""`�8type."".CreateNetworkOptions�@go.string."CreateNetworkOptions"�"go.importpath."".��8type."".CreateNetworkOptions�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�4type..hashfunc."".Endpoint,type..hash."".Endpoint�0type..eqfunc."".Endpoint(type..eq."".Endpoint�*type..alg."".Endpoint 4type..hashfunc."".Endpoint0type..eqfunc."".Endpoint�6go.string."docker.Endpoint"@@docker.Endpoint 6go.string."docker.Endpoint"�.go.string."json:\"id\""@4 json:"id" .go.string."json:\"id\""�&go.string."Network"00Network &go.string."Network"�8go.string."json:\"network\""@>json:"network" 8go.string."json:\"network\""�(go.string."Endpoint"@2Endpoint (go.string."Endpoint"� type."".Endpoint��0�� & *type..alg."".Endpoint0bruntime.gcbits.0x48484800000000000000000000000000P6go.string."docker.Endpoint"p"type.*"".Endpoint�"runtime.zerovalue�� type."".Endpoint� go.string."Name"�type.string�2go.string."json:\"name\""�go.string."ID"�type.string�.go.string."json:\"id\""�&go.string."Network"�type.string�8go.string."json:\"network\""`� type."".Endpoint�(go.string."Endpoint"�"go.importpath."".�� type."".Endpoint�8go.string."*docker.Endpoint"PB*docker.Endpoint 8go.string."*docker.Endpoint"�"type.*"".Endpoint��W��t6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P8go.string."*docker.Endpoint"p4go.weak.type.**"".Endpoint�"runtime.zerovalue� type."".Endpoint�docker.Network 4go.string."docker.Network"�2go.string."json:\"type\""@8 json:"type" 2go.string."json:\"type\""�*go.string."Endpoints"@4 Endpoints *go.string."Endpoints"�type.*"".ExportContainerOptions����/�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PTgo.string."*docker.ExportContainerOptions"pPgo.weak.type.**"".ExportContainerOptions�"runtime.zerovalue�type.*"".ExportContainerOptions�"runtime.zerovalue��type..alg."".ExportImageOptions Htype..hashfunc."".ExportImageOptionsDtype..eqfunc."".ExportImageOptions�Lgo.string."*docker.ExportImageOptions"`V*docker.ExportImageOptions Lgo.string."*docker.ExportImageOptions"�6type.*"".ExportImageOptions��޼�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PLgo.string."*docker.ExportImageOptions"pHgo.weak.type.**"".ExportImageOptions�"runtime.zerovalue�4type."".ExportImageOptions�Jgo.string."docker.ExportImageOptions"`Tdocker.ExportImageOptions Jgo.string."docker.ExportImageOptions"�type..alg."".ExportImageOptions0bruntime.gcbits.0x488c0000000000000000000000000000PJgo.string."docker.ExportImageOptions"p6type.*"".ExportImageOptions�"runtime.zerovalue��4type."".ExportImageOptions� go.string."Name"�type.string�0go.string."OutputStream"�type.io.Writer`�4type."".ExportImageOptions�go.string."ExportImagesOptions"PHExportImagesOptions >go.string."ExportImagesOptions"�6type."".ExportImagesOptions��(�2W� � runtime.algarray0bruntime.gcbits.0x48c488448c0000000000000000000000PLgo.string."docker.ExportImagesOptions"p8type.*"".ExportImagesOptions�"runtime.zerovalue��6type."".ExportImagesOptions�"go.string."Names"�type.[]string�0go.string."OutputStream"�type.io.Writer�(go.string."qs:\"-\""`�6type."".ExportImagesOptions�>go.string."ExportImagesOptions"�"go.importpath."".��6type."".ExportImagesOptions��go.string."func(*docker.Client, docker.ExportImagesOptions) error"��6func(*docker.Client, docker.ExportImagesOptions) error �go.string."func(*docker.Client, docker.ExportImagesOptions) error"�ftype.func(*"".Client, "".ExportImagesOptions) error����'�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, docker.ExportImagesOptions) error"pxgo.weak.type.*func(*"".Client, "".ExportImagesOptions) error�"runtime.zerovalue��ftype.func(*"".Client, "".ExportImagesOptions) error��ftype.func(*"".Client, "".ExportImagesOptions) error�type.*"".Client�6type."".ExportImagesOptions�type.error�@go.string."*docker.ImageHistory"PJ*docker.ImageHistory @go.string."*docker.ImageHistory"�*type.*"".ImageHistory��RY�?6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P@go.string."*docker.ImageHistory"pgo.string."docker.ImageHistory"PHdocker.ImageHistory >go.string."docker.ImageHistory"� go.string."Tags"0*Tags go.string."Tags"�vgo.string."json:\"Tags,omitempty\" yaml:\"Tags,omitempty\""�x+json:"Tags,omitempty" yaml:"Tags,omitempty" vgo.string."json:\"Tags,omitempty\" yaml:\"Tags,omitempty\""�*go.string."CreatedBy"@4 CreatedBy *go.string."CreatedBy"��go.string."json:\"CreatedBy,omitempty\" yaml:\"CreatedBy,omitempty\""��5json:"CreatedBy,omitempty" yaml:"CreatedBy,omitempty" �go.string."json:\"CreatedBy,omitempty\" yaml:\"CreatedBy,omitempty\""�0go.string."ImageHistory"@: ImageHistory 0go.string."ImageHistory"�(type."".ImageHistory��H[Ę(0@2 � runtime.algarray0bruntime.gcbits.0x48484448848444844400000000000000P>go.string."docker.ImageHistory"p*type.*"".ImageHistory�"runtime.zerovalue��(type."".ImageHistory�go.string."ID"�type.string�Fgo.string."json:\"Id\" yaml:\"Id\""� go.string."Tags"�type.[]string�vgo.string."json:\"Tags,omitempty\" yaml:\"Tags,omitempty\""�&go.string."Created"�type.int64��go.string."json:\"Created,omitempty\" yaml:\"Created,omitempty\""�*go.string."CreatedBy"�type.string��go.string."json:\"CreatedBy,omitempty\" yaml:\"CreatedBy,omitempty\""� go.string."Size"�type.int64�vgo.string."json:\"Size,omitempty\" yaml:\"Size,omitempty\""`�(type."".ImageHistory�0go.string."ImageHistory"�"go.importpath."".��(type."".ImageHistory�Bgo.string."[]docker.ImageHistory"PL[]docker.ImageHistory Bgo.string."[]docker.ImageHistory"�,type.[]"".ImageHistory��}�c\ � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000PBgo.string."[]docker.ImageHistory"p>go.weak.type.*[]"".ImageHistory�"runtime.zerovalue�(type."".ImageHistory�fgo.typelink.[]docker.ImageHistory/[]"".ImageHistory,type.[]"".ImageHistory��go.string."func(*docker.Client, string) ([]docker.ImageHistory, error)"��;func(*docker.Client, string) ([]docker.ImageHistory, error) �go.string."func(*docker.Client, string) ([]docker.ImageHistory, error)"�ptype.func(*"".Client, string) ([]"".ImageHistory, error)��+ɌM3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, string) ([]docker.ImageHistory, error)"p�go.weak.type.*func(*"".Client, string) ([]"".ImageHistory, error)�"runtime.zerovalue��ptype.func(*"".Client, string) ([]"".ImageHistory, error)��ptype.func(*"".Client, string) ([]"".ImageHistory, error)�type.*"".Client�type.string�,type.[]"".ImageHistory�type.error�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·af3107c17ee1ab6f9f33230b5c7e3062�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�Htype..hashfunc."".ImportImageOptions@type..hash."".ImportImageOptions�Dtype..eqfunc."".ImportImageOptionstype..alg."".ImportImageOptions Htype..hashfunc."".ImportImageOptionsDtype..eqfunc."".ImportImageOptions�Lgo.string."*docker.ImportImageOptions"`V*docker.ImportImageOptions Lgo.string."*docker.ImportImageOptions"�6type.*"".ImportImageOptions��5��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PLgo.string."*docker.ImportImageOptions"pHgo.weak.type.**"".ImportImageOptions�"runtime.zerovalue�4type."".ImportImageOptions�bruntime.gcbits.0x4848488c8c848484c4c8480000000000 HHH�������H�Jgo.string."docker.ImportImageOptions"`Tdocker.ImportImageOptions Jgo.string."docker.ImportImageOptions"�$go.string."Source"0.Source $go.string."Source"�4go.string."qs:\"fromSrc\""@: qs:"fromSrc" 4go.string."qs:\"fromSrc\""�,go.string."qs:\"tag\""@2qs:"tag" ,go.string."qs:\"tag\""�type..alg."".ImportImageOptions0bruntime.gcbits.0x4848488c8c848484c4c8480000000000PJgo.string."docker.ImportImageOptions"p6type.*"".ImportImageOptions�"runtime.zerovalue��4type."".ImportImageOptions�,go.string."Repository"�type.string�.go.string."qs:\"repo\""�$go.string."Source"�type.string�4go.string."qs:\"fromSrc\""�go.string."Tag"�type.string�,go.string."qs:\"tag\""�.go.string."InputStream"�type.io.Reader�(go.string."qs:\"-\""�0go.string."OutputStream"�type.io.Writer�(go.string."qs:\"-\""�2go.string."RawJSONStream"�type.bool�(go.string."qs:\"-\""`�4type."".ImportImageOptions�type.func(*"".Env, string) bool����J�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PTgo.string."func(*docker.Env, string) bool"pPgo.weak.type.*func(*"".Env, string) bool�"runtime.zerovalue��>type.func(*"".Env, string) bool��>type.func(*"".Env, string) bool�type.*"".Env�type.string�type.bool�Xgo.string."func(*docker.Env, string) string"pb func(*docker.Env, string) string Xgo.string."func(*docker.Env, string) string"�Btype.func(*"".Env, string) string��@i�(3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PXgo.string."func(*docker.Env, string) string"pTgo.weak.type.*func(*"".Env, string) string�"runtime.zerovalue��Btype.func(*"".Env, string) string��Btype.func(*"".Env, string) string�type.*"".Env�type.string�type.string�Rgo.string."func(*docker.Env, string) int"`\func(*docker.Env, string) int Rgo.string."func(*docker.Env, string) int"�type.func(*"".Env, string, int)���r��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PTgo.string."func(*docker.Env, string, int)"pPgo.weak.type.*func(*"".Env, string, int)�"runtime.zerovalue��>type.func(*"".Env, string, int)��>type.func(*"".Env, string, int)�type.*"".Env�type.string�type.int�Xgo.string."func(*docker.Env, string, int64)"pb func(*docker.Env, string, int64) Xgo.string."func(*docker.Env, string, int64)"�Btype.func(*"".Env, string, int64)��g���3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PXgo.string."func(*docker.Env, string, int64)"pTgo.weak.type.*func(*"".Env, string, int64)�"runtime.zerovalue��Btype.func(*"".Env, string, int64)��Btype.func(*"".Env, string, int64)�type.*"".Env�type.string�type.int64�jgo.string."func(*docker.Env, string, []string) error"�t)func(*docker.Env, string, []string) error jgo.string."func(*docker.Env, string, []string) error"�Ttype.func(*"".Env, string, []string) error����D�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pjgo.string."func(*docker.Env, string, []string) error"pfgo.weak.type.*func(*"".Env, string, []string) error�"runtime.zerovalue��Ttype.func(*"".Env, string, []string) error��Ttype.func(*"".Env, string, []string) error�type.*"".Env�type.string�type.[]string�type.error�$go.string."Decode"0.Decode $go.string."Decode"�Bgo.string."func(io.Reader) error"PLfunc(io.Reader) error Bgo.string."func(io.Reader) error"�4type.func(io.Reader) error���&�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."func(io.Reader) error"pFgo.weak.type.*func(io.Reader) error�"runtime.zerovalue��4type.func(io.Reader) error��4type.func(io.Reader) error�type.io.Reader�type.error�$go.string."Exists"0.Exists $go.string."Exists"�:go.string."func(string) bool"PDfunc(string) bool :go.string."func(string) bool"�,type.func(string) bool��*��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."func(string) bool"p>go.weak.type.*func(string) bool�"runtime.zerovalue��,type.func(string) bool��,type.func(string) bool�type.string�type.bool�go.string."Get"0(Get go.string."Get"�>go.string."func(string) string"PHfunc(string) string >go.string."func(string) string"�0type.func(string) string��M���3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."func(string) string"pBgo.weak.type.*func(string) string�"runtime.zerovalue��0type.func(string) string��0type.func(string) string�type.string�type.string�&go.string."GetBool"00GetBool &go.string."GetBool"�$go.string."GetInt"0.GetInt $go.string."GetInt"�8go.string."func(string) int"PBfunc(string) int 8go.string."func(string) int"�*type.func(string) int��bU3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P8go.string."func(string) int"ptype.func(string, interface {})�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PLgo.string."func(string, interface {})"pPgo.weak.type.*func(string, interface {})�"runtime.zerovalue��>type.func(string, interface {})��>type.func(string, interface {})�type.string�"type.interface {}�&go.string."SetBool"00SetBool &go.string."SetBool"�go.weak.type.*func(string, int)�"runtime.zerovalue��,type.func(string, int)��,type.func(string, int)�type.string�type.int�(go.string."SetInt64"@2SetInt64 (go.string."SetInt64"�>go.string."func(string, int64)"PHfunc(string, int64) >go.string."func(string, int64)"�0type.func(string, int64)���?3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."func(string, int64)"pBgo.weak.type.*func(string, int64)�"runtime.zerovalue��0type.func(string, int64)��0type.func(string, int64)�type.string�type.int64�&go.string."SetJSON"00SetJSON &go.string."SetJSON"�&go.string."SetList"00SetList &go.string."SetList"�Pgo.string."func(string, []string) error"`Zfunc(string, []string) error Pgo.string."func(string, []string) error"�Btype.func(string, []string) error���1�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."func(string, []string) error"pTgo.weak.type.*func(string, []string) error�"runtime.zerovalue��Btype.func(string, []string) error��Btype.func(string, []string) error�type.string�type.[]string�type.error�type.*"".Env� � �96� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P.go.string."*docker.Env"p*go.weak.type.**"".Env�"runtime.zerovalue�type."".Env`�type.*"".Env��type.*"".Env�$go.string."Decode"�4type.func(io.Reader) error�Ftype.func(*"".Env, io.Reader) error� "".(*Env).Decode� "".(*Env).Decode�$go.string."Exists"�,type.func(string) bool�>type.func(*"".Env, string) bool� "".(*Env).Exists� "".(*Env).Exists�go.string."Get"�0type.func(string) string�Btype.func(*"".Env, string) string�"".(*Env).Get�"".(*Env).Get�&go.string."GetBool"�,type.func(string) bool�>type.func(*"".Env, string) bool�""".(*Env).GetBool�""".(*Env).GetBool�$go.string."GetInt"�*type.func(string) int�type.func(string, interface {})� Ptype.func(*"".Env, string, interface {})� """.(*Env).SetAuto� +""".(*Env).SetAuto� +&go.string."SetBool"� +.type.func(string, bool)� +@type.func(*"".Env, string, bool)� +""".(*Env).SetBool� +""".(*Env).SetBool� +$go.string."SetInt"� ,type.func(string, int)� >type.func(*"".Env, string, int)�  "".(*Env).SetInt�  "".(*Env).SetInt� (go.string."SetInt64"� 0type.func(string, int64)� Btype.func(*"".Env, string, int64)� $"".(*Env).SetInt64� $"".(*Env).SetInt64� &go.string."SetJSON"� Jtype.func(string, interface {}) error� \type.func(*"".Env, string, interface {}) error� """.(*Env).SetJSON� """.(*Env).SetJSON� &go.string."SetList"� Btype.func(string, []string) error� Ttype.func(*"".Env, string, []string) error� """.(*Env).SetList� """.(*Env).SetList�jgo.string."func(*docker.Client) (*docker.Env, error)"�t)func(*docker.Client) (*docker.Env, error) jgo.string."func(*docker.Client) (*docker.Env, error)"�Ltype.func(*"".Client) (*"".Env, error)��0�l�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pjgo.string."func(*docker.Client) (*docker.Env, error)"p^go.weak.type.*func(*"".Client) (*"".Env, error)�"runtime.zerovalue��Ltype.func(*"".Client) (*"".Env, error)��Ltype.func(*"".Client) (*"".Env, error)�type.*"".Client�type.*"".Env�type.error��go.string."func(*docker.Client, string) (*docker.Container, error)"��7func(*docker.Client, string) (*docker.Container, error) �go.string."func(*docker.Client, string) (*docker.Container, error)"�htype.func(*"".Client, string) (*"".Container, error)��TB3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, string) (*docker.Container, error)"pzgo.weak.type.*func(*"".Client, string) (*"".Container, error)�"runtime.zerovalue��htype.func(*"".Client, string) (*"".Container, error)��htype.func(*"".Client, string) (*"".Container, error)�type.*"".Client�type.string�$type.*"".Container�type.error�Jgo.string."*docker.ExecProcessConfig"`T*docker.ExecProcessConfig Jgo.string."*docker.ExecProcessConfig"�4type.*"".ExecProcessConfig��Y�916 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PJgo.string."*docker.ExecProcessConfig"pFgo.weak.type.**"".ExecProcessConfig�"runtime.zerovalue�2type."".ExecProcessConfig�bruntime.gcbits.0x84444848444884844400000000000000 �DHHDH��D�Hgo.string."docker.ExecProcessConfig"`Rdocker.ExecProcessConfig Hgo.string."docker.ExecProcessConfig"��go.string."json:\"privileged,omitempty\" yaml:\"privileged,omitempty\""��7json:"privileged,omitempty" yaml:"privileged,omitempty" �go.string."json:\"privileged,omitempty\" yaml:\"privileged,omitempty\""�vgo.string."json:\"user,omitempty\" yaml:\"user,omitempty\""�x+json:"user,omitempty" yaml:"user,omitempty" vgo.string."json:\"user,omitempty\" yaml:\"user,omitempty\""�rgo.string."json:\"tty,omitempty\" yaml:\"tty,omitempty\""�t)json:"tty,omitempty" yaml:"tty,omitempty" rgo.string."json:\"tty,omitempty\" yaml:\"tty,omitempty\""�,go.string."EntryPoint"@6 +EntryPoint ,go.string."EntryPoint"��go.string."json:\"entrypoint,omitempty\" yaml:\"entrypoint,omitempty\""��7json:"entrypoint,omitempty" yaml:"entrypoint,omitempty" �go.string."json:\"entrypoint,omitempty\" yaml:\"entrypoint,omitempty\""�*go.string."Arguments"@4 Arguments *go.string."Arguments"��go.string."json:\"arguments,omitempty\" yaml:\"arguments,omitempty\""��5json:"arguments,omitempty" yaml:"arguments,omitempty" �go.string."json:\"arguments,omitempty\" yaml:\"arguments,omitempty\""�:go.string."ExecProcessConfig"PDExecProcessConfig :go.string."ExecProcessConfig"�2type."".ExecProcessConfig��Hx6� 02 � runtime.algarray0bruntime.gcbits.0x84444848444884844400000000000000PHgo.string."docker.ExecProcessConfig"p4type.*"".ExecProcessConfig�"runtime.zerovalue��2type."".ExecProcessConfig�,go.string."Privileged"�type.bool��go.string."json:\"privileged,omitempty\" yaml:\"privileged,omitempty\""� go.string."User"�type.string�vgo.string."json:\"user,omitempty\" yaml:\"user,omitempty\""�go.string."Tty"�type.bool�rgo.string."json:\"tty,omitempty\" yaml:\"tty,omitempty\""�,go.string."EntryPoint"�type.string��go.string."json:\"entrypoint,omitempty\" yaml:\"entrypoint,omitempty\""�*go.string."Arguments"�type.[]string��go.string."json:\"arguments,omitempty\" yaml:\"arguments,omitempty\""`�2type."".ExecProcessConfig�:go.string."ExecProcessConfig"�"go.importpath."".��2type."".ExecProcessConfig�,.type..gc."".ExecInspect��6type..gcprog."".ExecInspect&&?Ve�e�YVV��fff�Z�go.string."*docker.ExecInspect"PH*docker.ExecInspect >go.string."*docker.ExecInspect"�(type.*"".ExecInspect��W�~6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."*docker.ExecInspect"p:go.weak.type.**"".ExecInspect�"runtime.zerovalue�&type."".ExecInspect��go.string."func(*docker.Client, string) (*docker.ExecInspect, error)"��9func(*docker.Client, string) (*docker.ExecInspect, error) �go.string."func(*docker.Client, string) (*docker.ExecInspect, error)"�ltype.func(*"".Client, string) (*"".ExecInspect, error)���>m�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, string) (*docker.ExecInspect, error)"p~go.weak.type.*func(*"".Client, string) (*"".ExecInspect, error)�"runtime.zerovalue��ltype.func(*"".Client, string) (*"".ExecInspect, error)��ltype.func(*"".Client, string) (*"".ExecInspect, error)�type.*"".Client�type.string�(type.*"".ExecInspect�type.error�~go.string."func(*docker.Client, string) (*docker.Image, error)"��3func(*docker.Client, string) (*docker.Image, error) ~go.string."func(*docker.Client, string) (*docker.Image, error)"�`type.func(*"".Client, string) (*"".Image, error)��a7:�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P~go.string."func(*docker.Client, string) (*docker.Image, error)"prgo.weak.type.*func(*"".Client, string) (*"".Image, error)�"runtime.zerovalue��`type.func(*"".Client, string) (*"".Image, error)��`type.func(*"".Client, string) (*"".Image, error)�type.*"".Client�type.string�type.*"".Image�type.error�4go.string."*docker.Signal"@>*docker.Signal 4go.string."*docker.Signal"�type.*"".Signal���p�^6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."*docker.Signal"p0go.weak.type.**"".Signal�"runtime.zerovalue�type."".Signal�2go.string."docker.Signal"@< docker.Signal 2go.string."docker.Signal"�$go.string."Signal"0.Signal $go.string."Signal"�type."".Signal�� ��� � runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P2go.string."docker.Signal"ptype.*"".Signal�"runtime.zerovalue`�type."".Signal�$go.string."Signal"�"go.importpath."".��type."".Signal�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�Ltype..hashfunc."".KillContainerOptionsDtype..hash."".KillContainerOptions�Htype..eqfunc."".KillContainerOptions@type..eq."".KillContainerOptions�Btype..alg."".KillContainerOptions Ltype..hashfunc."".KillContainerOptionsHtype..eqfunc."".KillContainerOptions�Pgo.string."*docker.KillContainerOptions"`Z*docker.KillContainerOptions Pgo.string."*docker.KillContainerOptions"�:type.*"".KillContainerOptions����6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."*docker.KillContainerOptions"pLgo.weak.type.**"".KillContainerOptions�"runtime.zerovalue�8type."".KillContainerOptions�Ngo.string."docker.KillContainerOptions"`Xdocker.KillContainerOptions Ngo.string."docker.KillContainerOptions"�@go.string."KillContainerOptions"PJKillContainerOptions @go.string."KillContainerOptions"�8type."".KillContainerOptions�����6 Btype..alg."".KillContainerOptions0bruntime.gcbits.0x48844400000000000000000000000000PNgo.string."docker.KillContainerOptions"p:type.*"".KillContainerOptions�"runtime.zerovalue��8type."".KillContainerOptions�go.string."ID"�type.string�(go.string."qs:\"-\""�$go.string."Signal"�type."".Signal`�8type."".KillContainerOptions�@go.string."KillContainerOptions"�"go.importpath."".��8type."".KillContainerOptions��go.string."func(*docker.Client, docker.KillContainerOptions) error"��7func(*docker.Client, docker.KillContainerOptions) error �go.string."func(*docker.Client, docker.KillContainerOptions) error"�htype.func(*"".Client, "".KillContainerOptions) error��Q��%3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, docker.KillContainerOptions) error"pzgo.weak.type.*func(*"".Client, "".KillContainerOptions) error�"runtime.zerovalue��htype.func(*"".Client, "".KillContainerOptions) error��htype.func(*"".Client, "".KillContainerOptions) error�type.*"".Client�8type."".KillContainerOptions�type.error�,go.string."[][]string"@6 +[][]string ,go.string."[][]string"�type.[][]string���:� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P,go.string."[][]string"p0go.weak.type.*[][]string�"runtime.zerovalue�type.[]string�Bgo.typelink.[][]string/[][]stringtype.[][]string�.go.string."[8][]string"@8 [8][]string .go.string."[8][]string"� type.[8][]string����e�r � runtime.algarray0bruntime.gcbits.0x48844448844448844448844400000000P.go.string."[8][]string"p2go.weak.type.*[8][]string�"runtime.zerovalue�type.[]string�type.[][]string�Fgo.typelink.[8][]string/[8][]string type.[8][]string�Ngo.string."*map.bucket[string][]string"`X*map.bucket[string][]string Ngo.string."*map.bucket[string][]string"�@type.*map.bucket[string][]string���Y�R6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PNgo.string."*map.bucket[string][]string"pRgo.weak.type.**map.bucket[string][]string�"runtime.zerovalue�>type.map.bucket[string][]string�,Ftype..gc.map.bucket[string][]string,�Ntype..gcprog.map.bucket[string][]string*����Y�eY�e �Lgo.string."map.bucket[string][]string"`Vmap.bucket[string][]string Lgo.string."map.bucket[string][]string"�>type.map.bucket[string][]string��P�TJ�Y�H � runtime.algarray0Ftype..gc.map.bucket[string][]string@Ntype..gcprog.map.bucket[string][]stringPLgo.string."map.bucket[string][]string"pPgo.weak.type.*map.bucket[string][]string�"runtime.zerovalue��>type.map.bucket[string][]string� go.string."keys"�type.[8]string�$go.string."values"� type.[8][]string�(go.string."overflow"�@type.*map.bucket[string][]string�Fgo.string."map.hdr[string][]string"PPmap.hdr[string][]string Fgo.string."map.hdr[string][]string"�8type.map.hdr[string][]string��0����  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000PFgo.string."map.hdr[string][]string"pJgo.weak.type.*map.hdr[string][]string�"runtime.zerovalue��8type.map.hdr[string][]string�&go.string."buckets"�@type.*map.bucket[string][]string�,go.string."oldbuckets"�@type.*map.bucket[string][]string�>go.string."map[string][]string"PHmap[string][]string >go.string."map[string][]string"�0type.map[string][]string��'�>@5P � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."map[string][]string"pBgo.weak.type.*map[string][]string�"runtime.zerovalue�type.string�type.[]string�>type.map.bucket[string][]string�8type.map.hdr[string][]string�fgo.typelink.map[string][]string/map[string][]string0type.map[string][]string�Rgo.string."*docker.ListContainersOptions"`\*docker.ListContainersOptions Rgo.string."*docker.ListContainersOptions"�go.weak.type.**"".APIContainers�"runtime.zerovalue�*type."".APIContainers�,2type..gc."".APIContainers$�:type..gcprog."".APIContainersf�Ye�@go.string."docker.APIContainers"PJdocker.APIContainers @go.string."docker.APIContainers"�&go.string."Command"00Command &go.string."Command"��go.string."json:\"Command,omitempty\" yaml:\"Command,omitempty\""��1json:"Command,omitempty" yaml:"Command,omitempty" �go.string."json:\"Command,omitempty\" yaml:\"Command,omitempty\""�$go.string."SizeRw"0.SizeRw $go.string."SizeRw"�~go.string."json:\"SizeRw,omitempty\" yaml:\"SizeRw,omitempty\""��/json:"SizeRw,omitempty" yaml:"SizeRw,omitempty" ~go.string."json:\"SizeRw,omitempty\" yaml:\"SizeRw,omitempty\""�,go.string."SizeRootFs"@6 +SizeRootFs ,go.string."SizeRootFs"��go.string."json:\"SizeRootFs,omitempty\" yaml:\"SizeRootFs,omitempty\""��7json:"SizeRootFs,omitempty" yaml:"SizeRootFs,omitempty" �go.string."json:\"SizeRootFs,omitempty\" yaml:\"SizeRootFs,omitempty\""�zgo.string."json:\"Names,omitempty\" yaml:\"Names,omitempty\""�|-json:"Names,omitempty" yaml:"Names,omitempty" zgo.string."json:\"Names,omitempty\" yaml:\"Names,omitempty\""�2go.string."APIContainers"@< APIContainers 2go.string."APIContainers"�*type."".APIContainers������ Y  08H`hpL � runtime.algarray02type..gc."".APIContainers@:type..gcprog."".APIContainersP@go.string."docker.APIContainers"p,type.*"".APIContainers�"runtime.zerovalue��*type."".APIContainers�go.string."ID"�type.string�Fgo.string."json:\"Id\" yaml:\"Id\""�"go.string."Image"�type.string�zgo.string."json:\"Image,omitempty\" yaml:\"Image,omitempty\""�&go.string."Command"�type.string��go.string."json:\"Command,omitempty\" yaml:\"Command,omitempty\""�&go.string."Created"�type.int64��go.string."json:\"Created,omitempty\" yaml:\"Created,omitempty\""�$go.string."Status"�type.string�~go.string."json:\"Status,omitempty\" yaml:\"Status,omitempty\""�"go.string."Ports"�"type.[]"".APIPort�zgo.string."json:\"Ports,omitempty\" yaml:\"Ports,omitempty\""�$go.string."SizeRw"�type.int64�~go.string."json:\"SizeRw,omitempty\" yaml:\"SizeRw,omitempty\""�,go.string."SizeRootFs"�type.int64��go.string."json:\"SizeRootFs,omitempty\" yaml:\"SizeRootFs,omitempty\""�"go.string."Names"�type.[]string�zgo.string."json:\"Names,omitempty\" yaml:\"Names,omitempty\""`�*type."".APIContainers�2go.string."APIContainers"�"go.importpath."".��*type."".APIContainers�Dgo.string."[]docker.APIContainers"PN[]docker.APIContainers Dgo.string."[]docker.APIContainers"�.type.[]"".APIContainers��� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000PDgo.string."[]docker.APIContainers"p@go.weak.type.*[]"".APIContainers�"runtime.zerovalue�*type."".APIContainers�jgo.typelink.[]docker.APIContainers/[]"".APIContainers.type.[]"".APIContainers��go.string."func(*docker.Client, docker.ListContainersOptions) ([]docker.APIContainers, error)"��Rfunc(*docker.Client, docker.ListContainersOptions) ([]docker.APIContainers, error) �go.string."func(*docker.Client, docker.ListContainersOptions) ([]docker.APIContainers, error)"��type.func(*"".Client, "".ListContainersOptions) ([]"".APIContainers, error)���s��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, docker.ListContainersOptions) ([]docker.APIContainers, error)"p�go.weak.type.*func(*"".Client, "".ListContainersOptions) ([]"".APIContainers, error)�"runtime.zerovalue���type.func(*"".Client, "".ListContainersOptions) ([]"".APIContainers, error)���type.func(*"".Client, "".ListContainersOptions) ([]"".APIContainers, error)�type.*"".Client�:type."".ListContainersOptions�.type.[]"".APIContainers�type.error�Jgo.string."*docker.ListImagesOptions"`T*docker.ListImagesOptions Jgo.string."*docker.ListImagesOptions"�4type.*"".ListImagesOptions���<��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PJgo.string."*docker.ListImagesOptions"pFgo.weak.type.**"".ListImagesOptions�"runtime.zerovalue�2type."".ListImagesOptions�bruntime.gcbits.0x84444800000000000000000000000000 �DH�Hgo.string."docker.ListImagesOptions"`Rdocker.ListImagesOptions Hgo.string."docker.ListImagesOptions"�&go.string."Digests"00Digests &go.string."Digests"�:go.string."ListImagesOptions"PDListImagesOptions :go.string."ListImagesOptions"�2type."".ListImagesOptions���Q Z � runtime.algarray0bruntime.gcbits.0x84444800000000000000000000000000PHgo.string."docker.ListImagesOptions"p4type.*"".ListImagesOptions�"runtime.zerovalue��2type."".ListImagesOptions�go.string."All"�type.bool�&go.string."Filters"�0type.map[string][]string�&go.string."Digests"�type.bool`�2type."".ListImagesOptions�:go.string."ListImagesOptions"�"go.importpath."".��2type."".ListImagesOptions�:go.string."*docker.APIImages"PD*docker.APIImages :go.string."*docker.APIImages"�$type.*"".APIImages���5{�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."*docker.APIImages"p6go.weak.type.**"".APIImages�"runtime.zerovalue�"type."".APIImages�bruntime.gcbits.0x48484444484884000000000000000000 HHDDHH��8go.string."docker.APIImages"PBdocker.APIImages 8go.string."docker.APIImages"�(go.string."RepoTags"@2RepoTags (go.string."RepoTags"��go.string."json:\"RepoTags,omitempty\" yaml:\"RepoTags,omitempty\""��3json:"RepoTags,omitempty" yaml:"RepoTags,omitempty" �go.string."json:\"RepoTags,omitempty\" yaml:\"RepoTags,omitempty\""�(go.string."ParentID"@2ParentID (go.string."ParentID"��go.string."json:\"ParentId,omitempty\" yaml:\"ParentId,omitempty\""��3json:"ParentId,omitempty" yaml:"ParentId,omitempty" �go.string."json:\"ParentId,omitempty\" yaml:\"ParentId,omitempty\""�.go.string."RepoDigests"@8 RepoDigests .go.string."RepoDigests"��go.string."json:\"RepoDigests,omitempty\" yaml:\"RepoDigests,omitempty\""��9json:"RepoDigests,omitempty" yaml:"RepoDigests,omitempty" �go.string."json:\"RepoDigests,omitempty\" yaml:\"RepoDigests,omitempty\""�*go.string."APIImages"@4 APIImages *go.string."APIImages"�"type."".APIImages��p#3[(08@PhD � runtime.algarray0bruntime.gcbits.0x48484444484884000000000000000000P8go.string."docker.APIImages"p$type.*"".APIImages�"runtime.zerovalue��"type."".APIImages�go.string."ID"�type.string�Fgo.string."json:\"Id\" yaml:\"Id\""�(go.string."RepoTags"�type.[]string��go.string."json:\"RepoTags,omitempty\" yaml:\"RepoTags,omitempty\""�&go.string."Created"�type.int64��go.string."json:\"Created,omitempty\" yaml:\"Created,omitempty\""� go.string."Size"�type.int64�vgo.string."json:\"Size,omitempty\" yaml:\"Size,omitempty\""�.go.string."VirtualSize"�type.int64��go.string."json:\"VirtualSize,omitempty\" yaml:\"VirtualSize,omitempty\""�(go.string."ParentID"�type.string��go.string."json:\"ParentId,omitempty\" yaml:\"ParentId,omitempty\""�.go.string."RepoDigests"�type.[]string��go.string."json:\"RepoDigests,omitempty\" yaml:\"RepoDigests,omitempty\""�$go.string."Labels"�,type.map[string]string�~go.string."json:\"Labels,omitempty\" yaml:\"Labels,omitempty\""`�"type."".APIImages�*go.string."APIImages"�"go.importpath."".��"type."".APIImages�go.string."*docker.LogsOptions"PH*docker.LogsOptions >go.string."*docker.LogsOptions"�(type.*"".LogsOptions�����I6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."*docker.LogsOptions"p:go.weak.type.**"".LogsOptions�"runtime.zerovalue�&type."".LogsOptions�bruntime.gcbits.0x488c8c44844400000000000000000000 H��D�D�qs:"fromImage" 8go.string."qs:\"fromImage\""�(go.string."Registry"@2Registry (go.string."Registry"�8go.string."PullImageOptions"PBPullImageOptions 8go.string."PullImageOptions"�0type."".PullImageOptions��Hh�6� 0@. :type..alg."".PullImageOptions0bruntime.gcbits.0x4848488c848484c44800000000000000PFgo.string."docker.PullImageOptions"p2type.*"".PullImageOptions�"runtime.zerovalue��0type."".PullImageOptions�,go.string."Repository"�type.string�8go.string."qs:\"fromImage\""�(go.string."Registry"�type.string�go.string."Tag"�type.string�0go.string."OutputStream"�type.io.Writer�(go.string."qs:\"-\""�2go.string."RawJSONStream"�type.bool�(go.string."qs:\"-\""`�0type."".PullImageOptions�8go.string."PullImageOptions"�"go.importpath."".��0type."".PullImageOptions��go.string."func(*docker.Client, docker.PullImageOptions, docker.AuthConfiguration) error"��Mfunc(*docker.Client, docker.PullImageOptions, docker.AuthConfiguration) error �go.string."func(*docker.Client, docker.PullImageOptions, docker.AuthConfiguration) error"��type.func(*"".Client, "".PullImageOptions, "".AuthConfiguration) error��M�ϧ3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, docker.PullImageOptions, docker.AuthConfiguration) error"p�go.weak.type.*func(*"".Client, "".PullImageOptions, "".AuthConfiguration) error�"runtime.zerovalue���type.func(*"".Client, "".PullImageOptions, "".AuthConfiguration) error���type.func(*"".Client, "".PullImageOptions, "".AuthConfiguration) error�type.*"".Client�0type."".PullImageOptions�2type."".AuthConfiguration�type.error�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·e13351f28add7c60853cb3aac0a0e34e�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�Dtype..hashfunc."".PushImageOptionstype.*"".RemoveContainerOptions��M�Ss6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PTgo.string."*docker.RemoveContainerOptions"pPgo.weak.type.**"".RemoveContainerOptions�"runtime.zerovalue�type.*"".RemoveContainerOptions�"runtime.zerovalue��type.*"".RenameContainerOptions��Ff�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PTgo.string."*docker.RenameContainerOptions"pPgo.weak.type.**"".RenameContainerOptions�"runtime.zerovalue�type.*"".RenameContainerOptions�"runtime.zerovalue��APIImageSearch 4go.string."APIImageSearch"�,type."".APIImageSearch��0��Ti(2 6type..alg."".APIImageSearch0bruntime.gcbits.0x48844400000000000000000000000000PBgo.string."docker.APIImageSearch"p.type.*"".APIImageSearch�"runtime.zerovalue��,type."".APIImageSearch�.go.string."Description"�type.string��go.string."json:\"description,omitempty\" yaml:\"description,omitempty\""�,go.string."IsOfficial"�type.bool��go.string."json:\"is_official,omitempty\" yaml:\"is_official,omitempty\""�.go.string."IsAutomated"�type.bool��go.string."json:\"is_automated,omitempty\" yaml:\"is_automated,omitempty\""� go.string."Name"�type.string�vgo.string."json:\"name,omitempty\" yaml:\"name,omitempty\""�*go.string."StarCount"�type.int��go.string."json:\"star_count,omitempty\" yaml:\"star_count,omitempty\""`�,type."".APIImageSearch�4go.string."APIImageSearch"�"go.importpath."".��,type."".APIImageSearch�Fgo.string."[]docker.APIImageSearch"PP[]docker.APIImageSearch Fgo.string."[]docker.APIImageSearch"�0type.[]"".APIImageSearch��S��� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000PFgo.string."[]docker.APIImageSearch"pBgo.weak.type.*[]"".APIImageSearch�"runtime.zerovalue�,type."".APIImageSearch�ngo.typelink.[]docker.APIImageSearch/[]"".APIImageSearch0type.[]"".APIImageSearch��go.string."func(*docker.Client, string) ([]docker.APIImageSearch, error)"��=func(*docker.Client, string) ([]docker.APIImageSearch, error) �go.string."func(*docker.Client, string) ([]docker.APIImageSearch, error)"�ttype.func(*"".Client, string) ([]"".APIImageSearch, error)��+��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, string) ([]docker.APIImageSearch, error)"p�go.weak.type.*func(*"".Client, string) ([]"".APIImageSearch, error)�"runtime.zerovalue��ttype.func(*"".Client, string) ([]"".APIImageSearch, error)��ttype.func(*"".Client, string) ([]"".APIImageSearch, error)�type.*"".Client�type.string�0type.[]"".APIImageSearch�type.error��go.string."func(*docker.Client, string, *docker.HostConfig) error"��6func(*docker.Client, string, *docker.HostConfig) error �go.string."func(*docker.Client, string, *docker.HostConfig) error"�ftype.func(*"".Client, string, *"".HostConfig) error��g�ڿ3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, string, *docker.HostConfig) error"pxgo.weak.type.*func(*"".Client, string, *"".HostConfig) error�"runtime.zerovalue��ftype.func(*"".Client, string, *"".HostConfig) error��ftype.func(*"".Client, string, *"".HostConfig) error�type.*"".Client�type.string�&type.*"".HostConfig�type.error�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·e13351f28add7c60853cb3aac0a0e34e�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�Dtype..hashfunc."".StartExecOptions :type..alg."".StartExecOptions0bruntime.gcbits.0xc4c8c848488c8c8c8400000000000000PFgo.string."docker.StartExecOptions"p2type.*"".StartExecOptions�"runtime.zerovalue��0type."".StartExecOptions�$go.string."Detach"�type.bool�~go.string."json:\"Detach,omitempty\" yaml:\"Detach,omitempty\""�go.string."Tty"�type.bool�rgo.string."json:\"Tty,omitempty\" yaml:\"Tty,omitempty\""�.go.string."InputStream"�type.io.Reader�(go.string."qs:\"-\""�0go.string."OutputStream"�type.io.Writer�(go.string."qs:\"-\""�.go.string."ErrorStream"�type.io.Writer�(go.string."qs:\"-\""�.go.string."RawTerminal"�type.bool�(go.string."qs:\"-\""�&go.string."Success"�&type.chan struct {}�,go.string."json:\"-\""`�0type."".StartExecOptions�8go.string."StartExecOptions"�"go.importpath."".��0type."".StartExecOptions��go.string."func(*docker.Client, string, docker.StartExecOptions) error"��;func(*docker.Client, string, docker.StartExecOptions) error �go.string."func(*docker.Client, string, docker.StartExecOptions) error"�ptype.func(*"".Client, string, "".StartExecOptions) error���$y3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, string, docker.StartExecOptions) error"p�go.weak.type.*func(*"".Client, string, "".StartExecOptions) error�"runtime.zerovalue��ptype.func(*"".Client, string, "".StartExecOptions) error��ptype.func(*"".Client, string, "".StartExecOptions) error�type.*"".Client�type.string�0type."".StartExecOptions�type.error�""..gostring.1� +� +wstruct { RxDropped uint64 "json:\"rx_dropped,omitempty\" yaml:\"rx_dropped,omitempty\""; RxBytes uint64 "json:\"rx_bytes,omitempty\" yaml:\"rx_bytes,omitempty\""; RxErrors uint64 "json:\"rx_errors,omitempty\" yaml:\"rx_errors,omitempty\""; TxPackets uint64 "json:\"tx_packets,omitempty\" yaml:\"tx_packets,omitempty\""; TxDropped uint64 "json:\"tx_dropped,omitempty\" yaml:\"tx_dropped,omitempty\""; RxPackets uint64 "json:\"rx_packets,omitempty\" yaml:\"rx_packets,omitempty\""; TxErrors uint64 "json:\"tx_errors,omitempty\" yaml:\"tx_errors,omitempty\""; TxBytes uint64 "json:\"tx_bytes,omitempty\" yaml:\"tx_bytes,omitempty\"" } ""..gostring.1�*go.string."RxDropped"@4 RxDropped *go.string."RxDropped"��go.string."json:\"rx_dropped,omitempty\" yaml:\"rx_dropped,omitempty\""��7json:"rx_dropped,omitempty" yaml:"rx_dropped,omitempty" �go.string."json:\"rx_dropped,omitempty\" yaml:\"rx_dropped,omitempty\""�&go.string."RxBytes"00RxBytes &go.string."RxBytes"��go.string."json:\"rx_bytes,omitempty\" yaml:\"rx_bytes,omitempty\""��3json:"rx_bytes,omitempty" yaml:"rx_bytes,omitempty" �go.string."json:\"rx_bytes,omitempty\" yaml:\"rx_bytes,omitempty\""�(go.string."RxErrors"@2RxErrors (go.string."RxErrors"��go.string."json:\"rx_errors,omitempty\" yaml:\"rx_errors,omitempty\""��5json:"rx_errors,omitempty" yaml:"rx_errors,omitempty" �go.string."json:\"rx_errors,omitempty\" yaml:\"rx_errors,omitempty\""�*go.string."TxPackets"@4 TxPackets *go.string."TxPackets"��go.string."json:\"tx_packets,omitempty\" yaml:\"tx_packets,omitempty\""��7json:"tx_packets,omitempty" yaml:"tx_packets,omitempty" �go.string."json:\"tx_packets,omitempty\" yaml:\"tx_packets,omitempty\""�*go.string."TxDropped"@4 TxDropped *go.string."TxDropped"��go.string."json:\"tx_dropped,omitempty\" yaml:\"tx_dropped,omitempty\""��7json:"tx_dropped,omitempty" yaml:"tx_dropped,omitempty" �go.string."json:\"tx_dropped,omitempty\" yaml:\"tx_dropped,omitempty\""�*go.string."RxPackets"@4 RxPackets *go.string."RxPackets"��go.string."json:\"rx_packets,omitempty\" yaml:\"rx_packets,omitempty\""��7json:"rx_packets,omitempty" yaml:"rx_packets,omitempty" �go.string."json:\"rx_packets,omitempty\" yaml:\"rx_packets,omitempty\""�(go.string."TxErrors"@2TxErrors (go.string."TxErrors"��go.string."json:\"tx_errors,omitempty\" yaml:\"tx_errors,omitempty\""��5json:"tx_errors,omitempty" yaml:"tx_errors,omitempty" �go.string."json:\"tx_errors,omitempty\" yaml:\"tx_errors,omitempty\""�&go.string."TxBytes"00TxBytes &go.string."TxBytes"��go.string."json:\"tx_bytes,omitempty\" yaml:\"tx_bytes,omitempty\""��3json:"tx_bytes,omitempty" yaml:"tx_bytes,omitempty" �go.string."json:\"tx_bytes,omitempty\" yaml:\"tx_bytes,omitempty\""�� type.struct { RxDropped uint64 "json:\"rx_dropped,omitempty\" yaml:\"rx_dropped,omitempty\""; RxBytes uint64 "json:\"rx_bytes,omitempty\" yaml:\"rx_bytes,omitempty\""; RxErrors uint64 "json:\"rx_errors,omitempty\" yaml:\"rx_errors,omitempty\""; TxPackets uint64 "json:\"tx_packets,omitempty\" yaml:\"tx_packets,omitempty\""; TxDropped uint64 "json:\"tx_dropped,omitempty\" yaml:\"tx_dropped,omitempty\""; RxPackets uint64 "json:\"rx_packets,omitempty\" yaml:\"rx_packets,omitempty\""; TxErrors uint64 "json:\"tx_errors,omitempty\" yaml:\"tx_errors,omitempty\""; TxBytes uint64 "json:\"tx_bytes,omitempty\" yaml:\"tx_bytes,omitempty\"" }��@/>*^� (08<  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P""..gostring.1p� +go.weak.type.*struct { RxDropped uint64 "json:\"rx_dropped,omitempty\" yaml:\"rx_dropped,omitempty\""; RxBytes uint64 "json:\"rx_bytes,omitempty\" yaml:\"rx_bytes,omitempty\""; RxErrors uint64 "json:\"rx_errors,omitempty\" yaml:\"rx_errors,omitempty\""; TxPackets uint64 "json:\"tx_packets,omitempty\" yaml:\"tx_packets,omitempty\""; TxDropped uint64 "json:\"tx_dropped,omitempty\" yaml:\"tx_dropped,omitempty\""; RxPackets uint64 "json:\"rx_packets,omitempty\" yaml:\"rx_packets,omitempty\""; TxErrors uint64 "json:\"tx_errors,omitempty\" yaml:\"tx_errors,omitempty\""; TxBytes uint64 "json:\"tx_bytes,omitempty\" yaml:\"tx_bytes,omitempty\"" }�"runtime.zerovalue��� type.struct { RxDropped uint64 "json:\"rx_dropped,omitempty\" yaml:\"rx_dropped,omitempty\""; RxBytes uint64 "json:\"rx_bytes,omitempty\" yaml:\"rx_bytes,omitempty\""; RxErrors uint64 "json:\"rx_errors,omitempty\" yaml:\"rx_errors,omitempty\""; TxPackets uint64 "json:\"tx_packets,omitempty\" yaml:\"tx_packets,omitempty\""; TxDropped uint64 "json:\"tx_dropped,omitempty\" yaml:\"tx_dropped,omitempty\""; RxPackets uint64 "json:\"rx_packets,omitempty\" yaml:\"rx_packets,omitempty\""; TxErrors uint64 "json:\"tx_errors,omitempty\" yaml:\"tx_errors,omitempty\""; TxBytes uint64 "json:\"tx_bytes,omitempty\" yaml:\"tx_bytes,omitempty\"" }�*go.string."RxDropped"�type.uint64��go.string."json:\"rx_dropped,omitempty\" yaml:\"rx_dropped,omitempty\""�&go.string."RxBytes"�type.uint64��go.string."json:\"rx_bytes,omitempty\" yaml:\"rx_bytes,omitempty\""�(go.string."RxErrors"�type.uint64��go.string."json:\"rx_errors,omitempty\" yaml:\"rx_errors,omitempty\""�*go.string."TxPackets"�type.uint64��go.string."json:\"tx_packets,omitempty\" yaml:\"tx_packets,omitempty\""�*go.string."TxDropped"�type.uint64��go.string."json:\"tx_dropped,omitempty\" yaml:\"tx_dropped,omitempty\""�*go.string."RxPackets"�type.uint64��go.string."json:\"rx_packets,omitempty\" yaml:\"rx_packets,omitempty\""�(go.string."TxErrors"�type.uint64��go.string."json:\"tx_errors,omitempty\" yaml:\"tx_errors,omitempty\""�&go.string."TxBytes"�type.uint64��go.string."json:\"tx_bytes,omitempty\" yaml:\"tx_bytes,omitempty\""�""..gostring.2�'�'� struct { TotalPgmafault uint64 "json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""; Cache uint64 "json:\"cache,omitempty\" yaml:\"cache,omitempty\""; MappedFile uint64 "json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""; TotalInactiveFile uint64 "json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""; Pgpgout uint64 "json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""; Rss uint64 "json:\"rss,omitempty\" yaml:\"rss,omitempty\""; TotalMappedFile uint64 "json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""; Writeback uint64 "json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""; Unevictable uint64 "json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""; Pgpgin uint64 "json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""; TotalUnevictable uint64 "json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""; Pgmajfault uint64 "json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""; TotalRss uint64 "json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""; TotalRssHuge uint64 "json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""; TotalWriteback uint64 "json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""; TotalInactiveAnon uint64 "json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""; RssHuge uint64 "json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""; HierarchicalMemoryLimit uint64 "json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""; TotalPgfault uint64 "json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""; TotalActiveFile uint64 "json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""; ActiveAnon uint64 "json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""; TotalActiveAnon uint64 "json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""; TotalPgpgout uint64 "json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""; TotalCache uint64 "json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""; InactiveAnon uint64 "json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""; ActiveFile uint64 "json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""; Pgfault uint64 "json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""; InactiveFile uint64 "json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""; TotalPgpgin uint64 "json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\"" } ""..gostring.2�4go.string."TotalPgmafault"@>TotalPgmafault 4go.string."TotalPgmafault"��go.string."json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""��Ajson:"total_pgmafault,omitempty" yaml:"total_pgmafault,omitempty" �go.string."json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""�"go.string."Cache"0,Cache "go.string."Cache"�zgo.string."json:\"cache,omitempty\" yaml:\"cache,omitempty\""�|-json:"cache,omitempty" yaml:"cache,omitempty" zgo.string."json:\"cache,omitempty\" yaml:\"cache,omitempty\""�,go.string."MappedFile"@6 +MappedFile ,go.string."MappedFile"��go.string."json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""��9json:"mapped_file,omitempty" yaml:"mapped_file,omitempty" �go.string."json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""�:go.string."TotalInactiveFile"PDTotalInactiveFile :go.string."TotalInactiveFile"��go.string."json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""��Ijson:"total_inactive_file,omitempty" yaml:"total_inactive_file,omitempty" �go.string."json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""�&go.string."Pgpgout"00Pgpgout &go.string."Pgpgout"��go.string."json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""��1json:"pgpgout,omitempty" yaml:"pgpgout,omitempty" �go.string."json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""�go.string."Rss"0(Rss go.string."Rss"�rgo.string."json:\"rss,omitempty\" yaml:\"rss,omitempty\""�t)json:"rss,omitempty" yaml:"rss,omitempty" rgo.string."json:\"rss,omitempty\" yaml:\"rss,omitempty\""�6go.string."TotalMappedFile"@@TotalMappedFile 6go.string."TotalMappedFile"��go.string."json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""��Ejson:"total_mapped_file,omitempty" yaml:"total_mapped_file,omitempty" �go.string."json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""�*go.string."Writeback"@4 Writeback *go.string."Writeback"��go.string."json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""��5json:"writeback,omitempty" yaml:"writeback,omitempty" �go.string."json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""�.go.string."Unevictable"@8 Unevictable .go.string."Unevictable"��go.string."json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""��9json:"unevictable,omitempty" yaml:"unevictable,omitempty" �go.string."json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""�$go.string."Pgpgin"0.Pgpgin $go.string."Pgpgin"�~go.string."json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""��/json:"pgpgin,omitempty" yaml:"pgpgin,omitempty" ~go.string."json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""�8go.string."TotalUnevictable"PBTotalUnevictable 8go.string."TotalUnevictable"��go.string."json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""��Ejson:"total_unevictable,omitempty" yaml:"total_unevictable,omitempty" �go.string."json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""�,go.string."Pgmajfault"@6 +Pgmajfault ,go.string."Pgmajfault"��go.string."json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""��7json:"pgmajfault,omitempty" yaml:"pgmajfault,omitempty" �go.string."json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""�(go.string."TotalRss"@2TotalRss (go.string."TotalRss"��go.string."json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""��5json:"total_rss,omitempty" yaml:"total_rss,omitempty" �go.string."json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""�0go.string."TotalRssHuge"@: TotalRssHuge 0go.string."TotalRssHuge"��go.string."json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""��?json:"total_rss_huge,omitempty" yaml:"total_rss_huge,omitempty" �go.string."json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""�4go.string."TotalWriteback"@>TotalWriteback 4go.string."TotalWriteback"��go.string."json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""��Ajson:"total_writeback,omitempty" yaml:"total_writeback,omitempty" �go.string."json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""�:go.string."TotalInactiveAnon"PDTotalInactiveAnon :go.string."TotalInactiveAnon"��go.string."json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""��Ijson:"total_inactive_anon,omitempty" yaml:"total_inactive_anon,omitempty" �go.string."json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""�&go.string."RssHuge"00RssHuge &go.string."RssHuge"��go.string."json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""��3json:"rss_huge,omitempty" yaml:"rss_huge,omitempty" �go.string."json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""�Fgo.string."HierarchicalMemoryLimit"PPHierarchicalMemoryLimit Fgo.string."HierarchicalMemoryLimit"��go.string."json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""��Ujson:"hierarchical_memory_limit,omitempty" yaml:"hierarchical_memory_limit,omitempty" �go.string."json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""�0go.string."TotalPgfault"@: TotalPgfault 0go.string."TotalPgfault"��go.string."json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""��=json:"total_pgfault,omitempty" yaml:"total_pgfault,omitempty" �go.string."json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""�6go.string."TotalActiveFile"@@TotalActiveFile 6go.string."TotalActiveFile"��go.string."json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""��Ejson:"total_active_file,omitempty" yaml:"total_active_file,omitempty" �go.string."json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""�,go.string."ActiveAnon"@6 +ActiveAnon ,go.string."ActiveAnon"��go.string."json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""��9json:"active_anon,omitempty" yaml:"active_anon,omitempty" �go.string."json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""�6go.string."TotalActiveAnon"@@TotalActiveAnon 6go.string."TotalActiveAnon"��go.string."json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""��Ejson:"total_active_anon,omitempty" yaml:"total_active_anon,omitempty" �go.string."json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""�0go.string."TotalPgpgout"@: TotalPgpgout 0go.string."TotalPgpgout"��go.string."json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""��=json:"total_pgpgout,omitempty" yaml:"total_pgpgout,omitempty" �go.string."json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""�,go.string."TotalCache"@6 +TotalCache ,go.string."TotalCache"��go.string."json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""��9json:"total_cache,omitempty" yaml:"total_cache,omitempty" �go.string."json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""�0go.string."InactiveAnon"@: InactiveAnon 0go.string."InactiveAnon"��go.string."json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""��=json:"inactive_anon,omitempty" yaml:"inactive_anon,omitempty" �go.string."json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""�,go.string."ActiveFile"@6 +ActiveFile ,go.string."ActiveFile"��go.string."json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""��9json:"active_file,omitempty" yaml:"active_file,omitempty" �go.string."json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""�&go.string."Pgfault"00Pgfault &go.string."Pgfault"��go.string."json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""��1json:"pgfault,omitempty" yaml:"pgfault,omitempty" �go.string."json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""�0go.string."InactiveFile"@: InactiveFile 0go.string."InactiveFile"��go.string."json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""��=json:"inactive_file,omitempty" yaml:"inactive_file,omitempty" �go.string."json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""�.go.string."TotalPgpgin"@8 TotalPgpgin .go.string."TotalPgpgin"��go.string."json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\""��;json:"total_pgpgin,omitempty" yaml:"total_pgpgin,omitempty" �go.string."json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\""��'type.struct { TotalPgmafault uint64 "json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""; Cache uint64 "json:\"cache,omitempty\" yaml:\"cache,omitempty\""; MappedFile uint64 "json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""; TotalInactiveFile uint64 "json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""; Pgpgout uint64 "json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""; Rss uint64 "json:\"rss,omitempty\" yaml:\"rss,omitempty\""; TotalMappedFile uint64 "json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""; Writeback uint64 "json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""; Unevictable uint64 "json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""; Pgpgin uint64 "json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""; TotalUnevictable uint64 "json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""; Pgmajfault uint64 "json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""; TotalRss uint64 "json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""; TotalRssHuge uint64 "json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""; TotalWriteback uint64 "json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""; TotalInactiveAnon uint64 "json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""; RssHuge uint64 "json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""; HierarchicalMemoryLimit uint64 "json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""; TotalPgfault uint64 "json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""; TotalActiveFile uint64 "json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""; ActiveAnon uint64 "json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""; TotalActiveAnon uint64 "json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""; TotalPgpgout uint64 "json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""; TotalCache uint64 "json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""; InactiveAnon uint64 "json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""; ActiveFile uint64 "json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""; Pgfault uint64 "json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""; InactiveFile uint64 "json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""; TotalPgpgin uint64 "json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\"" }���齪�� (08@HPX`hpx��������������  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P""..gostring.2p�'go.weak.type.*struct { TotalPgmafault uint64 "json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""; Cache uint64 "json:\"cache,omitempty\" yaml:\"cache,omitempty\""; MappedFile uint64 "json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""; TotalInactiveFile uint64 "json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""; Pgpgout uint64 "json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""; Rss uint64 "json:\"rss,omitempty\" yaml:\"rss,omitempty\""; TotalMappedFile uint64 "json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""; Writeback uint64 "json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""; Unevictable uint64 "json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""; Pgpgin uint64 "json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""; TotalUnevictable uint64 "json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""; Pgmajfault uint64 "json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""; TotalRss uint64 "json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""; TotalRssHuge uint64 "json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""; TotalWriteback uint64 "json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""; TotalInactiveAnon uint64 "json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""; RssHuge uint64 "json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""; HierarchicalMemoryLimit uint64 "json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""; TotalPgfault uint64 "json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""; TotalActiveFile uint64 "json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""; ActiveAnon uint64 "json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""; TotalActiveAnon uint64 "json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""; TotalPgpgout uint64 "json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""; TotalCache uint64 "json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""; InactiveAnon uint64 "json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""; ActiveFile uint64 "json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""; Pgfault uint64 "json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""; InactiveFile uint64 "json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""; TotalPgpgin uint64 "json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\"" }�"runtime.zerovalue���'type.struct { TotalPgmafault uint64 "json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""; Cache uint64 "json:\"cache,omitempty\" yaml:\"cache,omitempty\""; MappedFile uint64 "json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""; TotalInactiveFile uint64 "json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""; Pgpgout uint64 "json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""; Rss uint64 "json:\"rss,omitempty\" yaml:\"rss,omitempty\""; TotalMappedFile uint64 "json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""; Writeback uint64 "json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""; Unevictable uint64 "json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""; Pgpgin uint64 "json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""; TotalUnevictable uint64 "json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""; Pgmajfault uint64 "json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""; TotalRss uint64 "json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""; TotalRssHuge uint64 "json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""; TotalWriteback uint64 "json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""; TotalInactiveAnon uint64 "json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""; RssHuge uint64 "json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""; HierarchicalMemoryLimit uint64 "json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""; TotalPgfault uint64 "json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""; TotalActiveFile uint64 "json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""; ActiveAnon uint64 "json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""; TotalActiveAnon uint64 "json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""; TotalPgpgout uint64 "json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""; TotalCache uint64 "json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""; InactiveAnon uint64 "json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""; ActiveFile uint64 "json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""; Pgfault uint64 "json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""; InactiveFile uint64 "json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""; TotalPgpgin uint64 "json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\"" }�4go.string."TotalPgmafault"�type.uint64��go.string."json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""�"go.string."Cache"�type.uint64�zgo.string."json:\"cache,omitempty\" yaml:\"cache,omitempty\""�,go.string."MappedFile"�type.uint64��go.string."json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""�:go.string."TotalInactiveFile"�type.uint64��go.string."json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""�&go.string."Pgpgout"�type.uint64��go.string."json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""�go.string."Rss"�type.uint64�rgo.string."json:\"rss,omitempty\" yaml:\"rss,omitempty\""�6go.string."TotalMappedFile"�type.uint64��go.string."json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""�*go.string."Writeback"�type.uint64��go.string."json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""�.go.string."Unevictable"�type.uint64��go.string."json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""�$go.string."Pgpgin"�type.uint64�~go.string."json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""�8go.string."TotalUnevictable"�type.uint64��go.string."json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""�,go.string."Pgmajfault"�type.uint64��go.string."json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""� (go.string."TotalRss"� type.uint64� �go.string."json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""� 0go.string."TotalRssHuge"� type.uint64� +�go.string."json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""� +4go.string."TotalWriteback"� +type.uint64� +�go.string."json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""� +:go.string."TotalInactiveAnon"� type.uint64� �go.string."json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""� &go.string."RssHuge"� type.uint64� �go.string."json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""� Fgo.string."HierarchicalMemoryLimit"� type.uint64� �go.string."json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""� 0go.string."TotalPgfault"� type.uint64� �go.string."json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""� 6go.string."TotalActiveFile"� type.uint64� �go.string."json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""�,go.string."ActiveAnon"�type.uint64��go.string."json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""�6go.string."TotalActiveAnon"�type.uint64��go.string."json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""�0go.string."TotalPgpgout"�type.uint64��go.string."json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""�,go.string."TotalCache"�type.uint64��go.string."json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""�0go.string."InactiveAnon"�type.uint64��go.string."json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""�,go.string."ActiveFile"�type.uint64��go.string."json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""�&go.string."Pgfault"�type.uint64��go.string."json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""�0go.string."InactiveFile"�type.uint64��go.string."json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""�.go.string."TotalPgpgin"�type.uint64��go.string."json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\""�""..gostring.3�-�-D struct { Stats struct { TotalPgmafault uint64 "json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""; Cache uint64 "json:\"cache,omitempty\" yaml:\"cache,omitempty\""; MappedFile uint64 "json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""; TotalInactiveFile uint64 "json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""; Pgpgout uint64 "json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""; Rss uint64 "json:\"rss,omitempty\" yaml:\"rss,omitempty\""; TotalMappedFile uint64 "json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""; Writeback uint64 "json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""; Unevictable uint64 "json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""; Pgpgin uint64 "json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""; TotalUnevictable uint64 "json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""; Pgmajfault uint64 "json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""; TotalRss uint64 "json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""; TotalRssHuge uint64 "json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""; TotalWriteback uint64 "json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""; TotalInactiveAnon uint64 "json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""; RssHuge uint64 "json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""; HierarchicalMemoryLimit uint64 "json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""; TotalPgfault uint64 "json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""; TotalActiveFile uint64 "json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""; ActiveAnon uint64 "json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""; TotalActiveAnon uint64 "json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""; TotalPgpgout uint64 "json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""; TotalCache uint64 "json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""; InactiveAnon uint64 "json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""; ActiveFile uint64 "json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""; Pgfault uint64 "json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""; InactiveFile uint64 "json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""; TotalPgpgin uint64 "json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\"" } "json:\"stats,omitempty\" yaml:\"stats,omitempty\""; MaxUsage uint64 "json:\"max_usage,omitempty\" yaml:\"max_usage,omitempty\""; Usage uint64 "json:\"usage,omitempty\" yaml:\"usage,omitempty\""; Failcnt uint64 "json:\"failcnt,omitempty\" yaml:\"failcnt,omitempty\""; Limit uint64 "json:\"limit,omitempty\" yaml:\"limit,omitempty\"" } ""..gostring.3�"go.string."Stats"0,Stats "go.string."Stats"�zgo.string."json:\"stats,omitempty\" yaml:\"stats,omitempty\""�|-json:"stats,omitempty" yaml:"stats,omitempty" zgo.string."json:\"stats,omitempty\" yaml:\"stats,omitempty\""�(go.string."MaxUsage"@2MaxUsage (go.string."MaxUsage"��go.string."json:\"max_usage,omitempty\" yaml:\"max_usage,omitempty\""��5json:"max_usage,omitempty" yaml:"max_usage,omitempty" �go.string."json:\"max_usage,omitempty\" yaml:\"max_usage,omitempty\""�"go.string."Usage"0,Usage "go.string."Usage"�zgo.string."json:\"usage,omitempty\" yaml:\"usage,omitempty\""�|-json:"usage,omitempty" yaml:"usage,omitempty" zgo.string."json:\"usage,omitempty\" yaml:\"usage,omitempty\""�&go.string."Failcnt"00Failcnt &go.string."Failcnt"��go.string."json:\"failcnt,omitempty\" yaml:\"failcnt,omitempty\""��1json:"failcnt,omitempty" yaml:"failcnt,omitempty" �go.string."json:\"failcnt,omitempty\" yaml:\"failcnt,omitempty\""�zgo.string."json:\"limit,omitempty\" yaml:\"limit,omitempty\""�|-json:"limit,omitempty" yaml:"limit,omitempty" zgo.string."json:\"limit,omitempty\" yaml:\"limit,omitempty\""��-type.struct { Stats struct { TotalPgmafault uint64 "json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""; Cache uint64 "json:\"cache,omitempty\" yaml:\"cache,omitempty\""; MappedFile uint64 "json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""; TotalInactiveFile uint64 "json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""; Pgpgout uint64 "json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""; Rss uint64 "json:\"rss,omitempty\" yaml:\"rss,omitempty\""; TotalMappedFile uint64 "json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""; Writeback uint64 "json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""; Unevictable uint64 "json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""; Pgpgin uint64 "json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""; TotalUnevictable uint64 "json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""; Pgmajfault uint64 "json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""; TotalRss uint64 "json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""; TotalRssHuge uint64 "json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""; TotalWriteback uint64 "json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""; TotalInactiveAnon uint64 "json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""; RssHuge uint64 "json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""; HierarchicalMemoryLimit uint64 "json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""; TotalPgfault uint64 "json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""; TotalActiveFile uint64 "json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""; ActiveAnon uint64 "json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""; TotalActiveAnon uint64 "json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""; TotalPgpgout uint64 "json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""; TotalCache uint64 "json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""; InactiveAnon uint64 "json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""; ActiveFile uint64 "json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""; Pgfault uint64 "json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""; InactiveFile uint64 "json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""; TotalPgpgin uint64 "json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\"" } "json:\"stats,omitempty\" yaml:\"stats,omitempty\""; MaxUsage uint64 "json:\"max_usage,omitempty\" yaml:\"max_usage,omitempty\""; Usage uint64 "json:\"usage,omitempty\" yaml:\"usage,omitempty\""; Failcnt uint64 "json:\"failcnt,omitempty\" yaml:\"failcnt,omitempty\""; Limit uint64 "json:\"limit,omitempty\" yaml:\"limit,omitempty\"" }��� ������*  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P""..gostring.3p�-go.weak.type.*struct { Stats struct { TotalPgmafault uint64 "json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""; Cache uint64 "json:\"cache,omitempty\" yaml:\"cache,omitempty\""; MappedFile uint64 "json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""; TotalInactiveFile uint64 "json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""; Pgpgout uint64 "json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""; Rss uint64 "json:\"rss,omitempty\" yaml:\"rss,omitempty\""; TotalMappedFile uint64 "json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""; Writeback uint64 "json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""; Unevictable uint64 "json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""; Pgpgin uint64 "json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""; TotalUnevictable uint64 "json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""; Pgmajfault uint64 "json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""; TotalRss uint64 "json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""; TotalRssHuge uint64 "json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""; TotalWriteback uint64 "json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""; TotalInactiveAnon uint64 "json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""; RssHuge uint64 "json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""; HierarchicalMemoryLimit uint64 "json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""; TotalPgfault uint64 "json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""; TotalActiveFile uint64 "json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""; ActiveAnon uint64 "json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""; TotalActiveAnon uint64 "json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""; TotalPgpgout uint64 "json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""; TotalCache uint64 "json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""; InactiveAnon uint64 "json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""; ActiveFile uint64 "json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""; Pgfault uint64 "json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""; InactiveFile uint64 "json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""; TotalPgpgin uint64 "json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\"" } "json:\"stats,omitempty\" yaml:\"stats,omitempty\""; MaxUsage uint64 "json:\"max_usage,omitempty\" yaml:\"max_usage,omitempty\""; Usage uint64 "json:\"usage,omitempty\" yaml:\"usage,omitempty\""; Failcnt uint64 "json:\"failcnt,omitempty\" yaml:\"failcnt,omitempty\""; Limit uint64 "json:\"limit,omitempty\" yaml:\"limit,omitempty\"" }�"runtime.zerovalue���-type.struct { Stats struct { TotalPgmafault uint64 "json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""; Cache uint64 "json:\"cache,omitempty\" yaml:\"cache,omitempty\""; MappedFile uint64 "json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""; TotalInactiveFile uint64 "json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""; Pgpgout uint64 "json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""; Rss uint64 "json:\"rss,omitempty\" yaml:\"rss,omitempty\""; TotalMappedFile uint64 "json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""; Writeback uint64 "json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""; Unevictable uint64 "json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""; Pgpgin uint64 "json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""; TotalUnevictable uint64 "json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""; Pgmajfault uint64 "json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""; TotalRss uint64 "json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""; TotalRssHuge uint64 "json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""; TotalWriteback uint64 "json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""; TotalInactiveAnon uint64 "json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""; RssHuge uint64 "json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""; HierarchicalMemoryLimit uint64 "json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""; TotalPgfault uint64 "json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""; TotalActiveFile uint64 "json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""; ActiveAnon uint64 "json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""; TotalActiveAnon uint64 "json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""; TotalPgpgout uint64 "json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""; TotalCache uint64 "json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""; InactiveAnon uint64 "json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""; ActiveFile uint64 "json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""; Pgfault uint64 "json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""; InactiveFile uint64 "json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""; TotalPgpgin uint64 "json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\"" } "json:\"stats,omitempty\" yaml:\"stats,omitempty\""; MaxUsage uint64 "json:\"max_usage,omitempty\" yaml:\"max_usage,omitempty\""; Usage uint64 "json:\"usage,omitempty\" yaml:\"usage,omitempty\""; Failcnt uint64 "json:\"failcnt,omitempty\" yaml:\"failcnt,omitempty\""; Limit uint64 "json:\"limit,omitempty\" yaml:\"limit,omitempty\"" }�"go.string."Stats"��'type.struct { TotalPgmafault uint64 "json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""; Cache uint64 "json:\"cache,omitempty\" yaml:\"cache,omitempty\""; MappedFile uint64 "json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""; TotalInactiveFile uint64 "json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""; Pgpgout uint64 "json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""; Rss uint64 "json:\"rss,omitempty\" yaml:\"rss,omitempty\""; TotalMappedFile uint64 "json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""; Writeback uint64 "json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""; Unevictable uint64 "json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""; Pgpgin uint64 "json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""; TotalUnevictable uint64 "json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""; Pgmajfault uint64 "json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""; TotalRss uint64 "json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""; TotalRssHuge uint64 "json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""; TotalWriteback uint64 "json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""; TotalInactiveAnon uint64 "json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""; RssHuge uint64 "json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""; HierarchicalMemoryLimit uint64 "json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""; TotalPgfault uint64 "json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""; TotalActiveFile uint64 "json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""; ActiveAnon uint64 "json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""; TotalActiveAnon uint64 "json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""; TotalPgpgout uint64 "json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""; TotalCache uint64 "json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""; InactiveAnon uint64 "json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""; ActiveFile uint64 "json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""; Pgfault uint64 "json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""; InactiveFile uint64 "json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""; TotalPgpgin uint64 "json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\"" }�zgo.string."json:\"stats,omitempty\" yaml:\"stats,omitempty\""�(go.string."MaxUsage"�type.uint64��go.string."json:\"max_usage,omitempty\" yaml:\"max_usage,omitempty\""�"go.string."Usage"�type.uint64�zgo.string."json:\"usage,omitempty\" yaml:\"usage,omitempty\""�&go.string."Failcnt"�type.uint64��go.string."json:\"failcnt,omitempty\" yaml:\"failcnt,omitempty\""�"go.string."Limit"�type.uint64�zgo.string."json:\"limit,omitempty\" yaml:\"limit,omitempty\""�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�Btype..hashfunc."".BlkioStatsEntry:type..hash."".BlkioStatsEntry�>type..eqfunc."".BlkioStatsEntry6type..eq."".BlkioStatsEntry�8type..alg."".BlkioStatsEntry Btype..hashfunc."".BlkioStatsEntry>type..eqfunc."".BlkioStatsEntry�Fgo.string."*docker.BlkioStatsEntry"PP*docker.BlkioStatsEntry Fgo.string."*docker.BlkioStatsEntry"�0type.*"".BlkioStatsEntry��K��66 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PFgo.string."*docker.BlkioStatsEntry"pBgo.weak.type.**"".BlkioStatsEntry�"runtime.zerovalue�.type."".BlkioStatsEntry�bruntime.gcbits.0x44484484440000000000000000000000 DHD�D�Dgo.string."docker.BlkioStatsEntry"PNdocker.BlkioStatsEntry Dgo.string."docker.BlkioStatsEntry"�"go.string."Major"0,Major "go.string."Major"�zgo.string."json:\"major,omitempty\" yaml:\"major,omitempty\""�|-json:"major,omitempty" yaml:"major,omitempty" zgo.string."json:\"major,omitempty\" yaml:\"major,omitempty\""�"go.string."Minor"0,Minor "go.string."Minor"�zgo.string."json:\"minor,omitempty\" yaml:\"minor,omitempty\""�|-json:"minor,omitempty" yaml:"minor,omitempty" zgo.string."json:\"minor,omitempty\" yaml:\"minor,omitempty\""�go.string."Op"0&Op go.string."Op"�ngo.string."json:\"op,omitempty\" yaml:\"op,omitempty\""pp'json:"op,omitempty" yaml:"op,omitempty" ngo.string."json:\"op,omitempty\" yaml:\"op,omitempty\""�zgo.string."json:\"value,omitempty\" yaml:\"value,omitempty\""�|-json:"value,omitempty" yaml:"value,omitempty" zgo.string."json:\"value,omitempty\" yaml:\"value,omitempty\""�6go.string."BlkioStatsEntry"@@BlkioStatsEntry 6go.string."BlkioStatsEntry"�.type."".BlkioStatsEntry��(�-�g , 8type..alg."".BlkioStatsEntry0bruntime.gcbits.0x44484484440000000000000000000000PDgo.string."docker.BlkioStatsEntry"p0type.*"".BlkioStatsEntry�"runtime.zerovalue��.type."".BlkioStatsEntry�"go.string."Major"�type.uint64�zgo.string."json:\"major,omitempty\" yaml:\"major,omitempty\""�"go.string."Minor"�type.uint64�zgo.string."json:\"minor,omitempty\" yaml:\"minor,omitempty\""�go.string."Op"�type.string�ngo.string."json:\"op,omitempty\" yaml:\"op,omitempty\""�"go.string."Value"�type.uint64�zgo.string."json:\"value,omitempty\" yaml:\"value,omitempty\""`�.type."".BlkioStatsEntry�6go.string."BlkioStatsEntry"�"go.importpath."".��.type."".BlkioStatsEntry�Hgo.string."[]docker.BlkioStatsEntry"`R[]docker.BlkioStatsEntry Hgo.string."[]docker.BlkioStatsEntry"�2type.[]"".BlkioStatsEntry���/# � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000PHgo.string."[]docker.BlkioStatsEntry"pDgo.weak.type.*[]"".BlkioStatsEntry�"runtime.zerovalue�.type."".BlkioStatsEntry�rgo.typelink.[]docker.BlkioStatsEntry/[]"".BlkioStatsEntry2type.[]"".BlkioStatsEntry�""..gostring.4��struct { IOServiceBytesRecursive []docker.BlkioStatsEntry "json:\"io_service_bytes_recursive,omitempty\" yaml:\"io_service_bytes_recursive,omitempty\""; IOServicedRecursive []docker.BlkioStatsEntry "json:\"io_serviced_recursive,omitempty\" yaml:\"io_serviced_recursive,omitempty\""; IOQueueRecursive []docker.BlkioStatsEntry "json:\"io_queue_recursive,omitempty\" yaml:\"io_queue_recursive,omitempty\""; IOServiceTimeRecursive []docker.BlkioStatsEntry "json:\"io_service_time_recursive,omitempty\" yaml:\"io_service_time_recursive,omitempty\""; IOWaitTimeRecursive []docker.BlkioStatsEntry "json:\"io_wait_time_recursive,omitempty\" yaml:\"io_wait_time_recursive,omitempty\""; IOMergedRecursive []docker.BlkioStatsEntry "json:\"io_merged_recursive,omitempty\" yaml:\"io_merged_recursive,omitempty\""; IOTimeRecursive []docker.BlkioStatsEntry "json:\"io_time_recursive,omitempty\" yaml:\"io_time_recursive,omitempty\""; SectorsRecursive []docker.BlkioStatsEntry "json:\"sectors_recursive,omitempty\" yaml:\"sectors_recursive,omitempty\"" } ""..gostring.4�Fgo.string."IOServiceBytesRecursive"PPIOServiceBytesRecursive Fgo.string."IOServiceBytesRecursive"��go.string."json:\"io_service_bytes_recursive,omitempty\" yaml:\"io_service_bytes_recursive,omitempty\""��Wjson:"io_service_bytes_recursive,omitempty" yaml:"io_service_bytes_recursive,omitempty" �go.string."json:\"io_service_bytes_recursive,omitempty\" yaml:\"io_service_bytes_recursive,omitempty\""�>go.string."IOServicedRecursive"PHIOServicedRecursive >go.string."IOServicedRecursive"��go.string."json:\"io_serviced_recursive,omitempty\" yaml:\"io_serviced_recursive,omitempty\""��Mjson:"io_serviced_recursive,omitempty" yaml:"io_serviced_recursive,omitempty" �go.string."json:\"io_serviced_recursive,omitempty\" yaml:\"io_serviced_recursive,omitempty\""�8go.string."IOQueueRecursive"PBIOQueueRecursive 8go.string."IOQueueRecursive"��go.string."json:\"io_queue_recursive,omitempty\" yaml:\"io_queue_recursive,omitempty\""��Gjson:"io_queue_recursive,omitempty" yaml:"io_queue_recursive,omitempty" �go.string."json:\"io_queue_recursive,omitempty\" yaml:\"io_queue_recursive,omitempty\""�Dgo.string."IOServiceTimeRecursive"PNIOServiceTimeRecursive Dgo.string."IOServiceTimeRecursive"��go.string."json:\"io_service_time_recursive,omitempty\" yaml:\"io_service_time_recursive,omitempty\""��Ujson:"io_service_time_recursive,omitempty" yaml:"io_service_time_recursive,omitempty" �go.string."json:\"io_service_time_recursive,omitempty\" yaml:\"io_service_time_recursive,omitempty\""�>go.string."IOWaitTimeRecursive"PHIOWaitTimeRecursive >go.string."IOWaitTimeRecursive"��go.string."json:\"io_wait_time_recursive,omitempty\" yaml:\"io_wait_time_recursive,omitempty\""��Ojson:"io_wait_time_recursive,omitempty" yaml:"io_wait_time_recursive,omitempty" �go.string."json:\"io_wait_time_recursive,omitempty\" yaml:\"io_wait_time_recursive,omitempty\""�:go.string."IOMergedRecursive"PDIOMergedRecursive :go.string."IOMergedRecursive"��go.string."json:\"io_merged_recursive,omitempty\" yaml:\"io_merged_recursive,omitempty\""��Ijson:"io_merged_recursive,omitempty" yaml:"io_merged_recursive,omitempty" �go.string."json:\"io_merged_recursive,omitempty\" yaml:\"io_merged_recursive,omitempty\""�6go.string."IOTimeRecursive"@@IOTimeRecursive 6go.string."IOTimeRecursive"��go.string."json:\"io_time_recursive,omitempty\" yaml:\"io_time_recursive,omitempty\""��Ejson:"io_time_recursive,omitempty" yaml:"io_time_recursive,omitempty" �go.string."json:\"io_time_recursive,omitempty\" yaml:\"io_time_recursive,omitempty\""�8go.string."SectorsRecursive"PBSectorsRecursive 8go.string."SectorsRecursive"��go.string."json:\"sectors_recursive,omitempty\" yaml:\"sectors_recursive,omitempty\""��Ejson:"sectors_recursive,omitempty" yaml:"sectors_recursive,omitempty" �go.string."json:\"sectors_recursive,omitempty\" yaml:\"sectors_recursive,omitempty\""��type.struct { IOServiceBytesRecursive []"".BlkioStatsEntry "json:\"io_service_bytes_recursive,omitempty\" yaml:\"io_service_bytes_recursive,omitempty\""; IOServicedRecursive []"".BlkioStatsEntry "json:\"io_serviced_recursive,omitempty\" yaml:\"io_serviced_recursive,omitempty\""; IOQueueRecursive []"".BlkioStatsEntry "json:\"io_queue_recursive,omitempty\" yaml:\"io_queue_recursive,omitempty\""; IOServiceTimeRecursive []"".BlkioStatsEntry "json:\"io_service_time_recursive,omitempty\" yaml:\"io_service_time_recursive,omitempty\""; IOWaitTimeRecursive []"".BlkioStatsEntry "json:\"io_wait_time_recursive,omitempty\" yaml:\"io_wait_time_recursive,omitempty\""; IOMergedRecursive []"".BlkioStatsEntry "json:\"io_merged_recursive,omitempty\" yaml:\"io_merged_recursive,omitempty\""; IOTimeRecursive []"".BlkioStatsEntry "json:\"io_time_recursive,omitempty\" yaml:\"io_time_recursive,omitempty\""; SectorsRecursive []"".BlkioStatsEntry "json:\"sectors_recursive,omitempty\" yaml:\"sectors_recursive,omitempty\"" }����V�*0H`x��< � runtime.algarray0bruntime.gcbits.0x48844448844448844448844400000000P""..gostring.4p�go.weak.type.*struct { IOServiceBytesRecursive []"".BlkioStatsEntry "json:\"io_service_bytes_recursive,omitempty\" yaml:\"io_service_bytes_recursive,omitempty\""; IOServicedRecursive []"".BlkioStatsEntry "json:\"io_serviced_recursive,omitempty\" yaml:\"io_serviced_recursive,omitempty\""; IOQueueRecursive []"".BlkioStatsEntry "json:\"io_queue_recursive,omitempty\" yaml:\"io_queue_recursive,omitempty\""; IOServiceTimeRecursive []"".BlkioStatsEntry "json:\"io_service_time_recursive,omitempty\" yaml:\"io_service_time_recursive,omitempty\""; IOWaitTimeRecursive []"".BlkioStatsEntry "json:\"io_wait_time_recursive,omitempty\" yaml:\"io_wait_time_recursive,omitempty\""; IOMergedRecursive []"".BlkioStatsEntry "json:\"io_merged_recursive,omitempty\" yaml:\"io_merged_recursive,omitempty\""; IOTimeRecursive []"".BlkioStatsEntry "json:\"io_time_recursive,omitempty\" yaml:\"io_time_recursive,omitempty\""; SectorsRecursive []"".BlkioStatsEntry "json:\"sectors_recursive,omitempty\" yaml:\"sectors_recursive,omitempty\"" }�"runtime.zerovalue���type.struct { IOServiceBytesRecursive []"".BlkioStatsEntry "json:\"io_service_bytes_recursive,omitempty\" yaml:\"io_service_bytes_recursive,omitempty\""; IOServicedRecursive []"".BlkioStatsEntry "json:\"io_serviced_recursive,omitempty\" yaml:\"io_serviced_recursive,omitempty\""; IOQueueRecursive []"".BlkioStatsEntry "json:\"io_queue_recursive,omitempty\" yaml:\"io_queue_recursive,omitempty\""; IOServiceTimeRecursive []"".BlkioStatsEntry "json:\"io_service_time_recursive,omitempty\" yaml:\"io_service_time_recursive,omitempty\""; IOWaitTimeRecursive []"".BlkioStatsEntry "json:\"io_wait_time_recursive,omitempty\" yaml:\"io_wait_time_recursive,omitempty\""; IOMergedRecursive []"".BlkioStatsEntry "json:\"io_merged_recursive,omitempty\" yaml:\"io_merged_recursive,omitempty\""; IOTimeRecursive []"".BlkioStatsEntry "json:\"io_time_recursive,omitempty\" yaml:\"io_time_recursive,omitempty\""; SectorsRecursive []"".BlkioStatsEntry "json:\"sectors_recursive,omitempty\" yaml:\"sectors_recursive,omitempty\"" }�Fgo.string."IOServiceBytesRecursive"�2type.[]"".BlkioStatsEntry��go.string."json:\"io_service_bytes_recursive,omitempty\" yaml:\"io_service_bytes_recursive,omitempty\""�>go.string."IOServicedRecursive"�2type.[]"".BlkioStatsEntry��go.string."json:\"io_serviced_recursive,omitempty\" yaml:\"io_serviced_recursive,omitempty\""�8go.string."IOQueueRecursive"�2type.[]"".BlkioStatsEntry��go.string."json:\"io_queue_recursive,omitempty\" yaml:\"io_queue_recursive,omitempty\""�Dgo.string."IOServiceTimeRecursive"�2type.[]"".BlkioStatsEntry��go.string."json:\"io_service_time_recursive,omitempty\" yaml:\"io_service_time_recursive,omitempty\""�>go.string."IOWaitTimeRecursive"�2type.[]"".BlkioStatsEntry��go.string."json:\"io_wait_time_recursive,omitempty\" yaml:\"io_wait_time_recursive,omitempty\""�:go.string."IOMergedRecursive"�2type.[]"".BlkioStatsEntry��go.string."json:\"io_merged_recursive,omitempty\" yaml:\"io_merged_recursive,omitempty\""�6go.string."IOTimeRecursive"�2type.[]"".BlkioStatsEntry��go.string."json:\"io_time_recursive,omitempty\" yaml:\"io_time_recursive,omitempty\""�8go.string."SectorsRecursive"�2type.[]"".BlkioStatsEntry��go.string."json:\"sectors_recursive,omitempty\" yaml:\"sectors_recursive,omitempty\""�(go.string."[]uint64"@2[]uint64 (go.string."[]uint64"�type.[]uint64��?�i  � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P(go.string."[]uint64"p,go.weak.type.*[]uint64�"runtime.zerovalue�type.uint64�:go.typelink.[]uint64/[]uint64type.[]uint64�bruntime.gcbits.0x48444400000000000000000000000000 HDD�""..gostring.5���struct { PercpuUsage []uint64 "json:\"percpu_usage,omitempty\" yaml:\"percpu_usage,omitempty\""; UsageInUsermode uint64 "json:\"usage_in_usermode,omitempty\" yaml:\"usage_in_usermode,omitempty\""; TotalUsage uint64 "json:\"total_usage,omitempty\" yaml:\"total_usage,omitempty\""; UsageInKernelmode uint64 "json:\"usage_in_kernelmode,omitempty\" yaml:\"usage_in_kernelmode,omitempty\"" } ""..gostring.5�.go.string."PercpuUsage"@8 PercpuUsage .go.string."PercpuUsage"��go.string."json:\"percpu_usage,omitempty\" yaml:\"percpu_usage,omitempty\""��;json:"percpu_usage,omitempty" yaml:"percpu_usage,omitempty" �go.string."json:\"percpu_usage,omitempty\" yaml:\"percpu_usage,omitempty\""�6go.string."UsageInUsermode"@@UsageInUsermode 6go.string."UsageInUsermode"��go.string."json:\"usage_in_usermode,omitempty\" yaml:\"usage_in_usermode,omitempty\""��Ejson:"usage_in_usermode,omitempty" yaml:"usage_in_usermode,omitempty" �go.string."json:\"usage_in_usermode,omitempty\" yaml:\"usage_in_usermode,omitempty\""�,go.string."TotalUsage"@6 +TotalUsage ,go.string."TotalUsage"��go.string."json:\"total_usage,omitempty\" yaml:\"total_usage,omitempty\""��9json:"total_usage,omitempty" yaml:"total_usage,omitempty" �go.string."json:\"total_usage,omitempty\" yaml:\"total_usage,omitempty\""�:go.string."UsageInKernelmode"PDUsageInKernelmode :go.string."UsageInKernelmode"��go.string."json:\"usage_in_kernelmode,omitempty\" yaml:\"usage_in_kernelmode,omitempty\""��Ijson:"usage_in_kernelmode,omitempty" yaml:"usage_in_kernelmode,omitempty" �go.string."json:\"usage_in_kernelmode,omitempty\" yaml:\"usage_in_kernelmode,omitempty\""��type.struct { PercpuUsage []uint64 "json:\"percpu_usage,omitempty\" yaml:\"percpu_usage,omitempty\""; UsageInUsermode uint64 "json:\"usage_in_usermode,omitempty\" yaml:\"usage_in_usermode,omitempty\""; TotalUsage uint64 "json:\"total_usage,omitempty\" yaml:\"total_usage,omitempty\""; UsageInKernelmode uint64 "json:\"usage_in_kernelmode,omitempty\" yaml:\"usage_in_kernelmode,omitempty\"" }��0K�� ($ � runtime.algarray0bruntime.gcbits.0x48444400000000000000000000000000P""..gostring.5p�go.weak.type.*struct { PercpuUsage []uint64 "json:\"percpu_usage,omitempty\" yaml:\"percpu_usage,omitempty\""; UsageInUsermode uint64 "json:\"usage_in_usermode,omitempty\" yaml:\"usage_in_usermode,omitempty\""; TotalUsage uint64 "json:\"total_usage,omitempty\" yaml:\"total_usage,omitempty\""; UsageInKernelmode uint64 "json:\"usage_in_kernelmode,omitempty\" yaml:\"usage_in_kernelmode,omitempty\"" }�"runtime.zerovalue���type.struct { PercpuUsage []uint64 "json:\"percpu_usage,omitempty\" yaml:\"percpu_usage,omitempty\""; UsageInUsermode uint64 "json:\"usage_in_usermode,omitempty\" yaml:\"usage_in_usermode,omitempty\""; TotalUsage uint64 "json:\"total_usage,omitempty\" yaml:\"total_usage,omitempty\""; UsageInKernelmode uint64 "json:\"usage_in_kernelmode,omitempty\" yaml:\"usage_in_kernelmode,omitempty\"" }�.go.string."PercpuUsage"�type.[]uint64��go.string."json:\"percpu_usage,omitempty\" yaml:\"percpu_usage,omitempty\""�6go.string."UsageInUsermode"�type.uint64��go.string."json:\"usage_in_usermode,omitempty\" yaml:\"usage_in_usermode,omitempty\""�,go.string."TotalUsage"�type.uint64��go.string."json:\"total_usage,omitempty\" yaml:\"total_usage,omitempty\""�:go.string."UsageInKernelmode"�type.uint64��go.string."json:\"usage_in_kernelmode,omitempty\" yaml:\"usage_in_kernelmode,omitempty\""�""..gostring.6���struct { Periods uint64 "json:\"periods,omitempty\""; ThrottledPeriods uint64 "json:\"throttled_periods,omitempty\""; ThrottledTime uint64 "json:\"throttled_time,omitempty\"" } ""..gostring.6�&go.string."Periods"00Periods &go.string."Periods"�Lgo.string."json:\"periods,omitempty\""`Rjson:"periods,omitempty" Lgo.string."json:\"periods,omitempty\""�8go.string."ThrottledPeriods"PBThrottledPeriods 8go.string."ThrottledPeriods"�`go.string."json:\"throttled_periods,omitempty\""pf"json:"throttled_periods,omitempty" `go.string."json:\"throttled_periods,omitempty\""�2go.string."ThrottledTime"@< ThrottledTime 2go.string."ThrottledTime"�Zgo.string."json:\"throttled_time,omitempty\""``json:"throttled_time,omitempty" Zgo.string."json:\"throttled_time,omitempty\""��type.struct { Periods uint64 "json:\"periods,omitempty\""; ThrottledPeriods uint64 "json:\"throttled_periods,omitempty\""; ThrottledTime uint64 "json:\"throttled_time,omitempty\"" }��k��o�  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P""..gostring.6p�go.weak.type.*struct { Periods uint64 "json:\"periods,omitempty\""; ThrottledPeriods uint64 "json:\"throttled_periods,omitempty\""; ThrottledTime uint64 "json:\"throttled_time,omitempty\"" }�"runtime.zerovalue���type.struct { Periods uint64 "json:\"periods,omitempty\""; ThrottledPeriods uint64 "json:\"throttled_periods,omitempty\""; ThrottledTime uint64 "json:\"throttled_time,omitempty\"" }�&go.string."Periods"�type.uint64�Lgo.string."json:\"periods,omitempty\""�8go.string."ThrottledPeriods"�type.uint64�`go.string."json:\"throttled_periods,omitempty\""�2go.string."ThrottledTime"�type.uint64�Zgo.string."json:\"throttled_time,omitempty\""�8go.string."*docker.CPUStats"PB*docker.CPUStats 8go.string."*docker.CPUStats"�"type.*"".CPUStats���S�%6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P8go.string."*docker.CPUStats"p4go.weak.type.**"".CPUStats�"runtime.zerovalue� type."".CPUStats�bruntime.gcbits.0x48444444440000000000000000000000 HDDDD�6go.string."docker.CPUStats"@@docker.CPUStats 6go.string."docker.CPUStats"�(go.string."CPUUsage"@2CPUUsage (go.string."CPUUsage"��go.string."json:\"cpu_usage,omitempty\" yaml:\"cpu_usage,omitempty\""��5json:"cpu_usage,omitempty" yaml:"cpu_usage,omitempty" �go.string."json:\"cpu_usage,omitempty\" yaml:\"cpu_usage,omitempty\""�4go.string."SystemCPUUsage"@>SystemCPUUsage 4go.string."SystemCPUUsage"��go.string."json:\"system_cpu_usage,omitempty\" yaml:\"system_cpu_usage,omitempty\""��Cjson:"system_cpu_usage,omitempty" yaml:"system_cpu_usage,omitempty" �go.string."json:\"system_cpu_usage,omitempty\" yaml:\"system_cpu_usage,omitempty\""�4go.string."ThrottlingData"@>ThrottlingData 4go.string."ThrottlingData"��go.string."json:\"throttling_data,omitempty\" yaml:\"throttling_data,omitempty\""��Ajson:"throttling_data,omitempty" yaml:"throttling_data,omitempty" �go.string."json:\"throttling_data,omitempty\" yaml:\"throttling_data,omitempty\""�(go.string."CPUStats"@2CPUStats (go.string."CPUStats"� type."".CPUStats��PF���08& � runtime.algarray0bruntime.gcbits.0x48444444440000000000000000000000P6go.string."docker.CPUStats"p"type.*"".CPUStats�"runtime.zerovalue�� type."".CPUStats�(go.string."CPUUsage"��type.struct { PercpuUsage []uint64 "json:\"percpu_usage,omitempty\" yaml:\"percpu_usage,omitempty\""; UsageInUsermode uint64 "json:\"usage_in_usermode,omitempty\" yaml:\"usage_in_usermode,omitempty\""; TotalUsage uint64 "json:\"total_usage,omitempty\" yaml:\"total_usage,omitempty\""; UsageInKernelmode uint64 "json:\"usage_in_kernelmode,omitempty\" yaml:\"usage_in_kernelmode,omitempty\"" }��go.string."json:\"cpu_usage,omitempty\" yaml:\"cpu_usage,omitempty\""�4go.string."SystemCPUUsage"�type.uint64��go.string."json:\"system_cpu_usage,omitempty\" yaml:\"system_cpu_usage,omitempty\""�4go.string."ThrottlingData"��type.struct { Periods uint64 "json:\"periods,omitempty\""; ThrottledPeriods uint64 "json:\"throttled_periods,omitempty\""; ThrottledTime uint64 "json:\"throttled_time,omitempty\"" }��go.string."json:\"throttling_data,omitempty\" yaml:\"throttling_data,omitempty\""`� type."".CPUStats�(go.string."CPUStats"�"go.importpath."".�� type."".CPUStats�,"type..gc."".StatsZ�*type..gcprog."".Stats22XeUUUUUUUUUU�eY�eYVUeUU�0go.string."docker.Stats"@: docker.Stats 0go.string."docker.Stats"� go.string."Read"0*Read go.string."Read"�vgo.string."json:\"read,omitempty\" yaml:\"read,omitempty\""�x+json:"read,omitempty" yaml:"read,omitempty" vgo.string."json:\"read,omitempty\" yaml:\"read,omitempty\""��go.string."json:\"network,omitempty\" yaml:\"network,omitempty\""��1json:"network,omitempty" yaml:"network,omitempty" �go.string."json:\"network,omitempty\" yaml:\"network,omitempty\""�.go.string."MemoryStats"@8 MemoryStats .go.string."MemoryStats"��go.string."json:\"memory_stats,omitempty\" yaml:\"memory_stats,omitempty\""��;json:"memory_stats,omitempty" yaml:"memory_stats,omitempty" �go.string."json:\"memory_stats,omitempty\" yaml:\"memory_stats,omitempty\""�,go.string."BlkioStats"@6 +BlkioStats ,go.string."BlkioStats"��go.string."json:\"blkio_stats,omitempty\" yaml:\"blkio_stats,omitempty\""��9json:"blkio_stats,omitempty" yaml:"blkio_stats,omitempty" �go.string."json:\"blkio_stats,omitempty\" yaml:\"blkio_stats,omitempty\""��go.string."json:\"cpu_stats,omitempty\" yaml:\"cpu_stats,omitempty\""��5json:"cpu_stats,omitempty" yaml:"cpu_stats,omitempty" �go.string."json:\"cpu_stats,omitempty\" yaml:\"cpu_stats,omitempty\""�.go.string."PreCPUStats"@8 PreCPUStats .go.string."PreCPUStats"�Vgo.string."json:\"precpu_stats,omitempty\""`\json:"precpu_stats,omitempty" Vgo.string."json:\"precpu_stats,omitempty\""�type."".Stats����9�YX` p: � runtime.algarray0"type..gc."".Stats@*type..gcprog."".StatsP0go.string."docker.Stats"ptype.*"".Stats�"runtime.zerovalue��type."".Stats� go.string."Read"�type.time.Time�vgo.string."json:\"read,omitempty\" yaml:\"read,omitempty\""�&go.string."Network"�� type.struct { RxDropped uint64 "json:\"rx_dropped,omitempty\" yaml:\"rx_dropped,omitempty\""; RxBytes uint64 "json:\"rx_bytes,omitempty\" yaml:\"rx_bytes,omitempty\""; RxErrors uint64 "json:\"rx_errors,omitempty\" yaml:\"rx_errors,omitempty\""; TxPackets uint64 "json:\"tx_packets,omitempty\" yaml:\"tx_packets,omitempty\""; TxDropped uint64 "json:\"tx_dropped,omitempty\" yaml:\"tx_dropped,omitempty\""; RxPackets uint64 "json:\"rx_packets,omitempty\" yaml:\"rx_packets,omitempty\""; TxErrors uint64 "json:\"tx_errors,omitempty\" yaml:\"tx_errors,omitempty\""; TxBytes uint64 "json:\"tx_bytes,omitempty\" yaml:\"tx_bytes,omitempty\"" }��go.string."json:\"network,omitempty\" yaml:\"network,omitempty\""�.go.string."MemoryStats"��-type.struct { Stats struct { TotalPgmafault uint64 "json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""; Cache uint64 "json:\"cache,omitempty\" yaml:\"cache,omitempty\""; MappedFile uint64 "json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""; TotalInactiveFile uint64 "json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""; Pgpgout uint64 "json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""; Rss uint64 "json:\"rss,omitempty\" yaml:\"rss,omitempty\""; TotalMappedFile uint64 "json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""; Writeback uint64 "json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""; Unevictable uint64 "json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""; Pgpgin uint64 "json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""; TotalUnevictable uint64 "json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""; Pgmajfault uint64 "json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""; TotalRss uint64 "json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""; TotalRssHuge uint64 "json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""; TotalWriteback uint64 "json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""; TotalInactiveAnon uint64 "json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""; RssHuge uint64 "json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""; HierarchicalMemoryLimit uint64 "json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""; TotalPgfault uint64 "json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""; TotalActiveFile uint64 "json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""; ActiveAnon uint64 "json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""; TotalActiveAnon uint64 "json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""; TotalPgpgout uint64 "json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""; TotalCache uint64 "json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""; InactiveAnon uint64 "json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""; ActiveFile uint64 "json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""; Pgfault uint64 "json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""; InactiveFile uint64 "json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""; TotalPgpgin uint64 "json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\"" } "json:\"stats,omitempty\" yaml:\"stats,omitempty\""; MaxUsage uint64 "json:\"max_usage,omitempty\" yaml:\"max_usage,omitempty\""; Usage uint64 "json:\"usage,omitempty\" yaml:\"usage,omitempty\""; Failcnt uint64 "json:\"failcnt,omitempty\" yaml:\"failcnt,omitempty\""; Limit uint64 "json:\"limit,omitempty\" yaml:\"limit,omitempty\"" }��go.string."json:\"memory_stats,omitempty\" yaml:\"memory_stats,omitempty\""�,go.string."BlkioStats"��type.struct { IOServiceBytesRecursive []"".BlkioStatsEntry "json:\"io_service_bytes_recursive,omitempty\" yaml:\"io_service_bytes_recursive,omitempty\""; IOServicedRecursive []"".BlkioStatsEntry "json:\"io_serviced_recursive,omitempty\" yaml:\"io_serviced_recursive,omitempty\""; IOQueueRecursive []"".BlkioStatsEntry "json:\"io_queue_recursive,omitempty\" yaml:\"io_queue_recursive,omitempty\""; IOServiceTimeRecursive []"".BlkioStatsEntry "json:\"io_service_time_recursive,omitempty\" yaml:\"io_service_time_recursive,omitempty\""; IOWaitTimeRecursive []"".BlkioStatsEntry "json:\"io_wait_time_recursive,omitempty\" yaml:\"io_wait_time_recursive,omitempty\""; IOMergedRecursive []"".BlkioStatsEntry "json:\"io_merged_recursive,omitempty\" yaml:\"io_merged_recursive,omitempty\""; IOTimeRecursive []"".BlkioStatsEntry "json:\"io_time_recursive,omitempty\" yaml:\"io_time_recursive,omitempty\""; SectorsRecursive []"".BlkioStatsEntry "json:\"sectors_recursive,omitempty\" yaml:\"sectors_recursive,omitempty\"" }��go.string."json:\"blkio_stats,omitempty\" yaml:\"blkio_stats,omitempty\""�(go.string."CPUStats"� type."".CPUStats��go.string."json:\"cpu_stats,omitempty\" yaml:\"cpu_stats,omitempty\""�.go.string."PreCPUStats"� type."".CPUStats�Vgo.string."json:\"precpu_stats,omitempty\""`�type."".Stats�"go.string."Stats"�"go.importpath."".��type."".Stats�2go.string."*docker.Stats"@< *docker.Stats 2go.string."*docker.Stats"�type.*"".Stats����g6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P2go.string."*docker.Stats"p.go.weak.type.**"".Stats�"runtime.zerovalue�type."".Stats�@go.string."chan<- *docker.Stats"PJchan<- *docker.Stats @go.string."chan<- *docker.Stats"�*type.chan<- *"".Stats���bB2 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P@go.string."chan<- *docker.Stats"pgo.string."docker.StatsOptions"PHdocker.StatsOptions >go.string."docker.StatsOptions"�&go.string."Timeout"00Timeout &go.string."Timeout"�0go.string."StatsOptions"@: StatsOptions 0go.string."StatsOptions"�(type."".StatsOptions��0� �� (( 2type..alg."".StatsOptions0bruntime.gcbits.0x48484800000000000000000000000000P>go.string."docker.StatsOptions"p*type.*"".StatsOptions�"runtime.zerovalue��(type."".StatsOptions�go.string."ID"�type.string�"go.string."Stats"�*type.chan<- *"".Stats�$go.string."Stream"�type.bool� go.string."Done"� type.<-chan bool�&go.string."Timeout"�$type.time.Duration`�(type."".StatsOptions�0go.string."StatsOptions"�"go.importpath."".��(type."".StatsOptions�vgo.string."func(*docker.Client, docker.StatsOptions) error"��/func(*docker.Client, docker.StatsOptions) error vgo.string."func(*docker.Client, docker.StatsOptions) error"�Xtype.func(*"".Client, "".StatsOptions) error���VA3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pvgo.string."func(*docker.Client, docker.StatsOptions) error"pjgo.weak.type.*func(*"".Client, "".StatsOptions) error�"runtime.zerovalue��Xtype.func(*"".Client, "".StatsOptions) error��Xtype.func(*"".Client, "".StatsOptions) error�type.*"".Client�(type."".StatsOptions�type.error�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�Btype..hashfunc."".TagImageOptions:type..hash."".TagImageOptions�>type..eqfunc."".TagImageOptions6type..eq."".TagImageOptions�8type..alg."".TagImageOptions Btype..hashfunc."".TagImageOptions>type..eqfunc."".TagImageOptions�Fgo.string."*docker.TagImageOptions"PP*docker.TagImageOptions Fgo.string."*docker.TagImageOptions"�0type.*"".TagImageOptions����� 6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PFgo.string."*docker.TagImageOptions"pBgo.weak.type.**"".TagImageOptions�"runtime.zerovalue�.type."".TagImageOptions�bruntime.gcbits.0x48488484440000000000000000000000 HH��D�Dgo.string."docker.TagImageOptions"PNdocker.TagImageOptions Dgo.string."docker.TagImageOptions"� go.string."Repo"0*Repo go.string."Repo"�6go.string."TagImageOptions"@@TagImageOptions 6go.string."TagImageOptions"�.type."".TagImageOptions��(��# 8type..alg."".TagImageOptions0bruntime.gcbits.0x48488484440000000000000000000000PDgo.string."docker.TagImageOptions"p0type.*"".TagImageOptions�"runtime.zerovalue��.type."".TagImageOptions� go.string."Repo"�type.string�go.string."Tag"�type.string�"go.string."Force"�type.bool`�.type."".TagImageOptions�6go.string."TagImageOptions"�"go.importpath."".��.type."".TagImageOptions��go.string."func(*docker.Client, string, docker.TagImageOptions) error"��:func(*docker.Client, string, docker.TagImageOptions) error �go.string."func(*docker.Client, string, docker.TagImageOptions) error"�ntype.func(*"".Client, string, "".TagImageOptions) error���,�.3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, string, docker.TagImageOptions) error"p�go.weak.type.*func(*"".Client, string, "".TagImageOptions) error�"runtime.zerovalue��ntype.func(*"".Client, string, "".TagImageOptions) error��ntype.func(*"".Client, string, "".TagImageOptions) error�type.*"".Client�type.string�.type."".TagImageOptions�type.error�:go.string."*docker.TopResult"PD*docker.TopResult :go.string."*docker.TopResult"�$type.*"".TopResult���g6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."*docker.TopResult"p6go.weak.type.**"".TopResult�"runtime.zerovalue�"type."".TopResult�8go.string."docker.TopResult"PBdocker.TopResult 8go.string."docker.TopResult"�$go.string."Titles"0.Titles $go.string."Titles"�*go.string."Processes"@4 Processes *go.string."Processes"�*go.string."TopResult"@4 TopResult *go.string."TopResult"�"type."".TopResult��0�O�� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P8go.string."docker.TopResult"p$type.*"".TopResult�"runtime.zerovalue��"type."".TopResult�$go.string."Titles"�type.[]string�*go.string."Processes"�type.[][]string`�"type."".TopResult�*go.string."TopResult"�"go.importpath."".��"type."".TopResult��go.string."func(*docker.Client, string, string) (docker.TopResult, error)"��>func(*docker.Client, string, string) (docker.TopResult, error) �go.string."func(*docker.Client, string, string) (docker.TopResult, error)"�vtype.func(*"".Client, string, string) ("".TopResult, error)��8���3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, string, string) (docker.TopResult, error)"p�go.weak.type.*func(*"".Client, string, string) ("".TopResult, error)�"runtime.zerovalue��vtype.func(*"".Client, string, string) ("".TopResult, error)��vtype.func(*"".Client, string, string) ("".TopResult, error)�type.*"".Client�type.string�type.string�"type."".TopResult�type.error�jgo.string."func(*docker.Client, string) (int, error)"�t)func(*docker.Client, string) (int, error) jgo.string."func(*docker.Client, string) (int, error)"�Ttype.func(*"".Client, string) (int, error)����13 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pjgo.string."func(*docker.Client, string) (int, error)"pfgo.weak.type.*func(*"".Client, string) (int, error)�"runtime.zerovalue��Ttype.func(*"".Client, string) (int, error)��Ttype.func(*"".Client, string) (int, error)�type.*"".Client�type.string�type.int�type.error��go.string."func(*docker.Client, string, map[string]string, io.Reader, io.Writer, bool) error"��Qfunc(*docker.Client, string, map[string]string, io.Reader, io.Writer, bool) error �go.string."func(*docker.Client, string, map[string]string, io.Reader, io.Writer, bool) error"��type.func(*"".Client, string, map[string]string, io.Reader, io.Writer, bool) error��|4�Y3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, string, map[string]string, io.Reader, io.Writer, bool) error"p�go.weak.type.*func(*"".Client, string, map[string]string, io.Reader, io.Writer, bool) error�"runtime.zerovalue���type.func(*"".Client, string, map[string]string, io.Reader, io.Writer, bool) error���type.func(*"".Client, string, map[string]string, io.Reader, io.Writer, bool) error�type.*"".Client�type.string�,type.map[string]string�type.io.Reader�type.io.Writer�type.bool�type.error�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�6type..hashfunc."".doOptions.type..hash."".doOptions�2type..eqfunc."".doOptions*type..eq."".doOptions�,type..alg."".doOptions 6type..hashfunc."".doOptions2type..eqfunc."".doOptions�:go.string."*docker.doOptions"PD*docker.doOptions :go.string."*docker.doOptions"�$type.*"".doOptions���1z6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."*docker.doOptions"p6go.weak.type.**"".doOptions�"runtime.zerovalue�"type."".doOptions�bruntime.gcbits.0xccc44c00000000000000000000000000 ��L�8go.string."docker.doOptions"PBdocker.doOptions 8go.string."docker.doOptions"� go.string."data"0*data go.string."data"�*go.string."forceJSON"@4 forceJSON *go.string."forceJSON"�*go.string."doOptions"@4 doOptions *go.string."doOptions"�"type."".doOptions��?� ,type..alg."".doOptions0bruntime.gcbits.0xccc44c00000000000000000000000000P8go.string."docker.doOptions"p$type.*"".doOptions�"runtime.zerovalue��"type."".doOptions� go.string."data"�"go.importpath."".�"type.interface {}�*go.string."forceJSON"�"go.importpath."".�type.bool`�"type."".doOptions�*go.string."doOptions"�"go.importpath."".��"type."".doOptions��go.string."func(*docker.Client, string, string, docker.doOptions) ([]uint8, int, error)"��Lfunc(*docker.Client, string, string, docker.doOptions) ([]uint8, int, error) �go.string."func(*docker.Client, string, string, docker.doOptions) ([]uint8, int, error)"��type.func(*"".Client, string, string, "".doOptions) ([]uint8, int, error)����.3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, string, string, docker.doOptions) ([]uint8, int, error)"p�go.weak.type.*func(*"".Client, string, string, "".doOptions) ([]uint8, int, error)�"runtime.zerovalue���type.func(*"".Client, string, string, "".doOptions) ([]uint8, int, error)���type.func(*"".Client, string, string, "".doOptions) ([]uint8, int, error)�type.*"".Client�type.string�type.string�"type."".doOptions�type.[]uint8�type.int�type.error��go.string."func(*docker.Client, int64, chan *docker.APIEvents, chan error) error"��Efunc(*docker.Client, int64, chan *docker.APIEvents, chan error) error �go.string."func(*docker.Client, int64, chan *docker.APIEvents, chan error) error"��type.func(*"".Client, int64, chan *"".APIEvents, chan error) error���8[53 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, int64, chan *docker.APIEvents, chan error) error"p�go.weak.type.*func(*"".Client, int64, chan *"".APIEvents, chan error) error�"runtime.zerovalue���type.func(*"".Client, int64, chan *"".APIEvents, chan error) error���type.func(*"".Client, int64, chan *"".APIEvents, chan error) error�type.*"".Client�type.int64�.type.chan *"".APIEvents�type.chan error�type.error�`go.string."func(*docker.Client) (string, error)"pj$func(*docker.Client) (string, error) `go.string."func(*docker.Client) (string, error)"�Jtype.func(*"".Client) (string, error)���� G3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P`go.string."func(*docker.Client) (string, error)"p\go.weak.type.*func(*"".Client) (string, error)�"runtime.zerovalue��Jtype.func(*"".Client) (string, error)��Jtype.func(*"".Client) (string, error)�type.*"".Client�type.string�type.error�^go.string."func(*docker.Client, string) string"ph#func(*docker.Client, string) string ^go.string."func(*docker.Client, string) string"�Htype.func(*"".Client, string) string��Eh��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P^go.string."func(*docker.Client, string) string"pZgo.weak.type.*func(*"".Client, string) string�"runtime.zerovalue��Htype.func(*"".Client, string) string��Htype.func(*"".Client, string) string�type.*"".Client�type.string�type.string�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·af3107c17ee1ab6f9f33230b5c7e3062�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�>type..hashfunc."".hijackOptions6type..hash."".hijackOptions�:type..eqfunc."".hijackOptions2type..eq."".hijackOptions�4type..alg."".hijackOptions >type..hashfunc."".hijackOptions:type..eqfunc."".hijackOptions�Bgo.string."*docker.hijackOptions"PL*docker.hijackOptions Bgo.string."*docker.hijackOptions"�,type.*"".hijackOptions���d�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."*docker.hijackOptions"p>go.weak.type.**"".hijackOptions�"runtime.zerovalue�*type."".hijackOptions�bruntime.gcbits.0x488c8c8ccc0000000000000000000000 H�����@go.string."docker.hijackOptions"PJdocker.hijackOptions @go.string."docker.hijackOptions"�&go.string."success"00success &go.string."success"�4go.string."setRawTerminal"@>setRawTerminal 4go.string."setRawTerminal"�go.string."in"0&in go.string."in"�$go.string."stdout"0.stdout $go.string."stdout"�$go.string."stderr"0.stderr $go.string."stderr"�2go.string."hijackOptions"@< hijackOptions 2go.string."hijackOptions"�*type."".hijackOptions��P��� 0@8 4type..alg."".hijackOptions0bruntime.gcbits.0x488c8c8ccc0000000000000000000000P@go.string."docker.hijackOptions"p,type.*"".hijackOptions�"runtime.zerovalue��*type."".hijackOptions�&go.string."success"�"go.importpath."".�&type.chan struct {}�4go.string."setRawTerminal"�"go.importpath."".�type.bool�go.string."in"�"go.importpath."".�type.io.Reader�$go.string."stdout"�"go.importpath."".�type.io.Writer�$go.string."stderr"�"go.importpath."".�type.io.Writer� go.string."data"�"go.importpath."".�"type.interface {}`�*type."".hijackOptions�2go.string."hijackOptions"�"go.importpath."".��*type."".hijackOptions��go.string."func(*docker.Client, string, string, docker.hijackOptions) error"��@func(*docker.Client, string, string, docker.hijackOptions) error �go.string."func(*docker.Client, string, string, docker.hijackOptions) error"�ztype.func(*"".Client, string, string, "".hijackOptions) error���+�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, string, string, docker.hijackOptions) error"p�go.weak.type.*func(*"".Client, string, string, "".hijackOptions) error�"runtime.zerovalue��ztype.func(*"".Client, string, string, "".hijackOptions) error��ztype.func(*"".Client, string, string, "".hijackOptions) error�type.*"".Client�type.string�type.string�*type."".hijackOptions�type.error�Bgo.string."*docker.streamOptions"PL*docker.streamOptions Bgo.string."*docker.streamOptions"�,type.*"".streamOptions��Մ�46 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."*docker.streamOptions"p>go.weak.type.**"".streamOptions�"runtime.zerovalue�*type."".streamOptions�bruntime.gcbits.0x848c8c8c44c8c8c84800000000000000 ����D���H�@go.string."docker.streamOptions"PJdocker.streamOptions @go.string."docker.streamOptions"�2go.string."rawJSONStream"@< rawJSONStream 2go.string."rawJSONStream"�4go.string."useJSONDecoder"@>useJSONDecoder 4go.string."useJSONDecoder"�&go.string."headers"00headers &go.string."headers"�&go.string."timeout"00timeout &go.string."timeout"�2go.string."streamOptions"@< streamOptions 2go.string."streamOptions"�*type."".streamOptions��H�Ê\ 0@D � runtime.algarray0bruntime.gcbits.0x848c8c8c44c8c8c84800000000000000P@go.string."docker.streamOptions"p,type.*"".streamOptions�"runtime.zerovalue��*type."".streamOptions�4go.string."setRawTerminal"�"go.importpath."".�type.bool�2go.string."rawJSONStream"�"go.importpath."".�type.bool�4go.string."useJSONDecoder"�"go.importpath."".�type.bool�&go.string."headers"�"go.importpath."".�,type.map[string]string�go.string."in"�"go.importpath."".�type.io.Reader�$go.string."stdout"�"go.importpath."".�type.io.Writer�$go.string."stderr"�"go.importpath."".�type.io.Writer�&go.string."timeout"�"go.importpath."".�$type.time.Duration`�*type."".streamOptions�2go.string."streamOptions"�"go.importpath."".��*type."".streamOptions��go.string."func(*docker.Client, string, string, docker.streamOptions) error"��@func(*docker.Client, string, string, docker.streamOptions) error �go.string."func(*docker.Client, string, string, docker.streamOptions) error"�ztype.func(*"".Client, string, string, "".streamOptions) error���� 3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.Client, string, string, docker.streamOptions) error"p�go.weak.type.*func(*"".Client, string, string, "".streamOptions) error�"runtime.zerovalue��ztype.func(*"".Client, string, string, "".streamOptions) error��ztype.func(*"".Client, string, string, "".streamOptions) error�type.*"".Client�type.string�type.string�*type."".streamOptions�type.error�8go.string."AddEventListener"PBAddEventListener 8go.string."AddEventListener"�:go.string."AttachToContainer"PDAttachToContainer :go.string."AttachToContainer"�ngo.string."func(docker.AttachToContainerOptions) error"�x+func(docker.AttachToContainerOptions) error ngo.string."func(docker.AttachToContainerOptions) error"�Xtype.func("".AttachToContainerOptions) error�� r�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pngo.string."func(docker.AttachToContainerOptions) error"pjgo.weak.type.*func("".AttachToContainerOptions) error�"runtime.zerovalue��Xtype.func("".AttachToContainerOptions) error��Xtype.func("".AttachToContainerOptions) error�@type."".AttachToContainerOptions�type.error�*go.string."AuthCheck"@4 AuthCheck *go.string."AuthCheck"�bgo.string."func(*docker.AuthConfiguration) error"pl%func(*docker.AuthConfiguration) error bgo.string."func(*docker.AuthConfiguration) error"�Ltype.func(*"".AuthConfiguration) error��QKï3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pbgo.string."func(*docker.AuthConfiguration) error"p^go.weak.type.*func(*"".AuthConfiguration) error�"runtime.zerovalue��Ltype.func(*"".AuthConfiguration) error��Ltype.func(*"".AuthConfiguration) error�4type.*"".AuthConfiguration�type.error�,go.string."BuildImage"@6 +BuildImage ,go.string."BuildImage"�`go.string."func(docker.BuildImageOptions) error"pj$func(docker.BuildImageOptions) error `go.string."func(docker.BuildImageOptions) error"�Jtype.func("".BuildImageOptions) error���e�z3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P`go.string."func(docker.BuildImageOptions) error"p\go.weak.type.*func("".BuildImageOptions) error�"runtime.zerovalue��Jtype.func("".BuildImageOptions) error��Jtype.func("".BuildImageOptions) error�2type."".BuildImageOptions�type.error�6go.string."CommitContainer"@@CommitContainer 6go.string."CommitContainer"��go.string."func(docker.CommitContainerOptions) (*docker.Image, error)"��:func(docker.CommitContainerOptions) (*docker.Image, error) �go.string."func(docker.CommitContainerOptions) (*docker.Image, error)"�ntype.func("".CommitContainerOptions) (*"".Image, error)�����%3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(docker.CommitContainerOptions) (*docker.Image, error)"p�go.weak.type.*func("".CommitContainerOptions) (*"".Image, error)�"runtime.zerovalue��ntype.func("".CommitContainerOptions) (*"".Image, error)��ntype.func("".CommitContainerOptions) (*"".Image, error)�func(docker.CreateContainerOptions) (*docker.Container, error) �go.string."func(docker.CreateContainerOptions) (*docker.Container, error)"�vtype.func("".CreateContainerOptions) (*"".Container, error)��#� R3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(docker.CreateContainerOptions) (*docker.Container, error)"p�go.weak.type.*func("".CreateContainerOptions) (*"".Container, error)�"runtime.zerovalue��vtype.func("".CreateContainerOptions) (*"".Container, error)��vtype.func("".CreateContainerOptions) (*"".Container, error)�ListContainers 4go.string."ListContainers"��go.string."func(docker.ListContainersOptions) ([]docker.APIContainers, error)"��Bfunc(docker.ListContainersOptions) ([]docker.APIContainers, error) �go.string."func(docker.ListContainersOptions) ([]docker.APIContainers, error)"�~type.func("".ListContainersOptions) ([]"".APIContainers, error)��]��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(docker.ListContainersOptions) ([]docker.APIContainers, error)"p�go.weak.type.*func("".ListContainersOptions) ([]"".APIContainers, error)�"runtime.zerovalue��~type.func("".ListContainersOptions) ([]"".APIContainers, error)��~type.func("".ListContainersOptions) ([]"".APIContainers, error)�:type."".ListContainersOptions�.type.[]"".APIContainers�type.error�,go.string."ListImages"@6 +ListImages ,go.string."ListImages"��go.string."func(docker.ListImagesOptions) ([]docker.APIImages, error)"��:func(docker.ListImagesOptions) ([]docker.APIImages, error) �go.string."func(docker.ListImagesOptions) ([]docker.APIImages, error)"�ntype.func("".ListImagesOptions) ([]"".APIImages, error)����8�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(docker.ListImagesOptions) ([]docker.APIImages, error)"p�go.weak.type.*func("".ListImagesOptions) ([]"".APIImages, error)�"runtime.zerovalue��ntype.func("".ListImagesOptions) ([]"".APIImages, error)��ntype.func("".ListImagesOptions) ([]"".APIImages, error)�2type."".ListImagesOptions�&type.[]"".APIImages�type.error�0go.string."ListNetworks"@: ListNetworks 0go.string."ListNetworks"�Xgo.string."func() ([]docker.Network, error)"pb func() ([]docker.Network, error) Xgo.string."func() ([]docker.Network, error)"�Btype.func() ([]"".Network, error)���\�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PXgo.string."func() ([]docker.Network, error)"pTgo.weak.type.*func() ([]"".Network, error)�"runtime.zerovalue��Btype.func() ([]"".Network, error)��Btype.func() ([]"".Network, error)�"type.[]"".Network�type.error�*go.string."LoadImage"@4 LoadImage *go.string."LoadImage"�^go.string."func(docker.LoadImageOptions) error"ph#func(docker.LoadImageOptions) error ^go.string."func(docker.LoadImageOptions) error"�Htype.func("".LoadImageOptions) error���=�V3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P^go.string."func(docker.LoadImageOptions) error"pZgo.weak.type.*func("".LoadImageOptions) error�"runtime.zerovalue��Htype.func("".LoadImageOptions) error��Htype.func("".LoadImageOptions) error�0type."".LoadImageOptions�type.error�Tgo.string."func(docker.LogsOptions) error"`^func(docker.LogsOptions) error Tgo.string."func(docker.LogsOptions) error"�>type.func("".LogsOptions) error��F���3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PTgo.string."func(docker.LogsOptions) error"pPgo.weak.type.*func("".LogsOptions) error�"runtime.zerovalue��>type.func("".LogsOptions) error��>type.func("".LogsOptions) error�&type."".LogsOptions�type.error�.go.string."NetworkInfo"@8 NetworkInfo .go.string."NetworkInfo"�bgo.string."func(string) (*docker.Network, error)"pl%func(string) (*docker.Network, error) bgo.string."func(string) (*docker.Network, error)"�Ltype.func(string) (*"".Network, error)��^��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pbgo.string."func(string) (*docker.Network, error)"p^go.weak.type.*func(string) (*"".Network, error)�"runtime.zerovalue��Ltype.func(string) (*"".Network, error)��Ltype.func(string) (*"".Network, error)�type.string� type.*"".Network�type.error�4go.string."PauseContainer"@>PauseContainer 4go.string."PauseContainer"�go.string."RemoveEventListener"PHRemoveEventListener >go.string."RemoveEventListener"�\go.string."func(chan *docker.APIEvents) error"pf"func(chan *docker.APIEvents) error \go.string."func(chan *docker.APIEvents) error"�Ftype.func(chan *"".APIEvents) error��h9!�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P\go.string."func(chan *docker.APIEvents) error"pXgo.weak.type.*func(chan *"".APIEvents) error�"runtime.zerovalue��Ftype.func(chan *"".APIEvents) error��Ftype.func(chan *"".APIEvents) error�.type.chan *"".APIEvents�type.error�.go.string."RemoveImage"@8 RemoveImage .go.string."RemoveImage"�>go.string."RemoveImageExtended"PHRemoveImageExtended >go.string."RemoveImageExtended"�rgo.string."func(string, docker.RemoveImageOptions) error"�|-func(string, docker.RemoveImageOptions) error rgo.string."func(string, docker.RemoveImageOptions) error"�\type.func(string, "".RemoveImageOptions) error����r^3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Prgo.string."func(string, docker.RemoveImageOptions) error"pngo.weak.type.*func(string, "".RemoveImageOptions) error�"runtime.zerovalue��\type.func(string, "".RemoveImageOptions) error��\type.func(string, "".RemoveImageOptions) error�type.string�4type."".RemoveImageOptions�type.error�6go.string."RenameContainer"@@RenameContainer 6go.string."RenameContainer"�jgo.string."func(docker.RenameContainerOptions) error"�t)func(docker.RenameContainerOptions) error jgo.string."func(docker.RenameContainerOptions) error"�Ttype.func("".RenameContainerOptions) error������3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pjgo.string."func(docker.RenameContainerOptions) error"pfgo.weak.type.*func("".RenameContainerOptions) error�"runtime.zerovalue��Ttype.func("".RenameContainerOptions) error��Ttype.func("".RenameContainerOptions) error�StartContainer 4go.string."StartContainer"�dgo.string."func(string, *docker.HostConfig) error"pn&func(string, *docker.HostConfig) error dgo.string."func(string, *docker.HostConfig) error"�Ntype.func(string, *"".HostConfig) error���9�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pdgo.string."func(string, *docker.HostConfig) error"p`go.weak.type.*func(string, *"".HostConfig) error�"runtime.zerovalue��Ntype.func(string, *"".HostConfig) error��Ntype.func(string, *"".HostConfig) error�type.string�&type.*"".HostConfig�type.error�*go.string."StartExec"@4 StartExec *go.string."StartExec"�ngo.string."func(string, docker.StartExecOptions) error"�x+func(string, docker.StartExecOptions) error ngo.string."func(string, docker.StartExecOptions) error"�Xtype.func(string, "".StartExecOptions) error�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pngo.string."func(string, docker.StartExecOptions) error"pjgo.weak.type.*func(string, "".StartExecOptions) error�"runtime.zerovalue��Xtype.func(string, "".StartExecOptions) error��Xtype.func(string, "".StartExecOptions) error�type.string�0type."".StartExecOptions�type.error�Vgo.string."func(docker.StatsOptions) error"``func(docker.StatsOptions) error Vgo.string."func(docker.StatsOptions) error"�@type.func("".StatsOptions) error���u +3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PVgo.string."func(docker.StatsOptions) error"pRgo.weak.type.*func("".StatsOptions) error�"runtime.zerovalue��@type.func("".StatsOptions) error��@type.func("".StatsOptions) error�(type."".StatsOptions�type.error�2go.string."StopContainer"@< StopContainer 2go.string."StopContainer"�(go.string."TagImage"@2TagImage (go.string."TagImage"�lgo.string."func(string, docker.TagImageOptions) error"�v*func(string, docker.TagImageOptions) error lgo.string."func(string, docker.TagImageOptions) error"�Vtype.func(string, "".TagImageOptions) error��y�M3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Plgo.string."func(string, docker.TagImageOptions) error"phgo.weak.type.*func(string, "".TagImageOptions) error�"runtime.zerovalue��Vtype.func(string, "".TagImageOptions) error��Vtype.func(string, "".TagImageOptions) error�type.string�.type."".TagImageOptions�type.error�0go.string."TopContainer"@: TopContainer 0go.string."TopContainer"�tgo.string."func(string, string) (docker.TopResult, error)"�~.func(string, string) (docker.TopResult, error) tgo.string."func(string, string) (docker.TopResult, error)"�^type.func(string, string) ("".TopResult, error)�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ptgo.string."func(string, string) (docker.TopResult, error)"ppgo.weak.type.*func(string, string) ("".TopResult, error)�"runtime.zerovalue��^type.func(string, string) ("".TopResult, error)��^type.func(string, string) ("".TopResult, error)�type.string�type.string�"type."".TopResult�type.error�8go.string."UnpauseContainer"PBUnpauseContainer 8go.string."UnpauseContainer"�&go.string."Version"00Version &go.string."Version"�2go.string."WaitContainer"@< WaitContainer 2go.string."WaitContainer"�Jgo.string."func(string) (int, error)"`Tfunc(string) (int, error) Jgo.string."func(string) (int, error)"�type.func("".LogsOptions) error�Vtype.func(*"".Client, "".LogsOptions) error�""".(*Client).Logs�""".(*Client).Logs�.go.string."NetworkInfo"�Ltype.func(string) (*"".Network, error)�dtype.func(*"".Client, string) (*"".Network, error)�0"".(*Client).NetworkInfo�0"".(*Client).NetworkInfo�4go.string."PauseContainer"�.type.func(string) error�Ftype.func(*"".Client, string) error�6"".(*Client).PauseContainer�6"".(*Client).PauseContainer� go.string."Ping"�"type.func() error�6type.func(*"".Client) error�""".(*Client).Ping�""".(*Client).Ping�*go.string."PullImage"�ttype.func("".PullImageOptions, "".AuthConfiguration) error��type.func(*"".Client, "".PullImageOptions, "".AuthConfiguration) error�,"".(*Client).PullImage�,"".(*Client).PullImage�*go.string."PushImage"�ttype.func("".PushImageOptions, "".AuthConfiguration) error��type.func(*"".Client, "".PushImageOptions, "".AuthConfiguration) error�,"".(*Client).PushImage�,"".(*Client).PushImage�6go.string."RemoveContainer"�Ttype.func("".RemoveContainerOptions) error�ltype.func(*"".Client, "".RemoveContainerOptions) error�8"".(*Client).RemoveContainer�8"".(*Client).RemoveContainer�>go.string."RemoveEventListener"�Ftype.func(chan *"".APIEvents) error�^type.func(*"".Client, chan *"".APIEvents) error�@"".(*Client).RemoveEventListener�@"".(*Client).RemoveEventListener�.go.string."RemoveImage"�.type.func(string) error�Ftype.func(*"".Client, string) error�0"".(*Client).RemoveImage�0"".(*Client).RemoveImage�>go.string."RemoveImageExtended"�\type.func(string, "".RemoveImageOptions) error�ttype.func(*"".Client, string, "".RemoveImageOptions) error�@"".(*Client).RemoveImageExtended�@"".(*Client).RemoveImageExtended�6go.string."RenameContainer"�Ttype.func("".RenameContainerOptions) error�ltype.func(*"".Client, "".RenameContainerOptions) error�8"".(*Client).RenameContainer�8"".(*Client).RenameContainer�"".(*Client).ResizeContainerTTY�>"".(*Client).ResizeContainerTTY�2go.string."ResizeExecTTY"�Btype.func(string, int, int) error�Ztype.func(*"".Client, string, int, int) error�4"".(*Client).ResizeExecTTY�4"".(*Client).ResizeExecTTY�8go.string."RestartContainer"�:type.func(string, uint) error�Rtype.func(*"".Client, string, uint) error�:"".(*Client).RestartContainer�:"".(*Client).RestartContainer�0go.string."SearchImages"�\type.func(string) ([]"".APIImageSearch, error)�ttype.func(*"".Client, string) ([]"".APIImageSearch, error)�2"".(*Client).SearchImages�2"".(*Client).SearchImages�4go.string."StartContainer"�Ntype.func(string, *"".HostConfig) error�ftype.func(*"".Client, string, *"".HostConfig) error�6"".(*Client).StartContainer�6"".(*Client).StartContainer�*go.string."StartExec"� Xtype.func(string, "".StartExecOptions) error� ptype.func(*"".Client, string, "".StartExecOptions) error� ,"".(*Client).StartExec� ,"".(*Client).StartExec� "go.string."Stats"� @type.func("".StatsOptions) error�!Xtype.func(*"".Client, "".StatsOptions) error�!$"".(*Client).Stats�!$"".(*Client).Stats�!2go.string."StopContainer"�!:type.func(string, uint) error�!Rtype.func(*"".Client, string, uint) error�!4"".(*Client).StopContainer�"4"".(*Client).StopContainer�"(go.string."TagImage"�"Vtype.func(string, "".TagImageOptions) error�"ntype.func(*"".Client, string, "".TagImageOptions) error�"*"".(*Client).TagImage�"*"".(*Client).TagImage�"0go.string."TopContainer"�#^type.func(string, string) ("".TopResult, error)�#vtype.func(*"".Client, string, string) ("".TopResult, error)�#2"".(*Client).TopContainer�#2"".(*Client).TopContainer�#8go.string."UnpauseContainer"�#.type.func(string) error�$Ftype.func(*"".Client, string) error�$:"".(*Client).UnpauseContainer�$:"".(*Client).UnpauseContainer�$&go.string."Version"�$8type.func() (*"".Env, error)�$Ltype.func(*"".Client) (*"".Env, error)�$("".(*Client).Version�%("".(*Client).Version�%2go.string."WaitContainer"�%dockerCertPath 4go.string."dockerCertPath"�*go.string."dockerEnv"@4 dockerEnv *go.string."dockerEnv"�"type."".dockerEnv��(�A;?& ,type..alg."".dockerEnv0bruntime.gcbits.0x48848444480000000000000000000000P8go.string."docker.dockerEnv"p$type.*"".dockerEnv�"runtime.zerovalue��"type."".dockerEnv�,go.string."dockerHost"�"go.importpath."".�type.string�6go.string."dockerTLSVerify"�"go.importpath."".�type.bool�4go.string."dockerCertPath"�"go.importpath."".�type.string`�"type."".dockerEnv�*go.string."dockerEnv"�"go.importpath."".��"type."".dockerEnv�:go.string."*docker.dockerEnv"PD*docker.dockerEnv :go.string."*docker.dockerEnv"�$type.*"".dockerEnv��^�y6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."*docker.dockerEnv"p6go.weak.type.**"".dockerEnv�"runtime.zerovalue�"type."".dockerEnv�:go.string."[]tls.Certificate"PD[]tls.Certificate :go.string."[]tls.Certificate"�:type.[]crypto/tls.Certificate����B� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P:go.string."[]tls.Certificate"pLgo.weak.type.*[]crypto/tls.Certificate�"runtime.zerovalue�6type.crypto/tls.Certificate�lgo.typelink.[]tls.Certificate/[]crypto/tls.Certificate:type.[]crypto/tls.Certificate�bruntime.gcbits.0x48c48c448844cc488400000000000000 HČD�D�H��q�" � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P&go.string."[][]int"p*go.weak.type.*[][]int�"runtime.zerovalue�type.[]int�6go.typelink.[][]int/[][]inttype.[][]int�(go.string."[8][]int"@2[8][]int (go.string."[8][]int"�type.[8][]int������ � runtime.algarray0bruntime.gcbits.0x48844448844448844448844400000000P(go.string."[8][]int"p,go.weak.type.*[8][]int�"runtime.zerovalue�type.[]int�type.[][]int�:go.typelink.[8][]int/[8][]inttype.[8][]int�Hgo.string."*map.bucket[string][]int"`R*map.bucket[string][]int Hgo.string."*map.bucket[string][]int"�:type.*map.bucket[string][]int����6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PHgo.string."*map.bucket[string][]int"pLgo.weak.type.**map.bucket[string][]int�"runtime.zerovalue�8type.map.bucket[string][]int�,@type..gc.map.bucket[string][]int,�Htype..gcprog.map.bucket[string][]int*����Y�eY�e �Fgo.string."map.bucket[string][]int"PPmap.bucket[string][]int Fgo.string."map.bucket[string][]int"�8type.map.bucket[string][]int��P< +0�Y�H � runtime.algarray0@type..gc.map.bucket[string][]int@Htype..gcprog.map.bucket[string][]intPFgo.string."map.bucket[string][]int"pJgo.weak.type.*map.bucket[string][]int�"runtime.zerovalue��8type.map.bucket[string][]int� go.string."keys"�type.[8]string�$go.string."values"�type.[8][]int�(go.string."overflow"�:type.*map.bucket[string][]int�@go.string."map.hdr[string][]int"PJmap.hdr[string][]int @go.string."map.hdr[string][]int"�2type.map.hdr[string][]int��0�ę  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000P@go.string."map.hdr[string][]int"pDgo.weak.type.*map.hdr[string][]int�"runtime.zerovalue��2type.map.hdr[string][]int�&go.string."buckets"�:type.*map.bucket[string][]int�,go.string."oldbuckets"�:type.*map.bucket[string][]int�8go.string."map[string][]int"PBmap[string][]int 8go.string."map[string][]int"�*type.map[string][]int���(.�5P � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P8go.string."map[string][]int"pgo.string."*[1]tls.Certificate"PH*[1]tls.Certificate >go.string."*[1]tls.Certificate"�>type.*[1]crypto/tls.Certificate��Y +��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."*[1]tls.Certificate"pPgo.weak.type.**[1]crypto/tls.Certificate�"runtime.zerovalue�go.string."*docker.jsonMessage"PH*docker.jsonMessage >go.string."*docker.jsonMessage"�(type.*"".jsonMessage��s�[6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."*docker.jsonMessage"p:go.weak.type.**"".jsonMessage�"runtime.zerovalue�&type."".jsonMessage�Dgo.string."*map.hdr[string]string"PN*map.hdr[string]string Dgo.string."*map.hdr[string]string"�6type.*map.hdr[string]string����Ƽ6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PDgo.string."*map.hdr[string]string"pHgo.weak.type.**map.hdr[string]string�"runtime.zerovalue�4type.map.hdr[string]string�Dgo.string."map.iter[string]string"PNmap.iter[string]string Dgo.string."map.iter[string]string"�6type.map.iter[string]string��P��\ (0( � runtime.algarray0bruntime.gcbits.0x88888844440000000000000000000000PDgo.string."map.iter[string]string"pHgo.weak.type.*map.iter[string]string�"runtime.zerovalue��6type.map.iter[string]string�go.string."key"�type.*string�go.string."val"�type.*string�go.string."t"�type.*uint8�go.string."h"�6type.*map.hdr[string]string�&go.string."buckets"�go.typelink.chan bool/chan booltype.chan bool�,go.string."*chan bool"@6 +*chan bool ,go.string."*chan bool"�type.*chan bool��IJ�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."*chan bool"p0go.weak.type.**chan bool�"runtime.zerovalue�type.chan bool�.go.string."*chan error"@8 *chan error .go.string."*chan error"� type.*chan error��o �{6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P.go.string."*chan error"p2go.weak.type.**chan error�"runtime.zerovalue�type.chan error�4go.string."**bufio.Reader"@>**bufio.Reader 4go.string."**bufio.Reader"�&type.**bufio.Reader����6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."**bufio.Reader"p8go.weak.type.***bufio.Reader�"runtime.zerovalue�$type.*bufio.Reader�bruntime.gcbits.0x84884888880000000000000000000000 ��H����go.string."struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *docker.hijackOptions; A3 **bufio.Reader }"��`struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *docker.hijackOptions; A3 **bufio.Reader } �go.string."struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *docker.hijackOptions; A3 **bufio.Reader }"�go.string."F"0$F go.string."F"�go.string."A0"0&A0 go.string."A0"�go.string."A1"0&A1 go.string."A1"�go.string."A2"0&A2 go.string."A2"�go.string."A3"0&A3 go.string."A3"��type.struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *"".hijackOptions; A3 **bufio.Reader }��(�-��  runtime.algarray0bruntime.gcbits.0x84884888880000000000000000000000P�go.string."struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *docker.hijackOptions; A3 **bufio.Reader }"p�go.weak.type.*struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *"".hijackOptions; A3 **bufio.Reader }�"runtime.zerovalue���type.struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *"".hijackOptions; A3 **bufio.Reader }�go.string."F"�type.uintptr�go.string."A0"�type.*chan bool�go.string."A1"� type.*chan error�go.string."A2"�,type.*"".hijackOptions�go.string."A3"�&type.**bufio.Reader�bruntime.gcbits.0x84880000000000000000000000000000 ����go.string."struct { F uintptr; A0 *docker.hijackOptions; A1 *net.Conn; A2 *chan error }"��Lstruct { F uintptr; A0 *docker.hijackOptions; A1 *net.Conn; A2 *chan error } �go.string."struct { F uintptr; A0 *docker.hijackOptions; A1 *net.Conn; A2 *chan error }"��type.struct { F uintptr; A0 *"".hijackOptions; A1 *net.Conn; A2 *chan error }�� ���0  runtime.algarray0bruntime.gcbits.0x84880000000000000000000000000000P�go.string."struct { F uintptr; A0 *docker.hijackOptions; A1 *net.Conn; A2 *chan error }"p�go.weak.type.*struct { F uintptr; A0 *"".hijackOptions; A1 *net.Conn; A2 *chan error }�"runtime.zerovalue���type.struct { F uintptr; A0 *"".hijackOptions; A1 *net.Conn; A2 *chan error }�go.string."F"�type.uintptr�go.string."A0"�,type.*"".hijackOptions�go.string."A1"�type.*net.Conn�go.string."A2"� type.*chan error��go.string."*struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *docker.hijackOptions; A3 **bufio.Reader }"��a*struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *docker.hijackOptions; A3 **bufio.Reader } �go.string."*struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *docker.hijackOptions; A3 **bufio.Reader }"��type.*struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *"".hijackOptions; A3 **bufio.Reader }��� �6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."*struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *docker.hijackOptions; A3 **bufio.Reader }"p�go.weak.type.**struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *"".hijackOptions; A3 **bufio.Reader }�"runtime.zerovalue��type.struct { F uintptr; A0 *chan bool; A1 *chan error; A2 *"".hijackOptions; A3 **bufio.Reader }��go.string."*struct { F uintptr; A0 *docker.hijackOptions; A1 *net.Conn; A2 *chan error }"��M*struct { F uintptr; A0 *docker.hijackOptions; A1 *net.Conn; A2 *chan error } �go.string."*struct { F uintptr; A0 *docker.hijackOptions; A1 *net.Conn; A2 *chan error }"��type.*struct { F uintptr; A0 *"".hijackOptions; A1 *net.Conn; A2 *chan error }�����6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."*struct { F uintptr; A0 *docker.hijackOptions; A1 *net.Conn; A2 *chan error }"p�go.weak.type.**struct { F uintptr; A0 *"".hijackOptions; A1 *net.Conn; A2 *chan error }�"runtime.zerovalue��type.struct { F uintptr; A0 *"".hijackOptions; A1 *net.Conn; A2 *chan error }�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�go.weak.type.**[]"".PortBinding�"runtime.zerovalue�*type.[]"".PortBinding�jgo.string."*map.hdr[docker.Port][]docker.PortBinding"�t)*map.hdr[docker.Port][]docker.PortBinding jgo.string."*map.hdr[docker.Port][]docker.PortBinding"�Ltype.*map.hdr["".Port][]"".PortBinding��Gi0�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pjgo.string."*map.hdr[docker.Port][]docker.PortBinding"p^go.weak.type.**map.hdr["".Port][]"".PortBinding�"runtime.zerovalue�Jtype.map.hdr["".Port][]"".PortBinding�jgo.string."map.iter[docker.Port][]docker.PortBinding"�t)map.iter[docker.Port][]docker.PortBinding jgo.string."map.iter[docker.Port][]docker.PortBinding"�Ltype.map.iter["".Port][]"".PortBinding��P.�Gv (0( � runtime.algarray0bruntime.gcbits.0x88888844440000000000000000000000Pjgo.string."map.iter[docker.Port][]docker.PortBinding"p^go.weak.type.*map.iter["".Port][]"".PortBinding�"runtime.zerovalue��Ltype.map.iter["".Port][]"".PortBinding�go.string."key"�type.*"".Port�go.string."val"�,type.*[]"".PortBinding�go.string."t"�type.*uint8�go.string."h"�Ltype.*map.hdr["".Port][]"".PortBinding�&go.string."buckets"�Rtype.*map.bucket["".Port][]"".PortBinding� go.string."bptr"�Rtype.*map.bucket["".Port][]"".PortBinding�"go.string."other"�type.[4]uintptr�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·e13351f28add7c60853cb3aac0a0e34e�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�Btype..hashfunc."".NoSuchContainer:type..hash."".NoSuchContainer�>type..eqfunc."".NoSuchContainer6type..eq."".NoSuchContainer�8type..alg."".NoSuchContainer Btype..hashfunc."".NoSuchContainer>type..eqfunc."".NoSuchContainer�Fgo.string."*docker.NoSuchContainer"PP*docker.NoSuchContainer Fgo.string."*docker.NoSuchContainer"�`go.string."func(*docker.NoSuchContainer) string"pj$func(*docker.NoSuchContainer) string `go.string."func(*docker.NoSuchContainer) string"�Jtype.func(*"".NoSuchContainer) string��g= �3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P`go.string."func(*docker.NoSuchContainer) string"p\go.weak.type.*func(*"".NoSuchContainer) string�"runtime.zerovalue��Jtype.func(*"".NoSuchContainer) string��Jtype.func(*"".NoSuchContainer) string�0type.*"".NoSuchContainer�type.string�0type.*"".NoSuchContainer��mc0 +6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PFgo.string."*docker.NoSuchContainer"pBgo.weak.type.**"".NoSuchContainer�"runtime.zerovalue�.type."".NoSuchContainer`�0type.*"".NoSuchContainer��0type.*"".NoSuchContainer�"go.string."Error"�$type.func() string�Jtype.func(*"".NoSuchContainer) string�6"".(*NoSuchContainer).Error�6"".(*NoSuchContainer).Error�Dgo.string."docker.NoSuchContainer"PNdocker.NoSuchContainer Dgo.string."docker.NoSuchContainer"�go.string."Err"0(Err go.string."Err"�6go.string."NoSuchContainer"@@NoSuchContainer 6go.string."NoSuchContainer"�.type."".NoSuchContainer�� ��7p 8type..alg."".NoSuchContainer0bruntime.gcbits.0x488c0000000000000000000000000000PDgo.string."docker.NoSuchContainer"p0type.*"".NoSuchContainer�"runtime.zerovalue��.type."".NoSuchContainer�go.string."ID"�type.string�go.string."Err"�type.error`�.type."".NoSuchContainer�6go.string."NoSuchContainer"�"go.importpath."".��.type."".NoSuchContainer�8go.string."*[]docker.Change"PB*[]docker.Change 8go.string."*[]docker.Change"�"type.*[]"".Change���x��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P8go.string."*[]docker.Change"p4go.weak.type.**[]"".Change�"runtime.zerovalue� type.[]"".Change�""..gostring.7��vstruct { *docker.Config; HostConfig *docker.HostConfig "json:\"HostConfig,omitempty\" yaml:\"HostConfig,omitempty\"" } ""..gostring.7��type.struct { *"".Config; HostConfig *"".HostConfig "json:\"HostConfig,omitempty\" yaml:\"HostConfig,omitempty\"" }����w � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P""..gostring.7p�go.weak.type.*struct { *"".Config; HostConfig *"".HostConfig "json:\"HostConfig,omitempty\" yaml:\"HostConfig,omitempty\"" }�"runtime.zerovalue���type.struct { *"".Config; HostConfig *"".HostConfig "json:\"HostConfig,omitempty\" yaml:\"HostConfig,omitempty\"" }�type.*"".Config�,go.string."HostConfig"�&type.*"".HostConfig��go.string."json:\"HostConfig,omitempty\" yaml:\"HostConfig,omitempty\""�Vgo.string."*docker.ContainerAlreadyRunning"``*docker.ContainerAlreadyRunning Vgo.string."*docker.ContainerAlreadyRunning"�pgo.string."func(*docker.ContainerAlreadyRunning) string"�z,func(*docker.ContainerAlreadyRunning) string pgo.string."func(*docker.ContainerAlreadyRunning) string"�Ztype.func(*"".ContainerAlreadyRunning) string���.m3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ppgo.string."func(*docker.ContainerAlreadyRunning) string"plgo.weak.type.*func(*"".ContainerAlreadyRunning) string�"runtime.zerovalue��Ztype.func(*"".ContainerAlreadyRunning) string��Ztype.func(*"".ContainerAlreadyRunning) string�@type.*"".ContainerAlreadyRunning�type.string�@type.*"".ContainerAlreadyRunning��I#H6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PVgo.string."*docker.ContainerAlreadyRunning"pRgo.weak.type.**"".ContainerAlreadyRunning�"runtime.zerovalue�>type."".ContainerAlreadyRunning`�@type.*"".ContainerAlreadyRunning��@type.*"".ContainerAlreadyRunning�"go.string."Error"�$type.func() string�Ztype.func(*"".ContainerAlreadyRunning) string�F"".(*ContainerAlreadyRunning).Error�F"".(*ContainerAlreadyRunning).Error�Tgo.string."docker.ContainerAlreadyRunning"`^docker.ContainerAlreadyRunning Tgo.string."docker.ContainerAlreadyRunning"�Fgo.string."ContainerAlreadyRunning"PPContainerAlreadyRunning Fgo.string."ContainerAlreadyRunning"�>type."".ContainerAlreadyRunning����� � runtime.algarray0bruntime.gcbits.0x48000000000000000000000000000000PTgo.string."docker.ContainerAlreadyRunning"p@type.*"".ContainerAlreadyRunning�"runtime.zerovalue��>type."".ContainerAlreadyRunning�go.string."ID"�type.string`�>type."".ContainerAlreadyRunning�Fgo.string."ContainerAlreadyRunning"�"go.importpath."".��>type."".ContainerAlreadyRunning�Ngo.string."*docker.ContainerNotRunning"`X*docker.ContainerNotRunning Ngo.string."*docker.ContainerNotRunning"�hgo.string."func(*docker.ContainerNotRunning) string"�r(func(*docker.ContainerNotRunning) string hgo.string."func(*docker.ContainerNotRunning) string"�Rtype.func(*"".ContainerNotRunning) string��H R3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Phgo.string."func(*docker.ContainerNotRunning) string"pdgo.weak.type.*func(*"".ContainerNotRunning) string�"runtime.zerovalue��Rtype.func(*"".ContainerNotRunning) string��Rtype.func(*"".ContainerNotRunning) string�8type.*"".ContainerNotRunning�type.string�8type.*"".ContainerNotRunning��D�z�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PNgo.string."*docker.ContainerNotRunning"pJgo.weak.type.**"".ContainerNotRunning�"runtime.zerovalue�6type."".ContainerNotRunning`�8type.*"".ContainerNotRunning��8type.*"".ContainerNotRunning�"go.string."Error"�$type.func() string�Rtype.func(*"".ContainerNotRunning) string�>"".(*ContainerNotRunning).Error�>"".(*ContainerNotRunning).Error�Lgo.string."docker.ContainerNotRunning"`Vdocker.ContainerNotRunning Lgo.string."docker.ContainerNotRunning"�>go.string."ContainerNotRunning"PHContainerNotRunning >go.string."ContainerNotRunning"�6type."".ContainerNotRunning���nV � runtime.algarray0bruntime.gcbits.0x48000000000000000000000000000000PLgo.string."docker.ContainerNotRunning"p8type.*"".ContainerNotRunning�"runtime.zerovalue��6type."".ContainerNotRunning�go.string."ID"�type.string`�6type."".ContainerNotRunning�>go.string."ContainerNotRunning"�"go.importpath."".��6type."".ContainerNotRunning�6go.string."**docker.Client"@@**docker.Client 6go.string."**docker.Client"� type.**"".Client��l��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."**docker.Client"p2go.weak.type.***"".Client�"runtime.zerovalue�type.*"".Client�6go.string."**io.PipeWriter"@@**io.PipeWriter 6go.string."**io.PipeWriter"�(type.**io.PipeWriter��8ch<6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."**io.PipeWriter"p:go.weak.type.***io.PipeWriter�"runtime.zerovalue�&type.*io.PipeWriter�""..gostring.8��estruct { F uintptr; A0 **docker.Client; A1 *docker.StatsOptions; A2 **io.PipeWriter; A3 *chan error } ""..gostring.8��type.struct { F uintptr; A0 **"".Client; A1 *"".StatsOptions; A2 **io.PipeWriter; A3 *chan error }��(&LY+  runtime.algarray0bruntime.gcbits.0x84884888880000000000000000000000P""..gostring.8p�go.weak.type.*struct { F uintptr; A0 **"".Client; A1 *"".StatsOptions; A2 **io.PipeWriter; A3 *chan error }�"runtime.zerovalue���type.struct { F uintptr; A0 **"".Client; A1 *"".StatsOptions; A2 **io.PipeWriter; A3 *chan error }�go.string."F"�type.uintptr�go.string."A0"� type.**"".Client�go.string."A1"�*type.*"".StatsOptions�go.string."A2"�(type.**io.PipeWriter�go.string."A3"� type.*chan error�6go.string."**io.PipeReader"@@**io.PipeReader 6go.string."**io.PipeReader"�(type.**io.PipeReader��#���6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."**io.PipeReader"p:go.weak.type.***io.PipeReader�"runtime.zerovalue�&type.*io.PipeReader�6go.string."*chan struct {}"@@*chan struct {} 6go.string."*chan struct {}"�(type.*chan struct {}��<^>�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."*chan struct {}"p:go.weak.type.**chan struct {}�"runtime.zerovalue�&type.chan struct {}��go.string."struct { F uintptr; A0 *docker.StatsOptions; A1 **io.PipeReader; A2 *chan struct {} }"��Ustruct { F uintptr; A0 *docker.StatsOptions; A1 **io.PipeReader; A2 *chan struct {} } �go.string."struct { F uintptr; A0 *docker.StatsOptions; A1 **io.PipeReader; A2 *chan struct {} }"��type.struct { F uintptr; A0 *"".StatsOptions; A1 **io.PipeReader; A2 *chan struct {} }�� �ѕ�  runtime.algarray0bruntime.gcbits.0x84880000000000000000000000000000P�go.string."struct { F uintptr; A0 *docker.StatsOptions; A1 **io.PipeReader; A2 *chan struct {} }"p�go.weak.type.*struct { F uintptr; A0 *"".StatsOptions; A1 **io.PipeReader; A2 *chan struct {} }�"runtime.zerovalue���type.struct { F uintptr; A0 *"".StatsOptions; A1 **io.PipeReader; A2 *chan struct {} }�go.string."F"�type.uintptr�go.string."A0"�*type.*"".StatsOptions�go.string."A1"�(type.**io.PipeReader�go.string."A2"�(type.*chan struct {}�4go.string."**docker.Stats"@>**docker.Stats 4go.string."**docker.Stats"�type.**"".Stats��cܮ�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."**docker.Stats"p0go.weak.type.***"".Stats�"runtime.zerovalue�type.*"".Stats��go.string."struct { F uintptr; A0 *docker.StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader }"��\struct { F uintptr; A0 *docker.StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader } �go.string."struct { F uintptr; A0 *docker.StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader }"��type.struct { F uintptr; A0 *"".StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader }��(����  runtime.algarray0bruntime.gcbits.0x84884888880000000000000000000000P�go.string."struct { F uintptr; A0 *docker.StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader }"p�go.weak.type.*struct { F uintptr; A0 *"".StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader }�"runtime.zerovalue���type.struct { F uintptr; A0 *"".StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader }�go.string."F"�type.uintptr�go.string."A0"�*type.*"".StatsOptions�go.string."A1"� type.*chan error�go.string."A2"�type.*error�go.string."A3"�(type.**io.PipeReader��go.string."*struct { F uintptr; A0 *docker.StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader }"��]*struct { F uintptr; A0 *docker.StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader } �go.string."*struct { F uintptr; A0 *docker.StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader }"��type.*struct { F uintptr; A0 *"".StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader }���'�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."*struct { F uintptr; A0 *docker.StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader }"p�go.weak.type.**struct { F uintptr; A0 *"".StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader }�"runtime.zerovalue��type.struct { F uintptr; A0 *"".StatsOptions; A1 *chan error; A2 *error; A3 **io.PipeReader }�""..gostring.9��f*struct { F uintptr; A0 **docker.Client; A1 *docker.StatsOptions; A2 **io.PipeWriter; A3 *chan error } ""..gostring.9��type.*struct { F uintptr; A0 **"".Client; A1 *"".StatsOptions; A2 **io.PipeWriter; A3 *chan error }��3t�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P""..gostring.9p�go.weak.type.**struct { F uintptr; A0 **"".Client; A1 *"".StatsOptions; A2 **io.PipeWriter; A3 *chan error }�"runtime.zerovalue��type.struct { F uintptr; A0 **"".Client; A1 *"".StatsOptions; A2 **io.PipeWriter; A3 *chan error }��go.string."*struct { F uintptr; A0 *docker.StatsOptions; A1 **io.PipeReader; A2 *chan struct {} }"��V*struct { F uintptr; A0 *docker.StatsOptions; A1 **io.PipeReader; A2 *chan struct {} } �go.string."*struct { F uintptr; A0 *docker.StatsOptions; A1 **io.PipeReader; A2 *chan struct {} }"��type.*struct { F uintptr; A0 *"".StatsOptions; A1 **io.PipeReader; A2 *chan struct {} }��Ǧ��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."*struct { F uintptr; A0 *docker.StatsOptions; A1 **io.PipeReader; A2 *chan struct {} }"p�go.weak.type.**struct { F uintptr; A0 *"".StatsOptions; A1 **io.PipeReader; A2 *chan struct {} }�"runtime.zerovalue��type.struct { F uintptr; A0 *"".StatsOptions; A1 **io.PipeReader; A2 *chan struct {} }�Jgo.string."struct { StatusCode int }"`Tstruct { StatusCode int } Jgo.string."struct { StatusCode int }"�,go.string."StatusCode"@6 +StatusCode ,go.string."StatusCode"�type.*struct { StatusCode int }����6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PLgo.string."*struct { StatusCode int }"pPgo.weak.type.**struct { StatusCode int }�"runtime.zerovalue�go.typelink.[1]string/[1]stringtype.[1]string�,go.string."*[1]string"@6 +*[1]string ,go.string."*[1]string"�type.*[1]string��l.!�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."*[1]string"p0go.weak.type.**[1]string�"runtime.zerovalue�type.[1]string�*go.string."*[]string"@4 *[]string *go.string."*[]string"�type.*[]string���"v�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P*go.string."*[]string"p.go.weak.type.**[]string�"runtime.zerovalue�type.[]string�2go.string."*interface {}"@< *interface {} 2go.string."*interface {}"�$type.*interface {}��O��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P2go.string."*interface {}"p6go.weak.type.**interface {}�"runtime.zerovalue�"type.interface {}�Pgo.string."*map.hdr[string]interface {}"`Z*map.hdr[string]interface {} Pgo.string."*map.hdr[string]interface {}"�Btype.*map.hdr[string]interface {}��� ( +6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."*map.hdr[string]interface {}"pTgo.weak.type.**map.hdr[string]interface {}�"runtime.zerovalue�@type.map.hdr[string]interface {}�Pgo.string."map.iter[string]interface {}"`Zmap.iter[string]interface {} Pgo.string."map.iter[string]interface {}"�Btype.map.iter[string]interface {}��Pm8� (0( � runtime.algarray0bruntime.gcbits.0x88888844440000000000000000000000PPgo.string."map.iter[string]interface {}"pTgo.weak.type.*map.iter[string]interface {}�"runtime.zerovalue��Btype.map.iter[string]interface {}�go.string."key"�type.*string�go.string."val"�$type.*interface {}�go.string."t"�type.*uint8�go.string."h"�Btype.*map.hdr[string]interface {}�&go.string."buckets"�Htype.*map.bucket[string]interface {}� go.string."bptr"�Htype.*map.bucket[string]interface {}�"go.string."other"�type.[4]uintptr�Jgo.string."*chan<- *docker.APIEvents"`T*chan<- *docker.APIEvents Jgo.string."*chan<- *docker.APIEvents"�4type.*chan<- *"".APIEvents���3�06 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PJgo.string."*chan<- *docker.APIEvents"pFgo.weak.type.**chan<- *"".APIEvents�"runtime.zerovalue�2type.chan<- *"".APIEvents�Ngo.string."*[]chan<- *docker.APIEvents"`X*[]chan<- *docker.APIEvents Ngo.string."*[]chan<- *docker.APIEvents"�8type.*[]chan<- *"".APIEvents�����&6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PNgo.string."*[]chan<- *docker.APIEvents"pJgo.weak.type.**[]chan<- *"".APIEvents�"runtime.zerovalue�6type.[]chan<- *"".APIEvents�Rgo.string."**docker.eventMonitoringState"`\**docker.eventMonitoringState Rgo.string."**docker.eventMonitoringState"�go.typelink.[3]*uint8/[3]*uint8type.[3]*uint8�(go.string."[]uint16"@2[]uint16 (go.string."[]uint16"�type.[]uint16����  � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P(go.string."[]uint16"p,go.weak.type.*[]uint16�"runtime.zerovalue�type.uint16�:go.typelink.[]uint16/[]uint16type.[]uint16�*go.string."[3]uint16"@4 [3]uint16 *go.string."[3]uint16"�type.[3]uint16���q|.�  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P*go.string."[3]uint16"p.go.weak.type.*[3]uint16�"runtime.zerovalue�type.uint16�type.[]uint16�>go.typelink.[3]uint16/[3]uint16type.[3]uint16�,�type..gc.struct { tcase uint16; ncase uint16; pollorder *uint8; lockorder *uint8; scase [3]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }; lockorderarr [3]*uint8; pollorderarr [3]uint16 }4��type..gcprog.struct { tcase uint16; ncase uint16; pollorder *uint8; lockorder *uint8; scase [3]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }; lockorderarr [3]*uint8; pollorderarr [3]uint16 }��i����""..gostring.13���struct { tcase uint16; ncase uint16; pollorder *uint8; lockorder *uint8; scase [3]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }; lockorderarr [3]*uint8; pollorderarr [3]uint16 } ""..gostring.13�"go.string."tcase"0,tcase "go.string."tcase"�"go.string."ncase"0,ncase "go.string."ncase"�*go.string."pollorder"@4 pollorder *go.string."pollorder"�*go.string."lockorder"@4 lockorder *go.string."lockorder"�"go.string."scase"0,scase "go.string."scase"�0go.string."lockorderarr"@: lockorderarr 0go.string."lockorderarr"�0go.string."pollorderarr"@: pollorderarr 0go.string."pollorderarr"��type.struct { tcase uint16; ncase uint16; pollorder *uint8; lockorder *uint8; scase [3]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }; lockorderarr [3]*uint8; pollorderarr [3]uint16 }����~�PY��8 � runtime.algarray0�type..gc.struct { tcase uint16; ncase uint16; pollorder *uint8; lockorder *uint8; scase [3]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }; lockorderarr [3]*uint8; pollorderarr [3]uint16 }@�type..gcprog.struct { tcase uint16; ncase uint16; pollorder *uint8; lockorder *uint8; scase [3]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }; lockorderarr [3]*uint8; pollorderarr [3]uint16 }P""..gostring.13p�go.weak.type.*struct { tcase uint16; ncase uint16; pollorder *uint8; lockorder *uint8; scase [3]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }; lockorderarr [3]*uint8; pollorderarr [3]uint16 }�"runtime.zerovalue���type.struct { tcase uint16; ncase uint16; pollorder *uint8; lockorder *uint8; scase [3]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }; lockorderarr [3]*uint8; pollorderarr [3]uint16 }�"go.string."tcase"�"go.importpath."".�type.uint16�"go.string."ncase"�"go.importpath."".�type.uint16�*go.string."pollorder"�"go.importpath."".�type.*uint8�*go.string."lockorder"�"go.importpath."".�type.*uint8�"go.string."scase"�"go.importpath."".��type.[3]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }�0go.string."lockorderarr"�"go.importpath."".�type.[3]*uint8�0go.string."pollorderarr"�"go.importpath."".�type.[3]uint16��go.string."*struct { F uintptr; A0 **docker.eventMonitoringState; A1 **docker.Client }"��K*struct { F uintptr; A0 **docker.eventMonitoringState; A1 **docker.Client } �go.string."*struct { F uintptr; A0 **docker.eventMonitoringState; A1 **docker.Client }"��type.*struct { F uintptr; A0 **"".eventMonitoringState; A1 **"".Client }�����@6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."*struct { F uintptr; A0 **docker.eventMonitoringState; A1 **docker.Client }"p�go.weak.type.**struct { F uintptr; A0 **"".eventMonitoringState; A1 **"".Client }�"runtime.zerovalue��type.struct { F uintptr; A0 **"".eventMonitoringState; A1 **"".Client }�Fgo.string."*chan *docker.APIEvents"PP*chan *docker.APIEvents Fgo.string."*chan *docker.APIEvents"�0type.*chan *"".APIEvents��~oN(6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PFgo.string."*chan *docker.APIEvents"pBgo.weak.type.**chan *"".APIEvents�"runtime.zerovalue�.type.chan *"".APIEvents��go.string."struct { F uintptr; A0 *error; A1 **docker.Client; A2 *chan *docker.APIEvents; A3 *chan error }"��_struct { F uintptr; A0 *error; A1 **docker.Client; A2 *chan *docker.APIEvents; A3 *chan error } �go.string."struct { F uintptr; A0 *error; A1 **docker.Client; A2 *chan *docker.APIEvents; A3 *chan error }"��type.struct { F uintptr; A0 *error; A1 **"".Client; A2 *chan *"".APIEvents; A3 *chan error }��(Ș�9  runtime.algarray0bruntime.gcbits.0x84884888880000000000000000000000P�go.string."struct { F uintptr; A0 *error; A1 **docker.Client; A2 *chan *docker.APIEvents; A3 *chan error }"p�go.weak.type.*struct { F uintptr; A0 *error; A1 **"".Client; A2 *chan *"".APIEvents; A3 *chan error }�"runtime.zerovalue���type.struct { F uintptr; A0 *error; A1 **"".Client; A2 *chan *"".APIEvents; A3 *chan error }�go.string."F"�type.uintptr�go.string."A0"�type.*error�go.string."A1"� type.**"".Client�go.string."A2"�0type.*chan *"".APIEvents�go.string."A3"� type.*chan error��go.string."*struct { F uintptr; A0 *error; A1 **docker.Client; A2 *chan *docker.APIEvents; A3 *chan error }"��`*struct { F uintptr; A0 *error; A1 **docker.Client; A2 *chan *docker.APIEvents; A3 *chan error } �go.string."*struct { F uintptr; A0 *error; A1 **docker.Client; A2 *chan *docker.APIEvents; A3 *chan error }"��type.*struct { F uintptr; A0 *error; A1 **"".Client; A2 *chan *"".APIEvents; A3 *chan error }���Қ�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."*struct { F uintptr; A0 *error; A1 **docker.Client; A2 *chan *docker.APIEvents; A3 *chan error }"p�go.weak.type.**struct { F uintptr; A0 *error; A1 **"".Client; A2 *chan *"".APIEvents; A3 *chan error }�"runtime.zerovalue��type.struct { F uintptr; A0 *error; A1 **"".Client; A2 *chan *"".APIEvents; A3 *chan error }�go.string."*[]docker.APIImages"PH*[]docker.APIImages >go.string."*[]docker.APIImages"�(type.*[]"".APIImages������6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."*[]docker.APIImages"p:go.weak.type.**[]"".APIImages�"runtime.zerovalue�&type.[]"".APIImages�Dgo.string."*[]docker.ImageHistory"PN*[]docker.ImageHistory Dgo.string."*[]docker.ImageHistory"�.type.*[]"".ImageHistory��|؄�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PDgo.string."*[]docker.ImageHistory"p@go.weak.type.**[]"".ImageHistory�"runtime.zerovalue�,type.[]"".ImageHistory�6type..gcprog."".ImagePre012((AfV��U�e�e�ffeY�i�json:"created" 8go.string."json:\"created\""�Pgo.string."json:\"container,omitempty\""`Vjson:"container,omitempty" Pgo.string."json:\"container,omitempty\""�^go.string."json:\"container_config,omitempty\""pd!json:"container_config,omitempty" ^go.string."json:\"container_config,omitempty\""�Zgo.string."json:\"docker_version,omitempty\""``json:"docker_version,omitempty" Zgo.string."json:\"docker_version,omitempty\""�Jgo.string."json:\"author,omitempty\""PPjson:"author,omitempty" Jgo.string."json:\"author,omitempty\""�Jgo.string."json:\"config,omitempty\""PPjson:"config,omitempty" Jgo.string."json:\"config,omitempty\""�Vgo.string."json:\"architecture,omitempty\""`\json:"architecture,omitempty" Vgo.string."json:\"architecture,omitempty\""�Fgo.string."json:\"size,omitempty\""PLjson:"size,omitempty" Fgo.string."json:\"size,omitempty\""�.go.string."ImagePre012"@8 ImagePre012 .go.string."ImagePre012"�&type."".ImagePre012� � ����Y  0HX����V � runtime.algarray@6type..gcprog."".ImagePre012Pgo.string."*docker.ImagePre012"PH*docker.ImagePre012 >go.string."*docker.ImagePre012"�(type.*"".ImagePre012�� �Nw6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."*docker.ImagePre012"p:go.weak.type.**"".ImagePre012�"runtime.zerovalue�&type."".ImagePre012�Hgo.string."*[]docker.APIImageSearch"`R*[]docker.APIImageSearch Hgo.string."*[]docker.APIImageSearch"�2type.*[]"".APIImageSearch��S���6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PHgo.string."*[]docker.APIImageSearch"pDgo.weak.type.**[]"".APIImageSearch�"runtime.zerovalue�0type.[]"".APIImageSearch�:go.string."*[]docker.Network"PD*[]docker.Network :go.string."*[]docker.Network"�$type.*[]"".Network��Mr�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."*[]docker.Network"p6go.weak.type.**[]"".Network�"runtime.zerovalue�"type.[]"".Network�Bgo.string."*docker.NoSuchNetwork"PL*docker.NoSuchNetwork Bgo.string."*docker.NoSuchNetwork"�\go.string."func(*docker.NoSuchNetwork) string"pf"func(*docker.NoSuchNetwork) string \go.string."func(*docker.NoSuchNetwork) string"�Ftype.func(*"".NoSuchNetwork) string����93 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P\go.string."func(*docker.NoSuchNetwork) string"pXgo.weak.type.*func(*"".NoSuchNetwork) string�"runtime.zerovalue��Ftype.func(*"".NoSuchNetwork) string��Ftype.func(*"".NoSuchNetwork) string�,type.*"".NoSuchNetwork�type.string�,type.*"".NoSuchNetwork���\�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."*docker.NoSuchNetwork"p>go.weak.type.**"".NoSuchNetwork�"runtime.zerovalue�*type."".NoSuchNetwork`�,type.*"".NoSuchNetwork��,type.*"".NoSuchNetwork�"go.string."Error"�$type.func() string�Ftype.func(*"".NoSuchNetwork) string�2"".(*NoSuchNetwork).Error�2"".(*NoSuchNetwork).Error�@go.string."docker.NoSuchNetwork"PJdocker.NoSuchNetwork @go.string."docker.NoSuchNetwork"�2go.string."NoSuchNetwork"@< NoSuchNetwork 2go.string."NoSuchNetwork"�*type."".NoSuchNetwork��^�`� � runtime.algarray0bruntime.gcbits.0x48000000000000000000000000000000P@go.string."docker.NoSuchNetwork"p,type.*"".NoSuchNetwork�"runtime.zerovalue��*type."".NoSuchNetwork�go.string."ID"�type.string`�*type."".NoSuchNetwork�2go.string."NoSuchNetwork"�"go.importpath."".��*type."".NoSuchNetwork�Pgo.string."docker.createNetworkResponse"`Zdocker.createNetworkResponse Pgo.string."docker.createNetworkResponse"�Bgo.string."createNetworkResponse"PLcreateNetworkResponse Bgo.string."createNetworkResponse"�@type."".createNetworkResponse·1���D�� � runtime.algarray0bruntime.gcbits.0x48000000000000000000000000000000PPgo.string."docker.createNetworkResponse"pBtype.*"".createNetworkResponse·1�"runtime.zerovalue��@type."".createNetworkResponse·1�go.string."ID"�type.string`�@type."".createNetworkResponse·1�Bgo.string."createNetworkResponse"�"go.importpath."".��@type."".createNetworkResponse·1�Rgo.string."*docker.createNetworkResponse"`\*docker.createNetworkResponse Rgo.string."*docker.createNetworkResponse"�Btype.*"".createNetworkResponse·1��i�#�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PRgo.string."*docker.createNetworkResponse"pTgo.weak.type.**"".createNetworkResponse·1�"runtime.zerovalue�@type."".createNetworkResponse·1�tgo.string."struct { F uintptr; A0 *string; A1 *[]string }"�~.struct { F uintptr; A0 *string; A1 *[]string } tgo.string."struct { F uintptr; A0 *string; A1 *[]string }"�ftype.struct { F uintptr; A0 *string; A1 *[]string }��K��'  runtime.algarray0bruntime.gcbits.0x84488800000000000000000000000000Ptgo.string."struct { F uintptr; A0 *string; A1 *[]string }"pxgo.weak.type.*struct { F uintptr; A0 *string; A1 *[]string }�"runtime.zerovalue��ftype.struct { F uintptr; A0 *string; A1 *[]string }�go.string."F"�type.uintptr�go.string."A0"�type.*string�go.string."A1"�type.*[]string�vgo.string."*struct { F uintptr; A0 *string; A1 *[]string }"��/*struct { F uintptr; A0 *string; A1 *[]string } vgo.string."*struct { F uintptr; A0 *string; A1 *[]string }"�htype.*struct { F uintptr; A0 *string; A1 *[]string }����4�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pvgo.string."*struct { F uintptr; A0 *string; A1 *[]string }"pzgo.weak.type.**struct { F uintptr; A0 *string; A1 *[]string }�"runtime.zerovalue�ftype.struct { F uintptr; A0 *string; A1 *[]string }�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·78fd77a07ab543a063c3a3049973febe �Xgo.string."interface { CloseWrite() error }"pb interface { CloseWrite() error } Xgo.string."interface { CloseWrite() error }"�,go.string."CloseWrite"@6 +CloseWrite ,go.string."CloseWrite"�Jtype.interface { CloseWrite() error }���Z� � runtime.algarray0bruntime.gcbits.0x8c000000000000000000000000000000PXgo.string."interface { CloseWrite() error }"p\go.weak.type.*interface { CloseWrite() error }�"runtime.zerovalue��Jtype.interface { CloseWrite() error }�,go.string."CloseWrite"�"type.func() error�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�go.string."docker.tlsClientCon"PHdocker.tlsClientCon >go.string."docker.tlsClientCon"�&go.string."rawConn"00rawConn &go.string."rawConn"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·57e1009a600f832f844e0e3c49ba5a89 +.�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·40de35fb9b773b345d1ee7cba691ea13 �Tgclocals·b0f264e78fa38c77ad79fe8a353279f7�Tgclocals·25609300e15c97db07af80faee4d2fd6 $.�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·57e1009a600f832f844e0e3c49ba5a89 +.�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·57e1009a600f832f844e0e3c49ba5a89 +.�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·13d3af77a5bf02af6db4588efb2ea811�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·da455f41cf2a78c8890074a4a256bdd4 .�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·9877a4ef732a0f966b889793f9b99b87 +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·0273bd9c87bb10f67d516fbf00fd7767��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·57e1009a600f832f844e0e3c49ba5a89 +.�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·86b4418f46455e3a0eb577619691d10f ��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·be9b149192cd561578dd28b30f28e84fn �Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·86b4418f46455e3a0eb577619691d10f ��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·be9b149192cd561578dd28b30f28e84fn �Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·86b4418f46455e3a0eb577619691d10f ��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·be9b149192cd561578dd28b30f28e84fn �Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·14c45952157723c8762210d9c661bf29 + +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·be4f16eacaf744756abcb34364e01385��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·9877a4ef732a0f966b889793f9b99b87 +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·0273bd9c87bb10f67d516fbf00fd7767��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·57e1009a600f832f844e0e3c49ba5a89 +.�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·9f0d5ba6770c4a1ed4fa771547e96df1 +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·4e44481e9dee421443081e94ffaa0dd2��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·9877a4ef732a0f966b889793f9b99b87 +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·0273bd9c87bb10f67d516fbf00fd7767��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·5dfce38b1d248a3900c6ec750de77702 +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·d93d6c9fc85d7888b8b1832756680f45.�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·6a2e5ab2d393a1bfd331903fbd0fd425�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·c776d40308d3cc87dab399555a94d3ca n�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·6a2e5ab2d393a1bfd331903fbd0fd425�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·c776d40308d3cc87dab399555a94d3ca n�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·6a2e5ab2d393a1bfd331903fbd0fd425�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·c776d40308d3cc87dab399555a94d3ca n�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·57e1009a600f832f844e0e3c49ba5a89 +.�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a99c50f5f5d34b1bf54d8ece6dad05c2&�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·adf7fd756b6e86afbfe88b4b789f56a2nB�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·f35b06e445e251bd5ec01f0c98f96353&�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·7f02f8d572b489939b1d1d8614f82cden�Vgo.string."func(docker.tlsClientCon) error"``func(docker.tlsClientCon) error Vgo.string."func(docker.tlsClientCon) error"�@type.func("".tlsClientCon) error��.�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PVgo.string."func(docker.tlsClientCon) error"pRgo.weak.type.*func("".tlsClientCon) error�"runtime.zerovalue��@type.func("".tlsClientCon) error��@type.func("".tlsClientCon) error�(type."".tlsClientCon�type.error�rgo.string."func(docker.tlsClientCon) tls.ConnectionState"�|-func(docker.tlsClientCon) tls.ConnectionState rgo.string."func(docker.tlsClientCon) tls.ConnectionState"�jtype.func("".tlsClientCon) crypto/tls.ConnectionState���d 3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Prgo.string."func(docker.tlsClientCon) tls.ConnectionState"p|go.weak.type.*func("".tlsClientCon) crypto/tls.ConnectionState�"runtime.zerovalue��jtype.func("".tlsClientCon) crypto/tls.ConnectionState��jtype.func("".tlsClientCon) crypto/tls.ConnectionState�(type."".tlsClientCon�>type.crypto/tls.ConnectionState�\go.string."func(docker.tlsClientCon) net.Addr"pf"func(docker.tlsClientCon) net.Addr \go.string."func(docker.tlsClientCon) net.Addr"�Ftype.func("".tlsClientCon) net.Addr��e�3'3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P\go.string."func(docker.tlsClientCon) net.Addr"pXgo.weak.type.*func("".tlsClientCon) net.Addr�"runtime.zerovalue��Ftype.func("".tlsClientCon) net.Addr��Ftype.func("".tlsClientCon) net.Addr�(type."".tlsClientCon�type.net.Addr�Zgo.string."func(docker.tlsClientCon) []uint8"pd!func(docker.tlsClientCon) []uint8 Zgo.string."func(docker.tlsClientCon) []uint8"�Dtype.func("".tlsClientCon) []uint8�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PZgo.string."func(docker.tlsClientCon) []uint8"pVgo.weak.type.*func("".tlsClientCon) []uint8�"runtime.zerovalue��Dtype.func("".tlsClientCon) []uint8��Dtype.func("".tlsClientCon) []uint8�(type."".tlsClientCon�type.[]uint8�vgo.string."func(docker.tlsClientCon, []uint8) (int, error)"��/func(docker.tlsClientCon, []uint8) (int, error) vgo.string."func(docker.tlsClientCon, []uint8) (int, error)"�`type.func("".tlsClientCon, []uint8) (int, error)���3��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pvgo.string."func(docker.tlsClientCon, []uint8) (int, error)"prgo.weak.type.*func("".tlsClientCon, []uint8) (int, error)�"runtime.zerovalue��`type.func("".tlsClientCon, []uint8) (int, error)��`type.func("".tlsClientCon, []uint8) (int, error)�(type."".tlsClientCon�type.[]uint8�type.int�type.error�lgo.string."func(docker.tlsClientCon, time.Time) error"�v*func(docker.tlsClientCon, time.Time) error lgo.string."func(docker.tlsClientCon, time.Time) error"�Vtype.func("".tlsClientCon, time.Time) error�� i��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Plgo.string."func(docker.tlsClientCon, time.Time) error"phgo.weak.type.*func("".tlsClientCon, time.Time) error�"runtime.zerovalue��Vtype.func("".tlsClientCon, time.Time) error��Vtype.func("".tlsClientCon, time.Time) error�(type."".tlsClientCon�type.time.Time�type.error�fgo.string."func(docker.tlsClientCon, string) error"pp'func(docker.tlsClientCon, string) error fgo.string."func(docker.tlsClientCon, string) error"�Ptype.func("".tlsClientCon, string) error��I�M3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pfgo.string."func(docker.tlsClientCon, string) error"pbgo.weak.type.*func("".tlsClientCon, string) error�"runtime.zerovalue��Ptype.func("".tlsClientCon, string) error��Ptype.func("".tlsClientCon, string) error�(type."".tlsClientCon�type.string�type.error��go.string."func(docker.tlsClientCon, []uint8) (*tls.sessionState, bool)"��<func(docker.tlsClientCon, []uint8) (*tls.sessionState, bool) �go.string."func(docker.tlsClientCon, []uint8) (*tls.sessionState, bool)"��type.func("".tlsClientCon, []uint8) (*crypto/tls.sessionState, bool)������3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(docker.tlsClientCon, []uint8) (*tls.sessionState, bool)"p�go.weak.type.*func("".tlsClientCon, []uint8) (*crypto/tls.sessionState, bool)�"runtime.zerovalue���type.func("".tlsClientCon, []uint8) (*crypto/tls.sessionState, bool)���type.func("".tlsClientCon, []uint8) (*crypto/tls.sessionState, bool)�(type."".tlsClientCon�type.[]uint8�:type.*crypto/tls.sessionState�type.bool��go.string."func(docker.tlsClientCon, *tls.sessionState) ([]uint8, error)"��=func(docker.tlsClientCon, *tls.sessionState) ([]uint8, error) �go.string."func(docker.tlsClientCon, *tls.sessionState) ([]uint8, error)"��type.func("".tlsClientCon, *crypto/tls.sessionState) ([]uint8, error)��g��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(docker.tlsClientCon, *tls.sessionState) ([]uint8, error)"p�go.weak.type.*func("".tlsClientCon, *crypto/tls.sessionState) ([]uint8, error)�"runtime.zerovalue���type.func("".tlsClientCon, *crypto/tls.sessionState) ([]uint8, error)���type.func("".tlsClientCon, *crypto/tls.sessionState) ([]uint8, error)�(type."".tlsClientCon�:type.*crypto/tls.sessionState�type.[]uint8�type.error�vgo.string."func(docker.tlsClientCon) (interface {}, error)"��/func(docker.tlsClientCon) (interface {}, error) vgo.string."func(docker.tlsClientCon) (interface {}, error)"�`type.func("".tlsClientCon) (interface {}, error)��m#�c3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pvgo.string."func(docker.tlsClientCon) (interface {}, error)"prgo.weak.type.*func("".tlsClientCon) (interface {}, error)�"runtime.zerovalue��`type.func("".tlsClientCon) (interface {}, error)��`type.func("".tlsClientCon) (interface {}, error)�(type."".tlsClientCon�"type.interface {}�type.error�vgo.string."func(docker.tlsClientCon, tls.recordType) error"��/func(docker.tlsClientCon, tls.recordType) error vgo.string."func(docker.tlsClientCon, tls.recordType) error"�ntype.func("".tlsClientCon, crypto/tls.recordType) error�����J3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pvgo.string."func(docker.tlsClientCon, tls.recordType) error"p�go.weak.type.*func("".tlsClientCon, crypto/tls.recordType) error�"runtime.zerovalue��ntype.func("".tlsClientCon, crypto/tls.recordType) error��ntype.func("".tlsClientCon, crypto/tls.recordType) error�(type."".tlsClientCon�4type.crypto/tls.recordType�type.error�lgo.string."func(docker.tlsClientCon, tls.alert) error"�v*func(docker.tlsClientCon, tls.alert) error lgo.string."func(docker.tlsClientCon, tls.alert) error"�dtype.func("".tlsClientCon, crypto/tls.alert) error����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Plgo.string."func(docker.tlsClientCon, tls.alert) error"pvgo.weak.type.*func("".tlsClientCon, crypto/tls.alert) error�"runtime.zerovalue��dtype.func("".tlsClientCon, crypto/tls.alert) error��dtype.func("".tlsClientCon, crypto/tls.alert) error�(type."".tlsClientCon�*type.crypto/tls.alert�type.error��go.string."func(docker.tlsClientCon, uint16, []uint16, uint16, bool, bool) *tls.cipherSuite"��Pfunc(docker.tlsClientCon, uint16, []uint16, uint16, bool, bool) *tls.cipherSuite �go.string."func(docker.tlsClientCon, uint16, []uint16, uint16, bool, bool) *tls.cipherSuite"��type.func("".tlsClientCon, uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite����ej3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(docker.tlsClientCon, uint16, []uint16, uint16, bool, bool) *tls.cipherSuite"p�go.weak.type.*func("".tlsClientCon, uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite�"runtime.zerovalue���type.func("".tlsClientCon, uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite���type.func("".tlsClientCon, uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite�(type."".tlsClientCon�type.uint16�type.[]uint16�type.uint16�type.bool�type.bool�8type.*crypto/tls.cipherSuite��go.string."func(docker.tlsClientCon, tls.recordType, []uint8) (int, error)"��?func(docker.tlsClientCon, tls.recordType, []uint8) (int, error) �go.string."func(docker.tlsClientCon, tls.recordType, []uint8) (int, error)"��type.func("".tlsClientCon, crypto/tls.recordType, []uint8) (int, error)����:3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(docker.tlsClientCon, tls.recordType, []uint8) (int, error)"p�go.weak.type.*func("".tlsClientCon, crypto/tls.recordType, []uint8) (int, error)�"runtime.zerovalue���type.func("".tlsClientCon, crypto/tls.recordType, []uint8) (int, error)���type.func("".tlsClientCon, crypto/tls.recordType, []uint8) (int, error)�(type."".tlsClientCon�4type.crypto/tls.recordType�type.[]uint8�type.int�type.error�0go.string."tlsClientCon"@: tlsClientCon 0go.string."tlsClientCon"�"go.string."Close"0,Close "go.string."Close"�6go.string."ConnectionState"@@ConnectionState 6go.string."ConnectionState"�Lgo.string."func() tls.ConnectionState"`Vfunc() tls.ConnectionState Lgo.string."func() tls.ConnectionState"�Ltype.func() crypto/tls.ConnectionState��%�33 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PLgo.string."func() tls.ConnectionState"p^go.weak.type.*func() crypto/tls.ConnectionState�"runtime.zerovalue��Ltype.func() crypto/tls.ConnectionState��Ltype.func() crypto/tls.ConnectionState�>type.crypto/tls.ConnectionState�*go.string."Handshake"@4 Handshake *go.string."Handshake"�*go.string."LocalAddr"@4 LocalAddr *go.string."LocalAddr"�6go.string."func() net.Addr"@@func() net.Addr 6go.string."func() net.Addr"�(type.func() net.Addr��WH�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."func() net.Addr"p:go.weak.type.*func() net.Addr�"runtime.zerovalue��(type.func() net.Addr��(type.func() net.Addr�type.net.Addr�0go.string."OCSPResponse"@: OCSPResponse 0go.string."OCSPResponse"�4go.string."func() []uint8"@>func() []uint8 4go.string."func() []uint8"�&type.func() []uint8���io%3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."func() []uint8"p8go.weak.type.*func() []uint8�"runtime.zerovalue��&type.func() []uint8��&type.func() []uint8�type.[]uint8�Lgo.string."func([]uint8) (int, error)"`Vfunc([]uint8) (int, error) Lgo.string."func([]uint8) (int, error)"�>type.func([]uint8) (int, error)���N4P3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PLgo.string."func([]uint8) (int, error)"pPgo.weak.type.*func([]uint8) (int, error)�"runtime.zerovalue��>type.func([]uint8) (int, error)��>type.func([]uint8) (int, error)�type.[]uint8�type.int�type.error�,go.string."RemoteAddr"@6 +RemoteAddr ,go.string."RemoteAddr"�.go.string."SetDeadline"@8 SetDeadline .go.string."SetDeadline"�Bgo.string."func(time.Time) error"PLfunc(time.Time) error Bgo.string."func(time.Time) error"�4type.func(time.Time) error��@Z�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."func(time.Time) error"pFgo.weak.type.*func(time.Time) error�"runtime.zerovalue��4type.func(time.Time) error��4type.func(time.Time) error�type.time.Time�type.error�6go.string."SetReadDeadline"@@SetReadDeadline 6go.string."SetReadDeadline"�8go.string."SetWriteDeadline"PBSetWriteDeadline 8go.string."SetWriteDeadline"�4go.string."VerifyHostname"@>VerifyHostname 4go.string."VerifyHostname"�"go.string."Write"0,Write "go.string."Write"�6go.string."clientHandshake"@@clientHandshake 6go.string."clientHandshake"�,go.string."crypto/tls"@6 +crypto/tls ,go.string."crypto/tls"�2go.importpath.crypto/tls. + ,go.string."crypto/tls"�2go.string."decryptTicket"@< decryptTicket 2go.string."decryptTicket"�fgo.string."func([]uint8) (*tls.sessionState, bool)"pp'func([]uint8) (*tls.sessionState, bool) fgo.string."func([]uint8) (*tls.sessionState, bool)"�ftype.func([]uint8) (*crypto/tls.sessionState, bool)���;O�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pfgo.string."func([]uint8) (*tls.sessionState, bool)"pxgo.weak.type.*func([]uint8) (*crypto/tls.sessionState, bool)�"runtime.zerovalue��ftype.func([]uint8) (*crypto/tls.sessionState, bool)��ftype.func([]uint8) (*crypto/tls.sessionState, bool)�type.[]uint8�:type.*crypto/tls.sessionState�type.bool�2go.string."encryptTicket"@< encryptTicket 2go.string."encryptTicket"�hgo.string."func(*tls.sessionState) ([]uint8, error)"�r(func(*tls.sessionState) ([]uint8, error) hgo.string."func(*tls.sessionState) ([]uint8, error)"�htype.func(*crypto/tls.sessionState) ([]uint8, error)��kI;h3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Phgo.string."func(*tls.sessionState) ([]uint8, error)"pzgo.weak.type.*func(*crypto/tls.sessionState) ([]uint8, error)�"runtime.zerovalue��htype.func(*crypto/tls.sessionState) ([]uint8, error)��htype.func(*crypto/tls.sessionState) ([]uint8, error)�:type.*crypto/tls.sessionState�type.[]uint8�type.error�2go.string."readHandshake"@< readHandshake 2go.string."readHandshake"�Pgo.string."func() (interface {}, error)"`Zfunc() (interface {}, error) Pgo.string."func() (interface {}, error)"�Btype.func() (interface {}, error)��j� /3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."func() (interface {}, error)"pTgo.weak.type.*func() (interface {}, error)�"runtime.zerovalue��Btype.func() (interface {}, error)��Btype.func() (interface {}, error)�"type.interface {}�type.error�,go.string."readRecord"@6 +readRecord ,go.string."readRecord"�Lgo.string."func(tls.recordType) error"`Vfunc(tls.recordType) error Lgo.string."func(tls.recordType) error"�Ltype.func(crypto/tls.recordType) error���w3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PLgo.string."func(tls.recordType) error"p^go.weak.type.*func(crypto/tls.recordType) error�"runtime.zerovalue��Ltype.func(crypto/tls.recordType) error��Ltype.func(crypto/tls.recordType) error�4type.crypto/tls.recordType�type.error�*go.string."sendAlert"@4 sendAlert *go.string."sendAlert"�Bgo.string."func(tls.alert) error"PLfunc(tls.alert) error Bgo.string."func(tls.alert) error"�Btype.func(crypto/tls.alert) error�� d��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."func(tls.alert) error"pTgo.weak.type.*func(crypto/tls.alert) error�"runtime.zerovalue��Btype.func(crypto/tls.alert) error��Btype.func(crypto/tls.alert) error�*type.crypto/tls.alert�type.error�6go.string."sendAlertLocked"@@sendAlertLocked 6go.string."sendAlertLocked"�6go.string."serverHandshake"@@serverHandshake 6go.string."serverHandshake"�4go.string."tryCipherSuite"@>tryCipherSuite 4go.string."tryCipherSuite"��go.string."func(uint16, []uint16, uint16, bool, bool) *tls.cipherSuite"��;func(uint16, []uint16, uint16, bool, bool) *tls.cipherSuite �go.string."func(uint16, []uint16, uint16, bool, bool) *tls.cipherSuite"��type.func(uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite��6~�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(uint16, []uint16, uint16, bool, bool) *tls.cipherSuite"p�go.weak.type.*func(uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite�"runtime.zerovalue���type.func(uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite���type.func(uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite�type.uint16�type.[]uint16�type.uint16�type.bool�type.bool�8type.*crypto/tls.cipherSuite�.go.string."writeRecord"@8 writeRecord .go.string."writeRecord"�lgo.string."func(tls.recordType, []uint8) (int, error)"�v*func(tls.recordType, []uint8) (int, error) lgo.string."func(tls.recordType, []uint8) (int, error)"�ltype.func(crypto/tls.recordType, []uint8) (int, error)����`3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Plgo.string."func(tls.recordType, []uint8) (int, error)"p~go.weak.type.*func(crypto/tls.recordType, []uint8) (int, error)�"runtime.zerovalue��ltype.func(crypto/tls.recordType, []uint8) (int, error)��ltype.func(crypto/tls.recordType, []uint8) (int, error)�4type.crypto/tls.recordType�type.[]uint8�type.int�type.error�(type."".tlsClientCon������� 2type..alg."".tlsClientCon0bruntime.gcbits.0xc8888c00000000000000000000000000P>go.string."docker.tlsClientCon"p*type.*"".tlsClientCon�"runtime.zerovalue��(type."".tlsClientCon�*type.*crypto/tls.Conn�&go.string."rawConn"�"go.importpath."".�type.net.Conn`�(type."".tlsClientCon�0go.string."tlsClientCon"�"go.importpath."".��(type."".tlsClientCon�"go.string."Close"�"type.func() error�@type.func("".tlsClientCon) error�0"".(*tlsClientCon).Close�*"".tlsClientCon.Close�6go.string."ConnectionState"�Ltype.func() crypto/tls.ConnectionState�jtype.func("".tlsClientCon) crypto/tls.ConnectionState�D"".(*tlsClientCon).ConnectionState�>"".tlsClientCon.ConnectionState�*go.string."Handshake"�"type.func() error�@type.func("".tlsClientCon) error�8"".(*tlsClientCon).Handshake�2"".tlsClientCon.Handshake�*go.string."LocalAddr"�(type.func() net.Addr�Ftype.func("".tlsClientCon) net.Addr�8"".(*tlsClientCon).LocalAddr�2"".tlsClientCon.LocalAddr�0go.string."OCSPResponse"�&type.func() []uint8�Dtype.func("".tlsClientCon) []uint8�>"".(*tlsClientCon).OCSPResponse�8"".tlsClientCon.OCSPResponse� go.string."Read"�>type.func([]uint8) (int, error)�`type.func("".tlsClientCon, []uint8) (int, error)�."".(*tlsClientCon).Read�("".tlsClientCon.Read�,go.string."RemoteAddr"�(type.func() net.Addr�Ftype.func("".tlsClientCon) net.Addr�:"".(*tlsClientCon).RemoteAddr�4"".tlsClientCon.RemoteAddr�.go.string."SetDeadline"�4type.func(time.Time) error� Vtype.func("".tlsClientCon, time.Time) error� <"".(*tlsClientCon).SetDeadline� 6"".tlsClientCon.SetDeadline� 6go.string."SetReadDeadline"� 4type.func(time.Time) error� Vtype.func("".tlsClientCon, time.Time) error� D"".(*tlsClientCon).SetReadDeadline� +>"".tlsClientCon.SetReadDeadline� +8go.string."SetWriteDeadline"� +4type.func(time.Time) error� +Vtype.func("".tlsClientCon, time.Time) error� +F"".(*tlsClientCon).SetWriteDeadline� +@"".tlsClientCon.SetWriteDeadline� +4go.string."VerifyHostname"� .type.func(string) error� Ptype.func("".tlsClientCon, string) error� B"".(*tlsClientCon).VerifyHostname� <"".tlsClientCon.VerifyHostname� "go.string."Write"� >type.func([]uint8) (int, error)� `type.func("".tlsClientCon, []uint8) (int, error)� 0"".(*tlsClientCon).Write� *"".tlsClientCon.Write� 6go.string."clientHandshake"� 2go.importpath.crypto/tls.� "type.func() error� @type.func("".tlsClientCon) error� Z"".(*tlsClientCon).crypto/tls.clientHandshake� T"".tlsClientCon.crypto/tls.clientHandshake� 2go.string."decryptTicket"� 2go.importpath.crypto/tls.� ftype.func([]uint8) (*crypto/tls.sessionState, bool)� �type.func("".tlsClientCon, []uint8) (*crypto/tls.sessionState, bool)� V"".(*tlsClientCon).crypto/tls.decryptTicket� P"".tlsClientCon.crypto/tls.decryptTicket� 2go.string."encryptTicket"�2go.importpath.crypto/tls.�htype.func(*crypto/tls.sessionState) ([]uint8, error)��type.func("".tlsClientCon, *crypto/tls.sessionState) ([]uint8, error)�V"".(*tlsClientCon).crypto/tls.encryptTicket�P"".tlsClientCon.crypto/tls.encryptTicket�2go.string."readHandshake"�2go.importpath.crypto/tls.�Btype.func() (interface {}, error)�`type.func("".tlsClientCon) (interface {}, error)�V"".(*tlsClientCon).crypto/tls.readHandshake�P"".tlsClientCon.crypto/tls.readHandshake�,go.string."readRecord"�2go.importpath.crypto/tls.�Ltype.func(crypto/tls.recordType) error�ntype.func("".tlsClientCon, crypto/tls.recordType) error�P"".(*tlsClientCon).crypto/tls.readRecord�J"".tlsClientCon.crypto/tls.readRecord�*go.string."sendAlert"�2go.importpath.crypto/tls.�Btype.func(crypto/tls.alert) error�dtype.func("".tlsClientCon, crypto/tls.alert) error�N"".(*tlsClientCon).crypto/tls.sendAlert�H"".tlsClientCon.crypto/tls.sendAlert�6go.string."sendAlertLocked"�2go.importpath.crypto/tls.�Btype.func(crypto/tls.alert) error�dtype.func("".tlsClientCon, crypto/tls.alert) error�Z"".(*tlsClientCon).crypto/tls.sendAlertLocked�T"".tlsClientCon.crypto/tls.sendAlertLocked�6go.string."serverHandshake"�2go.importpath.crypto/tls.�"type.func() error�@type.func("".tlsClientCon) error�Z"".(*tlsClientCon).crypto/tls.serverHandshake�T"".tlsClientCon.crypto/tls.serverHandshake�4go.string."tryCipherSuite"�2go.importpath.crypto/tls.��type.func(uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite��type.func("".tlsClientCon, uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite�X"".(*tlsClientCon).crypto/tls.tryCipherSuite�R"".tlsClientCon.crypto/tls.tryCipherSuite�.go.string."writeRecord"�2go.importpath.crypto/tls.�ltype.func(crypto/tls.recordType, []uint8) (int, error)��type.func("".tlsClientCon, crypto/tls.recordType, []uint8) (int, error)�R"".(*tlsClientCon).crypto/tls.writeRecord�L"".tlsClientCon.crypto/tls.writeRecord�@go.string."*docker.tlsClientCon"PJ*docker.tlsClientCon @go.string."*docker.tlsClientCon"�Xgo.string."func(*docker.tlsClientCon) error"pb func(*docker.tlsClientCon) error Xgo.string."func(*docker.tlsClientCon) error"�Btype.func(*"".tlsClientCon) error��S�}3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PXgo.string."func(*docker.tlsClientCon) error"pTgo.weak.type.*func(*"".tlsClientCon) error�"runtime.zerovalue��Btype.func(*"".tlsClientCon) error��Btype.func(*"".tlsClientCon) error�*type.*"".tlsClientCon�type.error�tgo.string."func(*docker.tlsClientCon) tls.ConnectionState"�~.func(*docker.tlsClientCon) tls.ConnectionState tgo.string."func(*docker.tlsClientCon) tls.ConnectionState"�ltype.func(*"".tlsClientCon) crypto/tls.ConnectionState��V�3�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ptgo.string."func(*docker.tlsClientCon) tls.ConnectionState"p~go.weak.type.*func(*"".tlsClientCon) crypto/tls.ConnectionState�"runtime.zerovalue��ltype.func(*"".tlsClientCon) crypto/tls.ConnectionState��ltype.func(*"".tlsClientCon) crypto/tls.ConnectionState�*type.*"".tlsClientCon�>type.crypto/tls.ConnectionState�^go.string."func(*docker.tlsClientCon) net.Addr"ph#func(*docker.tlsClientCon) net.Addr ^go.string."func(*docker.tlsClientCon) net.Addr"�Htype.func(*"".tlsClientCon) net.Addr��;�,93 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P^go.string."func(*docker.tlsClientCon) net.Addr"pZgo.weak.type.*func(*"".tlsClientCon) net.Addr�"runtime.zerovalue��Htype.func(*"".tlsClientCon) net.Addr��Htype.func(*"".tlsClientCon) net.Addr�*type.*"".tlsClientCon�type.net.Addr�\go.string."func(*docker.tlsClientCon) []uint8"pf"func(*docker.tlsClientCon) []uint8 \go.string."func(*docker.tlsClientCon) []uint8"�Ftype.func(*"".tlsClientCon) []uint8��=��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P\go.string."func(*docker.tlsClientCon) []uint8"pXgo.weak.type.*func(*"".tlsClientCon) []uint8�"runtime.zerovalue��Ftype.func(*"".tlsClientCon) []uint8��Ftype.func(*"".tlsClientCon) []uint8�*type.*"".tlsClientCon�type.[]uint8�xgo.string."func(*docker.tlsClientCon, []uint8) (int, error)"��0func(*docker.tlsClientCon, []uint8) (int, error) xgo.string."func(*docker.tlsClientCon, []uint8) (int, error)"�btype.func(*"".tlsClientCon, []uint8) (int, error)����j�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pxgo.string."func(*docker.tlsClientCon, []uint8) (int, error)"ptgo.weak.type.*func(*"".tlsClientCon, []uint8) (int, error)�"runtime.zerovalue��btype.func(*"".tlsClientCon, []uint8) (int, error)��btype.func(*"".tlsClientCon, []uint8) (int, error)�*type.*"".tlsClientCon�type.[]uint8�type.int�type.error�ngo.string."func(*docker.tlsClientCon, time.Time) error"�x+func(*docker.tlsClientCon, time.Time) error ngo.string."func(*docker.tlsClientCon, time.Time) error"�Xtype.func(*"".tlsClientCon, time.Time) error���ٗ^3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pngo.string."func(*docker.tlsClientCon, time.Time) error"pjgo.weak.type.*func(*"".tlsClientCon, time.Time) error�"runtime.zerovalue��Xtype.func(*"".tlsClientCon, time.Time) error��Xtype.func(*"".tlsClientCon, time.Time) error�*type.*"".tlsClientCon�type.time.Time�type.error�hgo.string."func(*docker.tlsClientCon, string) error"�r(func(*docker.tlsClientCon, string) error hgo.string."func(*docker.tlsClientCon, string) error"�Rtype.func(*"".tlsClientCon, string) error���R*3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Phgo.string."func(*docker.tlsClientCon, string) error"pdgo.weak.type.*func(*"".tlsClientCon, string) error�"runtime.zerovalue��Rtype.func(*"".tlsClientCon, string) error��Rtype.func(*"".tlsClientCon, string) error�*type.*"".tlsClientCon�type.string�type.error��go.string."func(*docker.tlsClientCon, []uint8) (*tls.sessionState, bool)"��=func(*docker.tlsClientCon, []uint8) (*tls.sessionState, bool) �go.string."func(*docker.tlsClientCon, []uint8) (*tls.sessionState, bool)"��type.func(*"".tlsClientCon, []uint8) (*crypto/tls.sessionState, bool)��{�7x3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.tlsClientCon, []uint8) (*tls.sessionState, bool)"p�go.weak.type.*func(*"".tlsClientCon, []uint8) (*crypto/tls.sessionState, bool)�"runtime.zerovalue���type.func(*"".tlsClientCon, []uint8) (*crypto/tls.sessionState, bool)���type.func(*"".tlsClientCon, []uint8) (*crypto/tls.sessionState, bool)�*type.*"".tlsClientCon�type.[]uint8�:type.*crypto/tls.sessionState�type.bool��go.string."func(*docker.tlsClientCon, *tls.sessionState) ([]uint8, error)"��>func(*docker.tlsClientCon, *tls.sessionState) ([]uint8, error) �go.string."func(*docker.tlsClientCon, *tls.sessionState) ([]uint8, error)"��type.func(*"".tlsClientCon, *crypto/tls.sessionState) ([]uint8, error)��`��O3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.tlsClientCon, *tls.sessionState) ([]uint8, error)"p�go.weak.type.*func(*"".tlsClientCon, *crypto/tls.sessionState) ([]uint8, error)�"runtime.zerovalue���type.func(*"".tlsClientCon, *crypto/tls.sessionState) ([]uint8, error)���type.func(*"".tlsClientCon, *crypto/tls.sessionState) ([]uint8, error)�*type.*"".tlsClientCon�:type.*crypto/tls.sessionState�type.[]uint8�type.error�xgo.string."func(*docker.tlsClientCon) (interface {}, error)"��0func(*docker.tlsClientCon) (interface {}, error) xgo.string."func(*docker.tlsClientCon) (interface {}, error)"�btype.func(*"".tlsClientCon) (interface {}, error)���M��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pxgo.string."func(*docker.tlsClientCon) (interface {}, error)"ptgo.weak.type.*func(*"".tlsClientCon) (interface {}, error)�"runtime.zerovalue��btype.func(*"".tlsClientCon) (interface {}, error)��btype.func(*"".tlsClientCon) (interface {}, error)�*type.*"".tlsClientCon�"type.interface {}�type.error�xgo.string."func(*docker.tlsClientCon, tls.recordType) error"��0func(*docker.tlsClientCon, tls.recordType) error xgo.string."func(*docker.tlsClientCon, tls.recordType) error"�ptype.func(*"".tlsClientCon, crypto/tls.recordType) error��J���3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pxgo.string."func(*docker.tlsClientCon, tls.recordType) error"p�go.weak.type.*func(*"".tlsClientCon, crypto/tls.recordType) error�"runtime.zerovalue��ptype.func(*"".tlsClientCon, crypto/tls.recordType) error��ptype.func(*"".tlsClientCon, crypto/tls.recordType) error�*type.*"".tlsClientCon�4type.crypto/tls.recordType�type.error�ngo.string."func(*docker.tlsClientCon, tls.alert) error"�x+func(*docker.tlsClientCon, tls.alert) error ngo.string."func(*docker.tlsClientCon, tls.alert) error"�ftype.func(*"".tlsClientCon, crypto/tls.alert) error���yb.3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pngo.string."func(*docker.tlsClientCon, tls.alert) error"pxgo.weak.type.*func(*"".tlsClientCon, crypto/tls.alert) error�"runtime.zerovalue��ftype.func(*"".tlsClientCon, crypto/tls.alert) error��ftype.func(*"".tlsClientCon, crypto/tls.alert) error�*type.*"".tlsClientCon�*type.crypto/tls.alert�type.error��go.string."func(*docker.tlsClientCon, uint16, []uint16, uint16, bool, bool) *tls.cipherSuite"��Qfunc(*docker.tlsClientCon, uint16, []uint16, uint16, bool, bool) *tls.cipherSuite �go.string."func(*docker.tlsClientCon, uint16, []uint16, uint16, bool, bool) *tls.cipherSuite"��type.func(*"".tlsClientCon, uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite���X��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.tlsClientCon, uint16, []uint16, uint16, bool, bool) *tls.cipherSuite"p�go.weak.type.*func(*"".tlsClientCon, uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite�"runtime.zerovalue���type.func(*"".tlsClientCon, uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite���type.func(*"".tlsClientCon, uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite�*type.*"".tlsClientCon�type.uint16�type.[]uint16�type.uint16�type.bool�type.bool�8type.*crypto/tls.cipherSuite��go.string."func(*docker.tlsClientCon, tls.recordType, []uint8) (int, error)"��@func(*docker.tlsClientCon, tls.recordType, []uint8) (int, error) �go.string."func(*docker.tlsClientCon, tls.recordType, []uint8) (int, error)"��type.func(*"".tlsClientCon, crypto/tls.recordType, []uint8) (int, error)���BB�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*docker.tlsClientCon, tls.recordType, []uint8) (int, error)"p�go.weak.type.*func(*"".tlsClientCon, crypto/tls.recordType, []uint8) (int, error)�"runtime.zerovalue���type.func(*"".tlsClientCon, crypto/tls.recordType, []uint8) (int, error)���type.func(*"".tlsClientCon, crypto/tls.recordType, []uint8) (int, error)�*type.*"".tlsClientCon�4type.crypto/tls.recordType�type.[]uint8�type.int�type.error�*type.*"".tlsClientCon��&�}�6� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P@go.string."*docker.tlsClientCon"p"".(*tlsClientCon).OCSPResponse�>"".(*tlsClientCon).OCSPResponse� go.string."Read"�>type.func([]uint8) (int, error)�btype.func(*"".tlsClientCon, []uint8) (int, error)�."".(*tlsClientCon).Read�."".(*tlsClientCon).Read�,go.string."RemoteAddr"�(type.func() net.Addr�Htype.func(*"".tlsClientCon) net.Addr�:"".(*tlsClientCon).RemoteAddr�:"".(*tlsClientCon).RemoteAddr�.go.string."SetDeadline"�4type.func(time.Time) error�Xtype.func(*"".tlsClientCon, time.Time) error�<"".(*tlsClientCon).SetDeadline�<"".(*tlsClientCon).SetDeadline�6go.string."SetReadDeadline"�4type.func(time.Time) error� Xtype.func(*"".tlsClientCon, time.Time) error� D"".(*tlsClientCon).SetReadDeadline� D"".(*tlsClientCon).SetReadDeadline� 8go.string."SetWriteDeadline"� 4type.func(time.Time) error� Xtype.func(*"".tlsClientCon, time.Time) error� F"".(*tlsClientCon).SetWriteDeadline� +F"".(*tlsClientCon).SetWriteDeadline� +4go.string."VerifyHostname"� +.type.func(string) error� +Rtype.func(*"".tlsClientCon, string) error� +B"".(*tlsClientCon).VerifyHostname� +B"".(*tlsClientCon).VerifyHostname� +"go.string."Write"� >type.func([]uint8) (int, error)� btype.func(*"".tlsClientCon, []uint8) (int, error)� 0"".(*tlsClientCon).Write� 0"".(*tlsClientCon).Write� 6go.string."clientHandshake"� 2go.importpath.crypto/tls.� "type.func() error� Btype.func(*"".tlsClientCon) error� Z"".(*tlsClientCon).crypto/tls.clientHandshake� Z"".(*tlsClientCon).crypto/tls.clientHandshake� 2go.string."decryptTicket"� 2go.importpath.crypto/tls.� ftype.func([]uint8) (*crypto/tls.sessionState, bool)� �type.func(*"".tlsClientCon, []uint8) (*crypto/tls.sessionState, bool)� V"".(*tlsClientCon).crypto/tls.decryptTicket� V"".(*tlsClientCon).crypto/tls.decryptTicket� 2go.string."encryptTicket"� 2go.importpath.crypto/tls.� htype.func(*crypto/tls.sessionState) ([]uint8, error)� �type.func(*"".tlsClientCon, *crypto/tls.sessionState) ([]uint8, error)� V"".(*tlsClientCon).crypto/tls.encryptTicket� V"".(*tlsClientCon).crypto/tls.encryptTicket� 2go.string."readHandshake"�2go.importpath.crypto/tls.�Btype.func() (interface {}, error)�btype.func(*"".tlsClientCon) (interface {}, error)�V"".(*tlsClientCon).crypto/tls.readHandshake�V"".(*tlsClientCon).crypto/tls.readHandshake�,go.string."readRecord"�2go.importpath.crypto/tls.�Ltype.func(crypto/tls.recordType) error�ptype.func(*"".tlsClientCon, crypto/tls.recordType) error�P"".(*tlsClientCon).crypto/tls.readRecord�P"".(*tlsClientCon).crypto/tls.readRecord�*go.string."sendAlert"�2go.importpath.crypto/tls.�Btype.func(crypto/tls.alert) error�ftype.func(*"".tlsClientCon, crypto/tls.alert) error�N"".(*tlsClientCon).crypto/tls.sendAlert�N"".(*tlsClientCon).crypto/tls.sendAlert�6go.string."sendAlertLocked"�2go.importpath.crypto/tls.�Btype.func(crypto/tls.alert) error�ftype.func(*"".tlsClientCon, crypto/tls.alert) error�Z"".(*tlsClientCon).crypto/tls.sendAlertLocked�Z"".(*tlsClientCon).crypto/tls.sendAlertLocked�6go.string."serverHandshake"�2go.importpath.crypto/tls.�"type.func() error�Btype.func(*"".tlsClientCon) error�Z"".(*tlsClientCon).crypto/tls.serverHandshake�Z"".(*tlsClientCon).crypto/tls.serverHandshake�4go.string."tryCipherSuite"�2go.importpath.crypto/tls.��type.func(uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite��type.func(*"".tlsClientCon, uint16, []uint16, uint16, bool, bool) *crypto/tls.cipherSuite�X"".(*tlsClientCon).crypto/tls.tryCipherSuite�X"".(*tlsClientCon).crypto/tls.tryCipherSuite�.go.string."writeRecord"�2go.importpath.crypto/tls.�ltype.func(crypto/tls.recordType, []uint8) (int, error)��type.func(*"".tlsClientCon, crypto/tls.recordType, []uint8) (int, error)�R"".(*tlsClientCon).crypto/tls.writeRecord�R"".(*tlsClientCon).crypto/tls.writeRecord�bruntime.gcbits.0x84000000000000000000000000000000 ��`go.string."struct { F uintptr; A0 *chan error }"pj$struct { F uintptr; A0 *chan error } `go.string."struct { F uintptr; A0 *chan error }"�Rtype.struct { F uintptr; A0 *chan error }��|Y"� � runtime.algarray0bruntime.gcbits.0x84000000000000000000000000000000P`go.string."struct { F uintptr; A0 *chan error }"pdgo.weak.type.*struct { F uintptr; A0 *chan error }�"runtime.zerovalue��Rtype.struct { F uintptr; A0 *chan error }�go.string."F"�type.uintptr�go.string."A0"� type.*chan error�,go.string."**tls.Conn"@6 +**tls.Conn ,go.string."**tls.Conn"�,type.**crypto/tls.Conn����]6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."**tls.Conn"p>go.weak.type.***crypto/tls.Conn�"runtime.zerovalue�*type.*crypto/tls.Conn�~go.string."struct { F uintptr; A0 *chan error; A1 **tls.Conn }"��3struct { F uintptr; A0 *chan error; A1 **tls.Conn } ~go.string."struct { F uintptr; A0 *chan error; A1 **tls.Conn }"�~type.struct { F uintptr; A0 *chan error; A1 **crypto/tls.Conn }���%�F  runtime.algarray0bruntime.gcbits.0x84488800000000000000000000000000P~go.string."struct { F uintptr; A0 *chan error; A1 **tls.Conn }"p�go.weak.type.*struct { F uintptr; A0 *chan error; A1 **crypto/tls.Conn }�"runtime.zerovalue��~type.struct { F uintptr; A0 *chan error; A1 **crypto/tls.Conn }�go.string."F"�type.uintptr�go.string."A0"� type.*chan error�go.string."A1"�,type.**crypto/tls.Conn�bgo.string."*struct { F uintptr; A0 *chan error }"pl%*struct { F uintptr; A0 *chan error } bgo.string."*struct { F uintptr; A0 *chan error }"�Ttype.*struct { F uintptr; A0 *chan error }���t6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pbgo.string."*struct { F uintptr; A0 *chan error }"pfgo.weak.type.**struct { F uintptr; A0 *chan error }�"runtime.zerovalue�Rtype.struct { F uintptr; A0 *chan error }��go.string."*struct { F uintptr; A0 *chan error; A1 **tls.Conn }"��4*struct { F uintptr; A0 *chan error; A1 **tls.Conn } �go.string."*struct { F uintptr; A0 *chan error; A1 **tls.Conn }"��type.*struct { F uintptr; A0 *chan error; A1 **crypto/tls.Conn }���*�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."*struct { F uintptr; A0 *chan error; A1 **tls.Conn }"p�go.weak.type.**struct { F uintptr; A0 *chan error; A1 **crypto/tls.Conn }�"runtime.zerovalue�~type.struct { F uintptr; A0 *chan error; A1 **crypto/tls.Conn }�bruntime.gcbits.0x88444888444800000000000000000000 �DH�DH�""..gostring.14��p[2]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 } ""..gostring.14��type.[2]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }��`��*� � runtime.algarray0bruntime.gcbits.0x88444888444800000000000000000000P""..gostring.14p�go.weak.type.*[2]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }�"runtime.zerovalue��type.struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }��type.[]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }��go.typelink.[2]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }/[2]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }�type.[2]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }�*go.string."[2]*uint8"@4 [2]*uint8 *go.string."[2]*uint8"�type.[2]*uint8����V � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P*go.string."[2]*uint8"p.go.weak.type.*[2]*uint8�"runtime.zerovalue�type.*uint8�type.[]*uint8�>go.typelink.[2]*uint8/[2]*uint8type.[2]*uint8�*go.string."[2]uint16"@4 [2]uint16 *go.string."[2]uint16"�type.[2]uint16�� �UI� � runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P*go.string."[2]uint16"p.go.weak.type.*[2]uint16�"runtime.zerovalue�type.uint16�type.[]uint16�>go.typelink.[2]uint16/[2]uint16type.[2]uint16�bruntime.gcbits.0x84884884844884844800000000000000 ��H��H��H�""..gostring.15���struct { tcase uint16; ncase uint16; pollorder *uint8; lockorder *uint8; scase [2]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }; lockorderarr [2]*uint8; pollorderarr [2]uint16 } ""..gostring.15��type.struct { tcase uint16; ncase uint16; pollorder *uint8; lockorder *uint8; scase [2]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }; lockorderarr [2]*uint8; pollorderarr [2]uint16 }�����x�x�6 � runtime.algarray0bruntime.gcbits.0x84884884844884844800000000000000P""..gostring.15p�go.weak.type.*struct { tcase uint16; ncase uint16; pollorder *uint8; lockorder *uint8; scase [2]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }; lockorderarr [2]*uint8; pollorderarr [2]uint16 }�"runtime.zerovalue���type.struct { tcase uint16; ncase uint16; pollorder *uint8; lockorder *uint8; scase [2]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }; lockorderarr [2]*uint8; pollorderarr [2]uint16 }�"go.string."tcase"�"go.importpath."".�type.uint16�"go.string."ncase"�"go.importpath."".�type.uint16�*go.string."pollorder"�"go.importpath."".�type.*uint8�*go.string."lockorder"�"go.importpath."".�type.*uint8�"go.string."scase"�"go.importpath."".��type.[2]struct { elem *uint8; chan *uint8; pc uintptr; kind uint16; so uint16; receivedp *uint8; releasetime uint64 }�0go.string."lockorderarr"�"go.importpath."".�type.[2]*uint8�0go.string."pollorderarr"�"go.importpath."".�type.[2]uint16�,go.string."*[8]string"@6 +*[8]string ,go.string."*[8]string"�type.*[8]string����o6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."*[8]string"p0go.weak.type.**[8]string�"runtime.zerovalue�type.[8]string�Pgo.string."*[8]docker.AuthConfiguration"`Z*[8]docker.AuthConfiguration Pgo.string."*[8]docker.AuthConfiguration"�:type.*[8]"".AuthConfiguration��B�U|6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."*[8]docker.AuthConfiguration"pLgo.weak.type.**[8]"".AuthConfiguration�"runtime.zerovalue�8type.[8]"".AuthConfiguration�Fgo.string."*[8]docker.dockerConfig"PP*[8]docker.dockerConfig Fgo.string."*[8]docker.dockerConfig"�0type.*[8]"".dockerConfig���)u�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PFgo.string."*[8]docker.dockerConfig"pBgo.weak.type.**[8]"".dockerConfig�"runtime.zerovalue�.type.[8]"".dockerConfig�6go.string."*[8]docker.Port"@@*[8]docker.Port 6go.string."*[8]docker.Port"� type.*[8]"".Port��D18�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."*[8]docker.Port"p2go.weak.type.**[8]"".Port�"runtime.zerovalue�type.[8]"".Port�8go.string."*[8]interface {}"PB*[8]interface {} 8go.string."*[8]interface {}"�*type.*[8]interface {}���aK6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P8go.string."*[8]interface {}"ptype..hash."".AuthConfiguration�$runtime.strhash·fruntime.strhash�@type..eq."".AuthConfiguration·f:type..eq."".AuthConfiguration�.type..hash.[8]string·f(type..hash.[8]string�*type..eq.[8]string·f$type..eq.[8]string�Jtype..hash.[8]"".AuthConfiguration·fDtype..hash.[8]"".AuthConfiguration�Ftype..eq.[8]"".AuthConfiguration·f@type..eq.[8]"".AuthConfiguration�.type..hash.[3]string·f(type..hash.[3]string�*type..eq.[3]string·f$type..eq.[3]string�.type..hash.[2]string·f(type..hash.[2]string�*type..eq.[2]string·f$type..eq.[2]string�:type..hash."".dockerConfig·f4type..hash."".dockerConfig�6type..eq."".dockerConfig·f0type..eq."".dockerConfig�@type..hash.[8]"".dockerConfig·f:type..hash.[8]"".dockerConfig�"".(*eventMonitoringState).Lock�F"".(*eventMonitoringState).RLock·f@"".(*eventMonitoringState).RLock�J"".(*eventMonitoringState).RLocker·fD"".(*eventMonitoringState).RLocker�J"".(*eventMonitoringState).RUnlock·fD"".(*eventMonitoringState).RUnlock�H"".(*eventMonitoringState).Unlock·fB"".(*eventMonitoringState).Unlock�B"".(*eventMonitoringState).Add·f<"".(*eventMonitoringState).Add�D"".(*eventMonitoringState).Done·f>"".(*eventMonitoringState).Done�D"".(*eventMonitoringState).Wait·f>"".(*eventMonitoringState).Wait�4"".(*APIVersion).String·f."".(*APIVersion).String�(runtime.panicwrap·f"runtime.panicwrap�8"".(*APIVersion).LessThan·f2"".(*APIVersion).LessThan�J"".(*APIVersion).LessThanOrEqualTo·fD"".(*APIVersion).LessThanOrEqualTo�>"".(*APIVersion).GreaterThan·f8"".(*APIVersion).GreaterThan�P"".(*APIVersion).GreaterThanOrEqualTo·fJ"".(*APIVersion).GreaterThanOrEqualTo�6"".(*APIVersion).compare·f0"".(*APIVersion).compare�Rtype..hash."".AttachToContainerOptions·fLtype..hash."".AttachToContainerOptions�(runtime.interhash·f"runtime.interhash�Ntype..eq."".AttachToContainerOptions·fHtype..eq."".AttachToContainerOptions�&runtime.memequal·f runtime.memequal�$"".(*Port).Port·f"".(*Port).Port�&"".(*Port).Proto·f "".(*Port).Proto�0type..hash.[8]"".Port·f*type..hash.[8]"".Port�,type..eq.[8]"".Port·f&type..eq.[8]"".Port�Ntype..hash."".CommitContainerOptions·fHtype..hash."".CommitContainerOptions�Jtype..eq."".CommitContainerOptions·fDtype..eq."".CommitContainerOptions�.type..hash."".Change·f(type..hash."".Change�*type..eq."".Change·f$type..eq."".Change�Rtype..hash."".CopyFromContainerOptions·fLtype..hash."".CopyFromContainerOptions�Ntype..eq."".CopyFromContainerOptions·fHtype..eq."".CopyFromContainerOptions�:type..hash."".KeyValuePair·f4type..hash."".KeyValuePair�6type..eq."".KeyValuePair·f0type..eq."".KeyValuePair�8type..hash."".PortBinding·f2type..hash."".PortBinding�4type..eq."".PortBinding·f.type..eq."".PortBinding�type..eq."".PullImageOptions·f8type..eq."".PullImageOptions�Btype..hash."".PushImageOptions·ftype..eq."".PushImageOptions·f8type..eq."".PushImageOptions�Ntype..hash."".RemoveContainerOptions·fHtype..hash."".RemoveContainerOptions�Jtype..eq."".RemoveContainerOptions·fDtype..eq."".RemoveContainerOptions�Ntype..hash."".RenameContainerOptions·fHtype..hash."".RenameContainerOptions�Jtype..eq."".RenameContainerOptions·fDtype..eq."".RenameContainerOptions�>type..hash."".APIImageSearch·f8type..hash."".APIImageSearch�:type..eq."".APIImageSearch·f4type..eq."".APIImageSearch�Btype..hash."".StartExecOptions·ftype..eq."".StartExecOptions·f8type..eq."".StartExecOptions�@type..hash."".BlkioStatsEntry·f:type..hash."".BlkioStatsEntry�"".tlsClientCon.ConnectionState�Jcrypto/tls.(*Conn).ConnectionState·fDcrypto/tls.(*Conn).ConnectionState�>"".(*tlsClientCon).Handshake·f8"".(*tlsClientCon).Handshake�8"".tlsClientCon.Handshake·f2"".tlsClientCon.Handshake�>"".(*tlsClientCon).LocalAddr·f8"".(*tlsClientCon).LocalAddr�8"".tlsClientCon.LocalAddr·f2"".tlsClientCon.LocalAddr�>crypto/tls.(*Conn).LocalAddr·f8crypto/tls.(*Conn).LocalAddr�D"".(*tlsClientCon).OCSPResponse·f>"".(*tlsClientCon).OCSPResponse�>"".tlsClientCon.OCSPResponse·f8"".tlsClientCon.OCSPResponse�Dcrypto/tls.(*Conn).OCSPResponse·f>crypto/tls.(*Conn).OCSPResponse�4"".(*tlsClientCon).Read·f."".(*tlsClientCon).Read�."".tlsClientCon.Read·f("".tlsClientCon.Read�4crypto/tls.(*Conn).Read·f.crypto/tls.(*Conn).Read�@"".(*tlsClientCon).RemoteAddr·f:"".(*tlsClientCon).RemoteAddr�:"".tlsClientCon.RemoteAddr·f4"".tlsClientCon.RemoteAddr�@crypto/tls.(*Conn).RemoteAddr·f:crypto/tls.(*Conn).RemoteAddr�B"".(*tlsClientCon).SetDeadline·f<"".(*tlsClientCon).SetDeadline�<"".tlsClientCon.SetDeadline·f6"".tlsClientCon.SetDeadline�Bcrypto/tls.(*Conn).SetDeadline·f"".tlsClientCon.SetReadDeadline�Jcrypto/tls.(*Conn).SetReadDeadline·fDcrypto/tls.(*Conn).SetReadDeadline�L"".(*tlsClientCon).SetWriteDeadline·fF"".(*tlsClientCon).SetWriteDeadline�F"".tlsClientCon.SetWriteDeadline·f@"".tlsClientCon.SetWriteDeadline�Lcrypto/tls.(*Conn).SetWriteDeadline·fFcrypto/tls.(*Conn).SetWriteDeadline�H"".(*tlsClientCon).VerifyHostname·fB"".(*tlsClientCon).VerifyHostname�B"".tlsClientCon.VerifyHostname·f<"".tlsClientCon.VerifyHostname�Hcrypto/tls.(*Conn).VerifyHostname·fBcrypto/tls.(*Conn).VerifyHostname�6"".(*tlsClientCon).Write·f0"".(*tlsClientCon).Write�0"".tlsClientCon.Write·f*"".tlsClientCon.Write�6crypto/tls.(*Conn).Write·f0crypto/tls.(*Conn).Write�`"".(*tlsClientCon).crypto/tls.clientHandshake·fZ"".(*tlsClientCon).crypto/tls.clientHandshake�Z"".tlsClientCon.crypto/tls.clientHandshake·fT"".tlsClientCon.crypto/tls.clientHandshake�Jcrypto/tls.(*Conn).clientHandshake·fDcrypto/tls.(*Conn).clientHandshake�\"".(*tlsClientCon).crypto/tls.decryptTicket·fV"".(*tlsClientCon).crypto/tls.decryptTicket�V"".tlsClientCon.crypto/tls.decryptTicket·fP"".tlsClientCon.crypto/tls.decryptTicket�Fcrypto/tls.(*Conn).decryptTicket·f@crypto/tls.(*Conn).decryptTicket�\"".(*tlsClientCon).crypto/tls.encryptTicket·fV"".(*tlsClientCon).crypto/tls.encryptTicket�V"".tlsClientCon.crypto/tls.encryptTicket·fP"".tlsClientCon.crypto/tls.encryptTicket�Fcrypto/tls.(*Conn).encryptTicket·f@crypto/tls.(*Conn).encryptTicket�\"".(*tlsClientCon).crypto/tls.readHandshake·fV"".(*tlsClientCon).crypto/tls.readHandshake�V"".tlsClientCon.crypto/tls.readHandshake·fP"".tlsClientCon.crypto/tls.readHandshake�Fcrypto/tls.(*Conn).readHandshake·f@crypto/tls.(*Conn).readHandshake�V"".(*tlsClientCon).crypto/tls.readRecord·fP"".(*tlsClientCon).crypto/tls.readRecord�P"".tlsClientCon.crypto/tls.readRecord·fJ"".tlsClientCon.crypto/tls.readRecord�@crypto/tls.(*Conn).readRecord·f:crypto/tls.(*Conn).readRecord�T"".(*tlsClientCon).crypto/tls.sendAlert·fN"".(*tlsClientCon).crypto/tls.sendAlert�N"".tlsClientCon.crypto/tls.sendAlert·fH"".tlsClientCon.crypto/tls.sendAlert�>crypto/tls.(*Conn).sendAlert·f8crypto/tls.(*Conn).sendAlert�`"".(*tlsClientCon).crypto/tls.sendAlertLocked·fZ"".(*tlsClientCon).crypto/tls.sendAlertLocked�Z"".tlsClientCon.crypto/tls.sendAlertLocked·fT"".tlsClientCon.crypto/tls.sendAlertLocked�Jcrypto/tls.(*Conn).sendAlertLocked·fDcrypto/tls.(*Conn).sendAlertLocked�`"".(*tlsClientCon).crypto/tls.serverHandshake·fZ"".(*tlsClientCon).crypto/tls.serverHandshake�Z"".tlsClientCon.crypto/tls.serverHandshake·fT"".tlsClientCon.crypto/tls.serverHandshake�Jcrypto/tls.(*Conn).serverHandshake·fDcrypto/tls.(*Conn).serverHandshake�^"".(*tlsClientCon).crypto/tls.tryCipherSuite·fX"".(*tlsClientCon).crypto/tls.tryCipherSuite�X"".tlsClientCon.crypto/tls.tryCipherSuite·fR"".tlsClientCon.crypto/tls.tryCipherSuite�Hcrypto/tls.(*Conn).tryCipherSuite·fBcrypto/tls.(*Conn).tryCipherSuite�X"".(*tlsClientCon).crypto/tls.writeRecord·fR"".(*tlsClientCon).crypto/tls.writeRecord�R"".tlsClientCon.crypto/tls.writeRecord·fL"".tlsClientCon.crypto/tls.writeRecord�Bcrypto/tls.(*Conn).writeRecord·f +__.PKGDEF 0 0 0 644 26062 ` +go object darwin amd64 go1.4.2 X:precisestack + +$$ +package logrus + import log "log" + import sync "sync" + import runtime "runtime" + import bufio "bufio" + import time "time" + import io "io" + import os "os" + import strings "strings" + import syscall "syscall" + import fmt "fmt" + import sort "sort" + import json "encoding/json" + import bytes "bytes" + import unsafe "unsafe" + type @"io".Writer interface { Write(@"io".p []byte) (@"io".n int, @"io".err error) } + type @"".Level uint8 + func (@"".level·2 @"".Level) String () (? string) + type @"".Hook interface { Fire(? *@"".Entry) (? error); Levels() (? []@"".Level) } + type @"".LevelHooks map[@"".Level][]@"".Hook + func (@"".hooks·1 @"".LevelHooks "esc:0x0") Add (@"".hook·2 @"".Hook) + func (@"".hooks·2 @"".LevelHooks "esc:0x0") Fire (@"".level·3 @"".Level, @"".entry·4 *@"".Entry) (? error) + type @"".Formatter interface { Format(? *@"".Entry) (? []byte, ? error) } + type @"sync".Mutex struct { @"sync".state int32; @"sync".sema uint32 } + func (@"sync".m·1 *@"sync".Mutex) Lock () + func (@"sync".m·1 *@"sync".Mutex) Unlock () + type @"".Fields map[string]interface {} + type @"sync".Locker interface { Lock(); Unlock() } + type @"sync".syncSema struct { @"sync".lock uintptr; @"sync".head @"unsafe".Pointer; @"sync".tail @"unsafe".Pointer } + type @"sync".copyChecker uintptr + func (@"sync".c·1 *@"sync".copyChecker) @"sync".check () + type @"sync".Cond struct { L @"sync".Locker; @"sync".sema @"sync".syncSema; @"sync".waiters uint32; @"sync".checker @"sync".copyChecker } + func (@"sync".c·1 *@"sync".Cond) Broadcast () + func (@"sync".c·1 *@"sync".Cond) Signal () + func (@"sync".c·1 *@"sync".Cond) Wait () + func (@"sync".c·1 *@"sync".Cond) @"sync".signalImpl (@"sync".all·2 bool) + type @"io".pipe struct { @"io".rl @"sync".Mutex; @"io".wl @"sync".Mutex; @"io".l @"sync".Mutex; @"io".data []byte; @"io".rwait @"sync".Cond; @"io".wwait @"sync".Cond; @"io".rerr error; @"io".werr error } + func (@"io".p·1 *@"io".pipe) @"io".rclose (@"io".err·2 error) + func (@"io".p·3 *@"io".pipe) @"io".read (@"io".b·4 []byte "esc:0x0") (@"io".n·1 int, @"io".err·2 error) + func (@"io".p·1 *@"io".pipe) @"io".wclose (@"io".err·2 error) + func (@"io".p·3 *@"io".pipe) @"io".write (@"io".b·4 []byte) (@"io".n·1 int, @"io".err·2 error) + type @"io".PipeWriter struct { @"io".p *@"io".pipe } + func (@"io".w·2 *@"io".PipeWriter) Close () (? error) + func (@"io".w·2 *@"io".PipeWriter) CloseWithError (@"io".err·3 error) (? error) + func (@"io".w·3 *@"io".PipeWriter) Write (@"io".data·4 []byte) (@"io".n·1 int, @"io".err·2 error) + type @"io".PipeReader struct { @"io".p *@"io".pipe } + func (@"io".r·2 *@"io".PipeReader) Close () (? error) + func (@"io".r·2 *@"io".PipeReader) CloseWithError (@"io".err·3 error) (? error) + func (@"io".r·3 *@"io".PipeReader) Read (@"io".data·4 []byte "esc:0x0") (@"io".n·1 int, @"io".err·2 error) + type @"".Logger struct { Out @"io".Writer; Hooks @"".LevelHooks; Formatter @"".Formatter; Level @"".Level; @"".mu @"sync".Mutex } + func (@"".logger·1 *@"".Logger) Debug (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Debugf (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Debugln (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Error (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Errorf (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Errorln (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Fatal (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Fatalf (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Fatalln (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Info (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Infof (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Infoln (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Panic (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Panicf (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Panicln (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Print (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Printf (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Println (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Warn (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Warnf (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Warning (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Warningf (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Warningln (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·1 *@"".Logger) Warnln (@"".args·2 ...interface {} "esc:0x0") + func (@"".logger·2 *@"".Logger) WithField (@"".key·3 string, @"".value·4 interface {}) (? *@"".Entry) + func (@"".logger·2 *@"".Logger) WithFields (@"".fields·3 @"".Fields "esc:0x0") (? *@"".Entry) + func (@"".logger·2 *@"".Logger) Writer () (? *@"io".PipeWriter) + func (@"".logger·1 *@"".Logger) @"".writerScanner (@"".reader·2 *@"io".PipeReader) + type @"time".zone struct { @"time".name string; @"time".offset int; @"time".isDST bool } + type @"time".zoneTrans struct { @"time".when int64; @"time".index uint8; @"time".isstd bool; @"time".isutc bool } + type @"time".Location struct { @"time".name string; @"time".zone []@"time".zone; @"time".tx []@"time".zoneTrans; @"time".cacheStart int64; @"time".cacheEnd int64; @"time".cacheZone *@"time".zone } + func (@"time".l·2 *@"time".Location "esc:0x0") String () (? string) + func (@"time".l·2 *@"time".Location "esc:0x0") @"time".firstZoneUsed () (? bool) + func (@"time".l·2 *@"time".Location "esc:0x2") @"time".get () (? *@"time".Location) + func (@"time".l·6 *@"time".Location "esc:0x1") @"time".lookup (@"time".sec·7 int64) (@"time".name·1 string, @"time".offset·2 int, @"time".isDST·3 bool, @"time".start·4 int64, @"time".end·5 int64) + func (@"time".l·2 *@"time".Location "esc:0x0") @"time".lookupFirstZone () (? int) + func (@"time".l·4 *@"time".Location "esc:0x0") @"time".lookupName (@"time".name·5 string "esc:0x0", @"time".unix·6 int64) (@"time".offset·1 int, @"time".isDST·2 bool, @"time".ok·3 bool) + type @"time".Duration int64 + func (@"time".d·2 @"time".Duration) Hours () (? float64) { var @"time".hour·3 @"time".Duration; ; @"time".hour·3 = @"time".d·2 / @"time".Duration(0x34630B8A000); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0x34630B8A000); return float64(@"time".hour·3) + float64(@"time".nsec·4) * 0x9C5FFF26ED75Fp-93 } + func (@"time".d·2 @"time".Duration) Minutes () (? float64) { var @"time".min·3 @"time".Duration; ; @"time".min·3 = @"time".d·2 / @"time".Duration(0xDF8475800); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0xDF8475800); return float64(@"time".min·3) + float64(@"time".nsec·4) * 0x9299FF347E9E9p-87 } + func (@"time".d·2 @"time".Duration) Nanoseconds () (? int64) { return int64(@"time".d·2) } + func (@"time".d·2 @"time".Duration) Seconds () (? float64) { var @"time".sec·3 @"time".Duration; ; @"time".sec·3 = @"time".d·2 / @"time".Duration(0x3B9ACA00); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0x3B9ACA00); return float64(@"time".sec·3) + float64(@"time".nsec·4) * 0x112E0BE826D695p-82 } + func (@"time".d·2 @"time".Duration) String () (? string) + type @"time".Month int + func (@"time".m·2 @"time".Month) String () (? string) { return @"time".months[@"time".m·2 - @"time".Month(0x1)] } + type @"time".Weekday int + func (@"time".d·2 @"time".Weekday) String () (? string) { return @"time".days[@"time".d·2] } + type @"time".Time struct { @"time".sec int64; @"time".nsec int32; @"time".loc *@"time".Location } + func (@"time".t·2 @"time".Time "esc:0x2") Add (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x2") AddDate (@"time".years·3 int, @"time".months·4 int, @"time".days·5 int) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") After (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec > @"time".u·3.@"time".sec || @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec > @"time".u·3.@"time".nsec } + func (@"time".t·2 @"time".Time "esc:0x0") Before (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec < @"time".u·3.@"time".sec || @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec < @"time".u·3.@"time".nsec } + func (@"time".t·4 @"time".Time "esc:0x0") Clock () (@"time".hour·1 int, @"time".min·2 int, @"time".sec·3 int) + func (@"time".t·4 @"time".Time "esc:0x0") Date () (@"time".year·1 int, @"time".month·2 @"time".Month, @"time".day·3 int) + func (@"time".t·2 @"time".Time "esc:0x0") Day () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") Equal (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec == @"time".u·3.@"time".nsec } + func (@"time".t·2 @"time".Time "esc:0x0") Format (@"time".layout·3 string "esc:0x0") (? string) + func (@"time".t·2 *@"time".Time "esc:0x0") GobDecode (@"time".data·3 []byte "esc:0x0") (? error) + func (@"time".t·3 @"time".Time "esc:0x0") GobEncode () (? []byte, ? error) + func (@"time".t·2 @"time".Time "esc:0x0") Hour () (? int) + func (@"time".t·3 @"time".Time "esc:0x0") ISOWeek () (@"time".year·1 int, @"time".week·2 int) + func (@"time".t·2 @"time".Time "esc:0x2") In (@"time".loc·3 *@"time".Location "esc:0x2") (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") IsZero () (? bool) { return @"time".t·2.@"time".sec == 0x0 && @"time".t·2.@"time".nsec == 0x0 } + func (@"time".t·2 @"time".Time "esc:0x2") Local () (? @"time".Time) { @"time".t·2.@"time".loc = @"time".Local; return @"time".t·2 } + func (@"time".t·2 @"time".Time "esc:0x2") Location () (? *@"time".Location) { var @"time".l·3 *@"time".Location; ; @"time".l·3 = @"time".t·2.@"time".loc; if @"time".l·3 == nil { @"time".l·3 = @"time".UTC }; return @"time".l·3 } + func (@"time".t·3 @"time".Time "esc:0x0") MarshalBinary () (? []byte, ? error) + func (@"time".t·3 @"time".Time "esc:0x0") MarshalJSON () (? []byte, ? error) + func (@"time".t·3 @"time".Time "esc:0x0") MarshalText () (? []byte, ? error) + func (@"time".t·2 @"time".Time "esc:0x0") Minute () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") Month () (? @"time".Month) + func (@"time".t·2 @"time".Time "esc:0x0") Nanosecond () (? int) { return int(@"time".t·2.@"time".nsec) } + func (@"time".t·2 @"time".Time "esc:0x2") Round (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") Second () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") String () (? string) + func (@"time".t·2 @"time".Time "esc:0x0") Sub (@"time".u·3 @"time".Time "esc:0x0") (? @"time".Duration) + func (@"time".t·2 @"time".Time "esc:0x2") Truncate (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x2") UTC () (? @"time".Time) { @"time".t·2.@"time".loc = @"time".UTC; return @"time".t·2 } + func (@"time".t·2 @"time".Time "esc:0x0") Unix () (? int64) { return @"time".t·2.@"time".sec + -0xE7791F700 } + func (@"time".t·2 @"time".Time "esc:0x0") UnixNano () (? int64) { return (@"time".t·2.@"time".sec + -0xE7791F700) * 0x3B9ACA00 + int64(@"time".t·2.@"time".nsec) } + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalBinary (@"time".data·3 []byte "esc:0x0") (? error) + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalJSON (@"time".data·3 []byte "esc:0x0") (@"time".err·1 error) + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalText (@"time".data·3 []byte "esc:0x0") (@"time".err·1 error) + func (@"time".t·2 @"time".Time "esc:0x0") Weekday () (? @"time".Weekday) + func (@"time".t·2 @"time".Time "esc:0x0") Year () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") YearDay () (? int) + func (@"time".t·3 @"time".Time "esc:0x0") Zone () (@"time".name·1 string, @"time".offset·2 int) + func (@"time".t·2 @"time".Time "esc:0x0") @"time".abs () (? uint64) + func (@"time".t·5 @"time".Time "esc:0x0") @"time".date (@"time".full·6 bool) (@"time".year·1 int, @"time".month·2 @"time".Month, @"time".day·3 int, @"time".yday·4 int) + func (@"time".t·4 @"time".Time "esc:0x1") @"time".locabs () (@"time".name·1 string, @"time".offset·2 int, @"time".abs·3 uint64) + type @"bytes".readOp int + type @"io".Reader interface { Read(@"io".p []byte) (@"io".n int, @"io".err error) } + type @"bytes".Buffer struct { @"bytes".buf []byte; @"bytes".off int; @"bytes".runeBytes [4]byte; @"bytes".bootstrap [64]byte; @"bytes".lastRead @"bytes".readOp } + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x1") Bytes () (? []byte) { return @"bytes".b·2.@"bytes".buf[@"bytes".b·2.@"bytes".off:] } + func (@"bytes".b·1 *@"bytes".Buffer) Grow (@"bytes".n·2 int) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") Len () (? int) { return len(@"bytes".b·2.@"bytes".buf) - @"bytes".b·2.@"bytes".off } + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x1") Next (@"bytes".n·3 int) (? []byte) + func (@"bytes".b·3 *@"bytes".Buffer) Read (@"bytes".p·4 []byte "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) ReadByte () (@"bytes".c·1 byte, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x0") ReadBytes (@"bytes".delim·4 byte) (@"bytes".line·1 []byte, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) ReadFrom (@"bytes".r·4 @"io".Reader) (@"bytes".n·1 int64, @"bytes".err·2 error) + func (@"bytes".b·4 *@"bytes".Buffer) ReadRune () (@"bytes".r·1 rune, @"bytes".size·2 int, @"bytes".err·3 error) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x0") ReadString (@"bytes".delim·4 byte) (@"bytes".line·1 string, @"bytes".err·2 error) + func (@"bytes".b·1 *@"bytes".Buffer) Reset () + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") String () (? string) { if @"bytes".b·2 == nil { return "" }; return string(@"bytes".b·2.@"bytes".buf[@"bytes".b·2.@"bytes".off:]) } + func (@"bytes".b·1 *@"bytes".Buffer) Truncate (@"bytes".n·2 int) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") UnreadByte () (? error) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") UnreadRune () (? error) + func (@"bytes".b·3 *@"bytes".Buffer) Write (@"bytes".p·4 []byte "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·2 *@"bytes".Buffer) WriteByte (@"bytes".c·3 byte) (? error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteRune (@"bytes".r·4 rune) (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteString (@"bytes".s·4 string "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteTo (@"bytes".w·4 @"io".Writer) (@"bytes".n·1 int64, @"bytes".err·2 error) + func (@"bytes".b·2 *@"bytes".Buffer) @"bytes".grow (@"bytes".n·3 int) (? int) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x1") @"bytes".readSlice (@"bytes".delim·4 byte) (@"bytes".line·1 []byte, @"bytes".err·2 error) + type @"".Entry struct { Logger *@"".Logger; Data @"".Fields; Time @"time".Time; Level @"".Level; Message string } + func (@"".entry·1 *@"".Entry) Debug (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Debugf (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Debugln (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Error (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Errorf (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Errorln (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Fatal (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Fatalf (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Fatalln (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Info (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Infof (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Infoln (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Panic (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Panicf (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Panicln (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Print (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Printf (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Println (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·3 *@"".Entry) Reader () (? *@"bytes".Buffer, ? error) + func (@"".entry·3 *@"".Entry) String () (? string, ? error) + func (@"".entry·1 *@"".Entry) Warn (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Warnf (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Warning (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Warningf (@"".format·2 string "esc:0x0", @"".args·3 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Warningln (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·1 *@"".Entry) Warnln (@"".args·2 ...interface {} "esc:0x0") + func (@"".entry·2 *@"".Entry) WithField (@"".key·3 string, @"".value·4 interface {}) (? *@"".Entry) + func (@"".entry·2 *@"".Entry) WithFields (@"".fields·3 @"".Fields "esc:0x0") (? *@"".Entry) + func (@"".entry·1 *@"".Entry) @"".log (@"".level·2 @"".Level, @"".msg·3 string) + func (@"".entry·2 *@"".Entry "esc:0x0") @"".sprintlnn (@"".args·3 ...interface {} "esc:0x0") (? string) + func @"".NewEntry (@"".logger·2 *@"".Logger) (? *@"".Entry) { return (&@"".Entry{ Logger:@"".logger·2, Data:make(@"".Fields, 0x5) }) } + func @"".StandardLogger () (? *@"".Logger) { return @"".std } + func @"".SetOutput (@"".out·1 @"io".Writer) + func @"".SetFormatter (@"".formatter·1 @"".Formatter) + func @"".SetLevel (@"".level·1 @"".Level) + func @"".GetLevel () (? @"".Level) + func @"".AddHook (@"".hook·1 @"".Hook) + func @"".WithField (@"".key·2 string, @"".value·3 interface {}) (? *@"".Entry) + func @"".WithFields (@"".fields·2 @"".Fields "esc:0x0") (? *@"".Entry) + func @"".Debug (@"".args·1 ...interface {} "esc:0x0") + func @"".Print (@"".args·1 ...interface {} "esc:0x0") + func @"".Info (@"".args·1 ...interface {} "esc:0x0") + func @"".Warn (@"".args·1 ...interface {} "esc:0x0") + func @"".Warning (@"".args·1 ...interface {} "esc:0x0") + func @"".Error (@"".args·1 ...interface {} "esc:0x0") + func @"".Panic (@"".args·1 ...interface {} "esc:0x0") + func @"".Fatal (@"".args·1 ...interface {} "esc:0x0") + func @"".Debugf (@"".format·1 string "esc:0x0", @"".args·2 ...interface {} "esc:0x0") + func @"".Printf (@"".format·1 string "esc:0x0", @"".args·2 ...interface {} "esc:0x0") + func @"".Infof (@"".format·1 string "esc:0x0", @"".args·2 ...interface {} "esc:0x0") + func @"".Warnf (@"".format·1 string "esc:0x0", @"".args·2 ...interface {} "esc:0x0") + func @"".Warningf (@"".format·1 string "esc:0x0", @"".args·2 ...interface {} "esc:0x0") + func @"".Errorf (@"".format·1 string "esc:0x0", @"".args·2 ...interface {} "esc:0x0") + func @"".Panicf (@"".format·1 string "esc:0x0", @"".args·2 ...interface {} "esc:0x0") + func @"".Fatalf (@"".format·1 string "esc:0x0", @"".args·2 ...interface {} "esc:0x0") + func @"".Debugln (@"".args·1 ...interface {} "esc:0x0") + func @"".Println (@"".args·1 ...interface {} "esc:0x0") + func @"".Infoln (@"".args·1 ...interface {} "esc:0x0") + func @"".Warnln (@"".args·1 ...interface {} "esc:0x0") + func @"".Warningln (@"".args·1 ...interface {} "esc:0x0") + func @"".Errorln (@"".args·1 ...interface {} "esc:0x0") + func @"".Panicln (@"".args·1 ...interface {} "esc:0x0") + func @"".Fatalln (@"".args·1 ...interface {} "esc:0x0") + const @"".DefaultTimestampFormat = "2006-01-02T15:04:05Z07:00" + type @"".JSONFormatter struct { TimestampFormat string } + func (@"".f·3 *@"".JSONFormatter "esc:0x0") Format (@"".entry·4 *@"".Entry) (? []byte, ? error) + func @"".New () (? *@"".Logger) { return (&@"".Logger{ Out:@"os".Stderr, Formatter:new(@"".TextFormatter), Hooks:make(@"".LevelHooks, 0x0), Level:@"".Level(0x4) }) } + func @"".ParseLevel (@"".lvl·3 string) (? @"".Level, ? error) + const @"".PanicLevel @"".Level = 0x0 + const @"".FatalLevel @"".Level = 0x1 + const @"".ErrorLevel @"".Level = 0x2 + const @"".WarnLevel @"".Level = 0x3 + const @"".InfoLevel @"".Level = 0x4 + const @"".DebugLevel @"".Level = 0x5 + type @"".StdLogger interface { Fatal(? ...interface {}); Fatalf(? string, ? ...interface {}); Fatalln(? ...interface {}); Panic(? ...interface {}); Panicf(? string, ? ...interface {}); Panicln(? ...interface {}); Print(? ...interface {}); Printf(? string, ? ...interface {}); Println(? ...interface {}) } + type @"".Termios struct { Iflag uint64; Oflag uint64; Cflag uint64; Lflag uint64; Cc [20]uint8; Pad_cgo_0 [4]byte; Ispeed uint64; Ospeed uint64 } + func @"".IsTerminal () (? bool) + type @"".TextFormatter struct { ForceColors bool; DisableColors bool; DisableTimestamp bool; FullTimestamp bool; TimestampFormat string; DisableSorting bool } + func (@"".f·3 *@"".TextFormatter "esc:0x0") Format (@"".entry·4 *@"".Entry) (? []byte, ? error) + func (@"".f·1 *@"".TextFormatter "esc:0x0") @"".appendKeyValue (@"".b·2 *@"bytes".Buffer, @"".key·3 string "esc:0x0", @"".value·4 interface {}) + func (@"".f·1 *@"".TextFormatter "esc:0x0") @"".printColored (@"".b·2 *@"bytes".Buffer, @"".entry·3 *@"".Entry, @"".keys·4 []string "esc:0x0") + func @"".init () + var @"time".months [12]string + var @"time".days [7]string + var @"time".Local *@"time".Location + var @"time".UTC *@"time".Location + var @"".std *@"".Logger + type @"os".dirInfo struct { @"os".buf []byte; @"os".nbuf int; @"os".bufp int } + type @"os".file struct { @"os".fd int; @"os".name string; @"os".dirinfo *@"os".dirInfo; @"os".nepipe int32 } + func (@"os".file·2 *@"os".file) @"os".close () (? error) + type @"os".FileMode uint32 + func (@"os".m·2 @"os".FileMode) IsDir () (? bool) { return @"os".m·2 & @"os".FileMode(0x80000000) != @"os".FileMode(0x0) } + func (@"os".m·2 @"os".FileMode) IsRegular () (? bool) { return @"os".m·2 & @"os".FileMode(0x8F000000) == @"os".FileMode(0x0) } + func (@"os".m·2 @"os".FileMode) Perm () (? @"os".FileMode) { return @"os".m·2 & @"os".FileMode(0x1FF) } + func (@"os".m·2 @"os".FileMode) String () (? string) + type @"os".FileInfo interface { IsDir() (? bool); ModTime() (? @"time".Time); Mode() (? @"os".FileMode); Name() (? string); Size() (? int64); Sys() (? interface {}) } + type @"os".File struct { @"os".? *@"os".file } + func (@"os".f·2 *@"os".File) Chdir () (? error) + func (@"os".f·2 *@"os".File) Chmod (@"os".mode·3 @"os".FileMode) (? error) + func (@"os".f·2 *@"os".File) Chown (@"os".uid·3 int, @"os".gid·4 int) (? error) + func (@"os".f·2 *@"os".File) Close () (? error) + func (@"os".f·2 *@"os".File "esc:0x0") Fd () (? uintptr) { if @"os".f·2 == nil { return 0xFFFFFFFFFFFFFFFF }; return uintptr(@"os".f·2.@"os".file.@"os".fd) } + func (@"os".f·2 *@"os".File "esc:0x1") Name () (? string) { return @"os".f·2.@"os".file.@"os".name } + func (@"os".f·3 *@"os".File) Read (@"os".b·4 []byte "esc:0x0") (@"os".n·1 int, @"os".err·2 error) + func (@"os".f·3 *@"os".File) ReadAt (@"os".b·4 []byte "esc:0x0", @"os".off·5 int64) (@"os".n·1 int, @"os".err·2 error) + func (@"os".f·3 *@"os".File "esc:0x0") Readdir (@"os".n·4 int) (@"os".fi·1 []@"os".FileInfo, @"os".err·2 error) + func (@"os".f·3 *@"os".File "esc:0x0") Readdirnames (@"os".n·4 int) (@"os".names·1 []string, @"os".err·2 error) + func (@"os".f·3 *@"os".File) Seek (@"os".offset·4 int64, @"os".whence·5 int) (@"os".ret·1 int64, @"os".err·2 error) + func (@"os".f·3 *@"os".File) Stat () (@"os".fi·1 @"os".FileInfo, @"os".err·2 error) + func (@"os".f·2 *@"os".File "esc:0x0") Sync () (@"os".err·1 error) + func (@"os".f·2 *@"os".File) Truncate (@"os".size·3 int64) (? error) + func (@"os".f·3 *@"os".File) Write (@"os".b·4 []byte "esc:0x0") (@"os".n·1 int, @"os".err·2 error) + func (@"os".f·3 *@"os".File) WriteAt (@"os".b·4 []byte "esc:0x0", @"os".off·5 int64) (@"os".n·1 int, @"os".err·2 error) + func (@"os".f·3 *@"os".File) WriteString (@"os".s·4 string "esc:0x0") (@"os".ret·1 int, @"os".err·2 error) + func (@"os".f·3 *@"os".File "esc:0x0") @"os".pread (@"os".b·4 []byte "esc:0x0", @"os".off·5 int64) (@"os".n·1 int, @"os".err·2 error) + func (@"os".f·3 *@"os".File "esc:0x0") @"os".pwrite (@"os".b·4 []byte "esc:0x0", @"os".off·5 int64) (@"os".n·1 int, @"os".err·2 error) + func (@"os".f·3 *@"os".File "esc:0x0") @"os".read (@"os".b·4 []byte "esc:0x0") (@"os".n·1 int, @"os".err·2 error) + func (@"os".f·3 *@"os".File "esc:0x0") @"os".readdir (@"os".n·4 int) (@"os".fi·1 []@"os".FileInfo, @"os".err·2 error) + func (@"os".f·3 *@"os".File "esc:0x0") @"os".readdirnames (@"os".n·4 int) (@"os".names·1 []string, @"os".err·2 error) + func (@"os".f·3 *@"os".File "esc:0x0") @"os".seek (@"os".offset·4 int64, @"os".whence·5 int) (@"os".ret·1 int64, @"os".err·2 error) + func (@"os".f·3 *@"os".File "esc:0x0") @"os".write (@"os".b·4 []byte "esc:0x0") (@"os".n·1 int, @"os".err·2 error) + var @"os".Stderr *@"os".File + +$$ +_go_.6 0 0 0 644 260091 ` +go object darwin amd64 go1.4.2 X:precisestack + +! +go13ldbytes.a +fmt.aio.aos.a time.aencoding/json.a sync.a +log.asyscall.aruntime.a sort.astrings.abufio.a�"".NewEntry��eH� %H;aw���H��(H�H�$H�D$�H�\$H�\$ H�H�$�H�L$H��H��tk1��H�L$H� $H�<$tKH�\$0H�\$�H�\$H�$H�<$t#H�$H�\$ H�\$�H�\$H�\$8H��(É%�ԉ%묉� + 0runtime.morestack_noctxt:type."".Fields^runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr P"".autotmp_0001type.*"".Entry"".autotmp_0000type."".Fields "".~r1type.*"".Entry"".loggertype.*"".LoggerP�OP�@�.4$0Tgclocals·a7c27d2bfcc924fa8a92b6b29b7218b1Tgclocals·e475e3c2360b557d64285d9b9a4e5064�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�$"".(*Entry).Reader��eH� %H�D$�H;Aw���H��H��$�HDŽ$�HDŽ$�H�*H���H�MH�E H�T$H�D$XH�$H�L$PH�Y ��H�T$H�L$H�D$ H�\$(H�\$@H�\$0H�\$HH�T$`H�T$xH�L$hH��$�H�D$pH��$�H�H�$�H�L$H��H��tx1��H�L$8H� $H�<$tXH�\$xH�\$H��$�H�\$H��$�H�\$�H�\$8H��$�H�\$@H��$�H�\$HH��$�H�ĐÉ%량넉E����� +*0runtime.morestack_noctxt� +�"type.bytes.Buffer�"runtime.newobject�� runtime.duffzero�2runtime.writebarrierslice@�"".autotmp_0003�$type.*bytes.Buffer"".autotmp_0002$type.*bytes.Bufferbytes.buf·2/type.[]uint8 "".err�type.error"".serialized_type.[]uint8 "".~r1 type.error "".~r0$type.*bytes.Buffer"".entrytype.*"".Entry"�����RBQ�nTNPTgclocals·363b18caf0020ca418fd378dbb75c855Tgclocals·0719ac7e4405ec7094b2d696ead0af25�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�$"".(*Entry).String��eH� %H;aw���H��`H�D$pH�D$xHDŽ$�HDŽ$�H�\$hH�$�H�D$H�T$H�t$H�t$@H��H�T$8t'H�D$pH�D$xH��$�H��$�H��`�H�D$(H�D$01�H9�u-H�H� H�CH�L$pH�D$xH��$�H��$�H��`�H�xH�PH�HH9�rYH�H��H)�H��H)�H��t H��H�H��H�D$HH�$H�t$PH�t$H�T$XH�T$�H�t$@H�T$8H�L$H�D$ �w���� + 0runtime.morestack_noctxt�$"".(*Entry).Reader�"go.string.""�2runtime.slicebytetostring�$runtime.panicsliceP� + "".~r0otype.string "".errOtype.error "".~r10type.error "".~r0type.string"".entrytype.*"".Entry$�}��E��r�`D" '�M�Tgclocals·896a3e2c9de7030cc72aa334f690557dTgclocals·44e348188e22fef6300f71ab26e45197�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�*"".(*Entry).WithField��eH� %H;aw���H��HH�H�$H�D$�H�D$H�\$XH�\$8H�\$`H�\$@H�\$hH�\$(H�\$pH�\$0H�H�$H�D$ H�D$H�\$8H�\$H�\$(H�\$�H�\$PH�$H�\$ H�\$�H�\$H�\$xH��H� + 0runtime.morestack_noctxt:type."".Fields^runtime.makemap�type."".Fields�$runtime.mapassign1�,"".(*Entry).WithFields`�"".autotmp_0016?"type.interface {}"".autotmp_0015type.string"".autotmp_0014Otype."".Fields "".~r2Ptype.*"".Entry"".value0"type.interface {} "".keytype.string"".entrytype.*"".Entry��� � t�.[Tgclocals·0723c8881b4d19cb48cb8887cfa073beTgclocals·396579fca70851935df9d21183ca82fd�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�,"".(*Entry).WithFields� � eH� %H�D$�H;Aw���H���H�H�$H�D$�H�\$H�\$ H��$�H�kH��$�1��H�H�$H�l$H��$�H�\$�H��$�1�H9���H��$�H���xH� H�CH��$�H���XH�3H�kH�L$pH�D$xH�t$PH��$�H�l$XH��$�H�L$0H��$�H�D$8H��$�H�H�$H�\$ H�\$H��$�H�\$H��$�H�\$�H��$�H�$�H��$�1�H9��8���H��$H��$�1��H�H�$H�L$H��$�H�\$�H��$�1�H9���H��$�H���VH� H�CH��$�H���6H�3H�kH�L$pH�D$xH�t$`H��$�H�l$hH��$�H�L$@H��$�H�D$HH��$�H�H�$H�\$ H�\$H��$�H�\$H��$�H�\$�H��$�H�$�H��$�1�H9��8���H�H�$�H�L$H��H��tw1��H�L$(H� $H�<$tWH��$�H�+H�l$�H�\$(H�$H�<$t)H�$H�\$ H�\$�H�\$(H��$H���É%�Ή%렉녉��������������������* +*0runtime.morestack_noctxtJtype."".Fieldsnruntime.makemap�� runtime.duffzero�type."".Fields�&runtime.mapiterinit�type."".Fields�$runtime.mapassign1�&runtime.mapiternext�� runtime.duffzero�type."".Fields�&runtime.mapiterinit�type."".Fields� $runtime.mapassign1� &runtime.mapiternext� type."".Entry� "runtime.newobject� +� runtime.duffzero� +.runtime.writebarrierptr� .runtime.writebarrierptr0�$"".autotmp_0027�type.*"".Entry"".autotmp_0026"type.interface {}"".autotmp_0025�"type.interface {}"".autotmp_0024"type.interface {}"".autotmp_0023type.string"".autotmp_0022Btype.map.iter[string]interface {}"".autotmp_0021type."".Fields"".autotmp_0020�"type.interface {}"".autotmp_0019�type.string"".autotmp_0018�Btype.map.iter[string]interface {}"".v�"type.interface {}"".k�type.string"".v�"type.interface {}"".k�type.string"".data�type."".Fields "".~r1 type.*"".Entry"".fieldstype."".Fields"".entrytype.*"".Entry"����<�4~"#�h$�h$�.6G��#:$VTgclocals·fdf817463ca91d173b8e929c420286bdTgclocals·cbbe1bd73f3c341fc477038dafd9ade4�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�"".(*Entry).log��eH� %H�D$�H;Aw���H����H�,$�T$H�L$H��$H�$H�<$�SH�$H�D$H��$�H�l$��$��T$H��$�H�L$ �H��$��$@�i(H� $H�<$��H�$0H��$H�\$H��$H�\$�H��$H�H�kH�,$��$�\$H�L$�H�T$H�L$ H�L$pH��H�T$h�bH��$H�+H�,$H�<$�[H�$,�H� H�L$X1�H9��H��$�H�H�CH��$�H����H��H��H��$�H��$�H��$�H�\$hH�$H�\$pH�\$�H�T$H�L$H��$�H�$H��$�H�T$H��$�H�L$�H�H�L$XH��$�H� $H��$�H�T$H�H�l$H��H��H�H�H��$�H�\$ H��$�H�\$(H��$�H�\$0�H��$H�+H�,$H�<$��H�$,�H��$H�$�H�\$H�\$PH�L$H�\$H��$�H��H�L$x�eH��$H�+H�,$H�<$�gH�$,�H� H�L$X1�H9��H��$�H�H�CH��$�H����H��H��H��$�H��$�H��$�H�\$xH�$H��$�H�\$�H�T$H�L$H��$�H�$H��$�H�T$H��$�H�L$�H�H�L$XH��$�H� $H��$�H�T$H�H�l$H��H��H�H�H��$�H�\$ H��$�H�\$(H��$�H�\$0�H��$H�+H�,$H�<$��H�$,�H��$H�+H�,$H�<$��H�$,�H��$H�+H�,$H�<$�|H�$,H� Qj�YYH���OH�\$PH�\$`H�1�H9��H��$H�>H����H�7H�<$H�H�H�L$`H��$�H�D$H��$�H�L$�H�t$(H�\$0H��$�H��H�t$x�H�H�D$X1�H9��BH��$�H�H�CH��$�H���H��H��H��$�H��$�H��$�H�4$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H� H�D$XH��$�H�$H��$�H�L$H�H�l$H��H��H�H�H��$�H�\$ H��$�H�\$(H��$�H�\$0���$��wH��$H�H�$H�D$� ��H���É�����H�H�$H�H�\$H�H�\$�H�t$xH�\$H�\$X���������H�H�$H�H�\$H�H�\$�H�D$�������H���É%�x����%�H����%���������H�H�$H�H�\$H�H�\$�H�\$H�\$X�����%�����%�!�����&���H�H�$H�H�\$H�H�\$�H�\$H�\$X������%�����%�����%����l +*0runtime.morestack_noctxtFtime.Now�0runtime.writebarrierfat3�4runtime.writebarrierstring�$"".LevelHooks.Fire�$sync.(*Mutex).Lock�4go.itab.*os.File.io.Writer�runtime.convI2E�2runtime.writebarrieriface�os.Stderr�Jgo.string."Failed to fire hook: %v\n"� fmt.Fprintf� (sync.(*Mutex).Unlock� +$"".(*Entry).Reader� $sync.(*Mutex).Lock� 4go.itab.*os.File.io.Writer� runtime.convI2E�2runtime.writebarrieriface�os.Stderr�Rgo.string."Failed to obtain reader, %v\n"�fmt.Fprintf�(sync.(*Mutex).Unlock�$sync.(*Mutex).Lock�.sync.(*Mutex).Unlock·f�"runtime.deferproc�>go.itab.*bytes.Buffer.io.Reader�io.Copy�4go.itab.*os.File.io.Writer�runtime.convI2E�2runtime.writebarrieriface�os.Stderr�Pgo.string."Failed to write to log, %v\n"�fmt.Fprintf�type.*"".Entry�runtime.gopanic�&runtime.deferreturn�type.*os.File�type.io.Writer�4go.itab.*os.File.io.Writer� runtime.typ2Itab�$type.*bytes.Buffer�type.io.Reader�>go.itab.*bytes.Buffer.io.Reader� runtime.typ2Itab�&runtime.deferreturn�type.*os.File�type.io.Writer�4go.itab.*os.File.io.Writer� runtime.typ2Itab�type.*os.File�type.io.Writer�4go.itab.*os.File.io.Writer� runtime.typ2Itab@�."".autotmp_0047"type.interface {}"".autotmp_0046*type.*[1]interface {}"".autotmp_0045&type.[]interface {}"".autotmp_0044type.*uint8"".autotmp_0043type.*uint8"".autotmp_0042"type.interface {}"".autotmp_0041*type.*[1]interface {}"".autotmp_0040&type.[]interface {}"".autotmp_0039type.*uint8"".autotmp_0038�"type.interface {}"".autotmp_0036_&type.[]interface {}"".autotmp_0035�type.*uint8"".autotmp_0033(type.[1]interface {}"".autotmp_0032�$type.*bytes.Buffer"".autotmp_0031(type.[1]interface {}"".autotmp_0030(type.[1]interface {}"".autotmp_0028/type.time.Time "".err�type.error"".reader�$type.*bytes.Buffer "".err�type.error "".msg type.string"".leveltype."".Level"".entrytype.*"".Entry<"������������"i ;?$�$-$�$$9r� C9  > >  \"�~�$`��$� ��*?C\V>Tgclocals·65a30d49934626502b3d799f3cf8d99eTgclocals·22d60cc41efa02a0ac67663f051098e8�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�""".(*Entry).Debug��eH� %H;aw���H��8H�l$@H�m�](��rSH�\$HH�$H�\$PH�\$H�\$XH�\$�H�L$H�D$ H�\$@H�$�D$H�L$(H�L$H�D$0H�D$�H��8� + 0runtime.morestack_noctxt�fmt.Sprint�"".(*Entry).log@p"".autotmp_0067type.string"".args&type.[]interface {}"".entrytype.*"".Entrypio ��S +IGTgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�""".(*Entry).Print��eH� %H;aw���H�� H�\$(H�$H�\$0H�\$H�\$8H�\$H�\$@H�\$�H�� � + 0runtime.morestack_noctxt� "".(*Entry).Info@@"".args&type.[]interface {}"".entrytype.*"".Entry@0?P�, + +ATgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go� "".(*Entry).Info��eH� %H;aw���H��8H�l$@H�m�](��rSH�\$HH�$H�\$PH�\$H�\$XH�\$�H�L$H�D$ H�\$@H�$�D$H�L$(H�L$H�D$0H�D$�H��8� + 0runtime.morestack_noctxt�fmt.Sprint�"".(*Entry).log@p"".autotmp_0068type.string"".args&type.[]interface {}"".entrytype.*"".Entrypio ��S +IGTgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go� "".(*Entry).Warn��eH� %H;aw���H��8H�l$@H�m�](��rSH�\$HH�$H�\$PH�\$H�\$XH�\$�H�L$H�D$ H�\$@H�$�D$H�L$(H�L$H�D$0H�D$�H��8� + 0runtime.morestack_noctxt�fmt.Sprint�"".(*Entry).log@p"".autotmp_0069type.string"".args&type.[]interface {}"".entrytype.*"".Entrypio ��S +IGTgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�&"".(*Entry).Warning��eH� %H;aw���H�� H�\$(H�$H�\$0H�\$H�\$8H�\$H�\$@H�\$�H�� � + 0runtime.morestack_noctxt� "".(*Entry).Warn@@"".args&type.[]interface {}"".entrytype.*"".Entry@0?P�, + +ATgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�""".(*Entry).Error��eH� %H;aw���H��8H�l$@H�m�](��rSH�\$HH�$H�\$PH�\$H�\$XH�\$�H�L$H�D$ H�\$@H�$�D$H�L$(H�L$H�D$0H�D$�H��8� + 0runtime.morestack_noctxt�fmt.Sprint�"".(*Entry).log@p"".autotmp_0070type.string"".args&type.[]interface {}"".entrytype.*"".Entrypio ��S +IGTgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�""".(*Entry).Fatal��eH� %H;aw���H��8H�l$@H�m�](��rSH�\$HH�$H�\$PH�\$H�\$XH�\$�H�L$H�D$ H�\$@H�$�D$H�L$(H�L$H�D$0H�D$�H�$�H��8� + + 0runtime.morestack_noctxt�fmt.Sprint�"".(*Entry).log�os.Exit@p"".autotmp_0071type.string"".args&type.[]interface {}"".entrytype.*"".Entrypvo��S  +IWTgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�""".(*Entry).Panic��eH� %H;aw���H��HH�l$PH�m�](��rSH�\$XH�$H�\$`H�\$H�\$hH�\$�H�L$H�D$ H�\$PH�$�D$H�L$8H�L$H�D$@H�D$�H�\$XH�$H�\$`H�\$H�\$hH�\$�H�\$H�\$(H�\$ H�\$0H�H�$H�\$(H�\$�H�\$H�,$H��H��H�H��  + 0runtime.morestack_noctxt�fmt.Sprint�"".(*Entry).log�fmt.Sprint�type.string�runtime.convT2E�runtime.gopanic@�"".autotmp_0073?type.string"".autotmp_0072type.string"".args&type.[]interface {}"".entrytype.*"".Entry����SqI�&Tgclocals·9ff42bf311af152488d11f0f78c8d5ceTgclocals·29f0050a5ee7c2b9348a75428171d7de�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�$"".(*Entry).Debugf��eH� %H;aw���H��H��$�H�m�](���H��$�H�$H��$�H�\$H��$�H�\$H��$�H�\$H��$�H�\$ �H�\$(H�\$HH�\$0H�\$PH�\$XH�H�CH�\$XH����H��H��H�\$hH�T$pH�L$xH�H�$H�\$HH�\$�H�L$H�D$H�\$hH�$H�L$8H�L$H�D$@H�D$�H��$�H�$H�\$hH�\$H�\$pH�\$H�\$xH�\$�H�ĀÉ�_��� + 0runtime.morestack_noctxt�fmt.Sprintf�type.string�runtime.convT2E�2runtime.writebarrieriface�""".(*Entry).Debug`�"".autotmp_0078�"type.interface {}"".autotmp_0076/&type.[]interface {}"".autotmp_0075otype.string"".autotmp_0074O(type.[1]interface {}"".args0&type.[]interface {}"".formattype.string"".entrytype.*"".Entry�������vn[!Tgclocals·e8d3240594e259421cd655d317fed5feTgclocals·1eb9d8ec9969f1d922533aa863dff6f6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�""".(*Entry).Infof��eH� %H;aw���H��H��$�H�m�](���H��$�H�$H��$�H�\$H��$�H�\$H��$�H�\$H��$�H�\$ �H�\$(H�\$HH�\$0H�\$PH�\$XH�H�CH�\$XH����H��H��H�\$hH�T$pH�L$xH�H�$H�\$HH�\$�H�L$H�D$H�\$hH�$H�L$8H�L$H�D$@H�D$�H��$�H�$H�\$hH�\$H�\$pH�\$H�\$xH�\$�H�ĀÉ�_��� + 0runtime.morestack_noctxt�fmt.Sprintf�type.string�runtime.convT2E�2runtime.writebarrieriface� "".(*Entry).Info`�"".autotmp_0086�"type.interface {}"".autotmp_0084/&type.[]interface {}"".autotmp_0083otype.string"".autotmp_0082O(type.[1]interface {}"".args0&type.[]interface {}"".formattype.string"".entrytype.*"".Entry�������vn[!Tgclocals·e8d3240594e259421cd655d317fed5feTgclocals·1eb9d8ec9969f1d922533aa863dff6f6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�$"".(*Entry).Printf��eH� %H;aw���H��0H�\$8H�$H�\$@H�\$H�\$HH�\$H�\$PH�\$H�\$XH�\$ H�\$`H�\$(�H��0� + 0runtime.morestack_noctxt�""".(*Entry).Infof``"".args0&type.[]interface {}"".formattype.string"".entrytype.*"".Entry`D_`�@ +U Tgclocals·0a3395567ab7eee3bb936aced49af517Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�""".(*Entry).Warnf��eH� %H;aw���H��H��$�H�m�](���H��$�H�$H��$�H�\$H��$�H�\$H��$�H�\$H��$�H�\$ �H�\$(H�\$HH�\$0H�\$PH�\$XH�H�CH�\$XH����H��H��H�\$hH�T$pH�L$xH�H�$H�\$HH�\$�H�L$H�D$H�\$hH�$H�L$8H�L$H�D$@H�D$�H��$�H�$H�\$hH�\$H�\$pH�\$H�\$xH�\$�H�ĀÉ�_��� + 0runtime.morestack_noctxt�fmt.Sprintf�type.string�runtime.convT2E�2runtime.writebarrieriface� "".(*Entry).Warn`�"".autotmp_0094�"type.interface {}"".autotmp_0092/&type.[]interface {}"".autotmp_0091otype.string"".autotmp_0090O(type.[1]interface {}"".args0&type.[]interface {}"".formattype.string"".entrytype.*"".Entry�������vn[!Tgclocals·e8d3240594e259421cd655d317fed5feTgclocals·1eb9d8ec9969f1d922533aa863dff6f6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�("".(*Entry).Warningf��eH� %H;aw���H��0H�\$8H�$H�\$@H�\$H�\$HH�\$H�\$PH�\$H�\$XH�\$ H�\$`H�\$(�H��0� + 0runtime.morestack_noctxt�""".(*Entry).Warnf``"".args0&type.[]interface {}"".formattype.string"".entrytype.*"".Entry`D_`�@ +U Tgclocals·0a3395567ab7eee3bb936aced49af517Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�$"".(*Entry).Errorf��eH� %H;aw���H��H��$�H�m�](���H��$�H�$H��$�H�\$H��$�H�\$H��$�H�\$H��$�H�\$ �H�\$(H�\$HH�\$0H�\$PH�\$XH�H�CH�\$XH����H��H��H�\$hH�T$pH�L$xH�H�$H�\$HH�\$�H�L$H�D$H�\$hH�$H�L$8H�L$H�D$@H�D$�H��$�H�$H�\$hH�\$H�\$pH�\$H�\$xH�\$�H�ĀÉ�_��� + 0runtime.morestack_noctxt�fmt.Sprintf�type.string�runtime.convT2E�2runtime.writebarrieriface�""".(*Entry).Error`�"".autotmp_0102�"type.interface {}"".autotmp_0100/&type.[]interface {}"".autotmp_0099otype.string"".autotmp_0098O(type.[1]interface {}"".args0&type.[]interface {}"".formattype.string"".entrytype.*"".Entry�������vn[!Tgclocals·e8d3240594e259421cd655d317fed5feTgclocals·1eb9d8ec9969f1d922533aa863dff6f6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�$"".(*Entry).Fatalf��eH� %H;aw���H��H��$�H�m�](���H��$�H�$H��$�H�\$H��$�H�\$H��$�H�\$H��$�H�\$ �H�\$(H�\$HH�\$0H�\$PH�\$XH�H�CH�\$XH����H��H��H�\$hH�T$pH�L$xH�H�$H�\$HH�\$�H�L$H�D$H�\$hH�$H�L$8H�L$H�D$@H�D$�H��$�H�$H�\$hH�\$H�\$pH�\$H�\$xH�\$�H�$�H�ĀÉ�R��� + 0runtime.morestack_noctxt�fmt.Sprintf�type.string�runtime.convT2E�2runtime.writebarrieriface�""".(*Entry).Fatal�os.Exit`�"".autotmp_0110�"type.interface {}"".autotmp_0108/&type.[]interface {}"".autotmp_0107otype.string"".autotmp_0106O(type.[1]interface {}"".args0&type.[]interface {}"".formattype.string"".entrytype.*"".Entry������� vn[ Tgclocals·e8d3240594e259421cd655d317fed5feTgclocals·1eb9d8ec9969f1d922533aa863dff6f6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�$"".(*Entry).Panicf��eH� %H;aw���H��H��$�H�m�](���H��$�H�$H��$�H�\$H��$�H�\$H��$�H�\$H��$�H�\$ �H�\$(H�\$HH�\$0H�\$PH�\$XH�H�CH�\$XH����H��H��H�\$hH�T$pH�L$xH�H�$H�\$HH�\$�H�L$H�D$H�\$hH�$H�L$8H�L$H�D$@H�D$�H��$�H�$H�\$hH�\$H�\$pH�\$H�\$xH�\$�H�ĀÉ�_��� + 0runtime.morestack_noctxt�fmt.Sprintf�type.string�runtime.convT2E�2runtime.writebarrieriface�""".(*Entry).Panic`�"".autotmp_0118�"type.interface {}"".autotmp_0116/&type.[]interface {}"".autotmp_0115otype.string"".autotmp_0114O(type.[1]interface {}"".args0&type.[]interface {}"".formattype.string"".entrytype.*"".Entry�������vn[!Tgclocals·e8d3240594e259421cd655d317fed5feTgclocals·1eb9d8ec9969f1d922533aa863dff6f6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�&"".(*Entry).Debugln��eH� %H;aw���H��xH��$�H�(�](����H�$H��$�H�\$H��$�H�\$H��$�H�\$�H�\$ H�\$@H�\$(H�\$HH�\$PH�H�CH�\$PH����H��H��H�\$`H�T$hH�L$pH�H�$H�\$@H�\$�H�L$H�D$H�\$`H�$H�L$0H�L$H�D$8H�D$�H��$�H�$H�\$`H�\$H�\$hH�\$H�\$pH�\$�H��xÉ�b��� + 0runtime.morestack_noctxt�*"".(*Entry).sprintlnn�type.string�runtime.convT2E�2runtime.writebarrieriface�""".(*Entry).Debug@� "".autotmp_0126�"type.interface {}"".autotmp_0124/&type.[]interface {}"".autotmp_0123otype.string"".autotmp_0122O(type.[1]interface {}"".args&type.[]interface {}"".entrytype.*"".Entry������"�]n[Tgclocals·b29a376724b9675f7c9e576a6dabc1e0Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�$"".(*Entry).Infoln��eH� %H;aw���H��xH��$�H�(�](����H�$H��$�H�\$H��$�H�\$H��$�H�\$�H�\$ H�\$@H�\$(H�\$HH�\$PH�H�CH�\$PH����H��H��H�\$`H�T$hH�L$pH�H�$H�\$@H�\$�H�L$H�D$H�\$`H�$H�L$0H�L$H�D$8H�D$�H��$�H�$H�\$`H�\$H�\$hH�\$H�\$pH�\$�H��xÉ�b��� + 0runtime.morestack_noctxt�*"".(*Entry).sprintlnn�type.string�runtime.convT2E�2runtime.writebarrieriface� "".(*Entry).Info@� "".autotmp_0134�"type.interface {}"".autotmp_0132/&type.[]interface {}"".autotmp_0131otype.string"".autotmp_0130O(type.[1]interface {}"".args&type.[]interface {}"".entrytype.*"".Entry������"�]n[Tgclocals·b29a376724b9675f7c9e576a6dabc1e0Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�&"".(*Entry).Println��eH� %H;aw���H�� H�\$(H�$H�\$0H�\$H�\$8H�\$H�\$@H�\$�H�� � + 0runtime.morestack_noctxt�$"".(*Entry).Infoln@@"".args&type.[]interface {}"".entrytype.*"".Entry@0?P�, + +ATgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�$"".(*Entry).Warnln��eH� %H;aw���H��xH��$�H�(�](����H�$H��$�H�\$H��$�H�\$H��$�H�\$�H�\$ H�\$@H�\$(H�\$HH�\$PH�H�CH�\$PH����H��H��H�\$`H�T$hH�L$pH�H�$H�\$@H�\$�H�L$H�D$H�\$`H�$H�L$0H�L$H�D$8H�D$�H��$�H�$H�\$`H�\$H�\$hH�\$H�\$pH�\$�H��xÉ�b��� + 0runtime.morestack_noctxt�*"".(*Entry).sprintlnn�type.string�runtime.convT2E�2runtime.writebarrieriface� "".(*Entry).Warn@� "".autotmp_0142�"type.interface {}"".autotmp_0140/&type.[]interface {}"".autotmp_0139otype.string"".autotmp_0138O(type.[1]interface {}"".args&type.[]interface {}"".entrytype.*"".Entry������"�]n[Tgclocals·b29a376724b9675f7c9e576a6dabc1e0Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�*"".(*Entry).Warningln��eH� %H;aw���H�� H�\$(H�$H�\$0H�\$H�\$8H�\$H�\$@H�\$�H�� � + 0runtime.morestack_noctxt�$"".(*Entry).Warnln@@"".args&type.[]interface {}"".entrytype.*"".Entry@0?P�, + +ATgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�&"".(*Entry).Errorln��eH� %H;aw���H��xH��$�H�(�](����H�$H��$�H�\$H��$�H�\$H��$�H�\$�H�\$ H�\$@H�\$(H�\$HH�\$PH�H�CH�\$PH����H��H��H�\$`H�T$hH�L$pH�H�$H�\$@H�\$�H�L$H�D$H�\$`H�$H�L$0H�L$H�D$8H�D$�H��$�H�$H�\$`H�\$H�\$hH�\$H�\$pH�\$�H��xÉ�b��� + 0runtime.morestack_noctxt�*"".(*Entry).sprintlnn�type.string�runtime.convT2E�2runtime.writebarrieriface�""".(*Entry).Error@� "".autotmp_0150�"type.interface {}"".autotmp_0148/&type.[]interface {}"".autotmp_0147otype.string"".autotmp_0146O(type.[1]interface {}"".args&type.[]interface {}"".entrytype.*"".Entry������"�]n[Tgclocals·b29a376724b9675f7c9e576a6dabc1e0Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�&"".(*Entry).Fatalln��eH� %H;aw���H��xH��$�H�(�](����H�$H��$�H�\$H��$�H�\$H��$�H�\$�H�\$ H�\$@H�\$(H�\$HH�\$PH�H�CH�\$PH����H��H��H�\$`H�T$hH�L$pH�H�$H�\$@H�\$�H�L$H�D$H�\$`H�$H�L$0H�L$H�D$8H�D$�H��$�H�$H�\$`H�\$H�\$hH�\$H�\$pH�\$�H�$�H��xÉ�U��� + 0runtime.morestack_noctxt�*"".(*Entry).sprintlnn�type.string�runtime.convT2E�2runtime.writebarrieriface�""".(*Entry).Fatal�os.Exit@� "".autotmp_0158�"type.interface {}"".autotmp_0156/&type.[]interface {}"".autotmp_0155otype.string"".autotmp_0154O(type.[1]interface {}"".args&type.[]interface {}"".entrytype.*"".Entry������"� ]n[ Tgclocals·b29a376724b9675f7c9e576a6dabc1e0Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�&"".(*Entry).Panicln��eH� %H;aw���H��xH��$�H�(�](����H�$H��$�H�\$H��$�H�\$H��$�H�\$�H�\$ H�\$@H�\$(H�\$HH�\$PH�H�CH�\$PH����H��H��H�\$`H�T$hH�L$pH�H�$H�\$@H�\$�H�L$H�D$H�\$`H�$H�L$0H�L$H�D$8H�D$�H��$�H�$H�\$`H�\$H�\$hH�\$H�\$pH�\$�H��xÉ�b��� + 0runtime.morestack_noctxt�*"".(*Entry).sprintlnn�type.string�runtime.convT2E�2runtime.writebarrieriface�""".(*Entry).Panic@� "".autotmp_0166�"type.interface {}"".autotmp_0164/&type.[]interface {}"".autotmp_0163otype.string"".autotmp_0162O(type.[1]interface {}"".args&type.[]interface {}"".entrytype.*"".Entry������"�]n[Tgclocals·b29a376724b9675f7c9e576a6dabc1e0Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�*"".(*Entry).sprintlnn��eH� %H;aw���H��(H�D$PH�D$XH�\$8H�$H�\$@H�\$H�\$HH�\$�H�t$H�L$ H��H��H9�rH�t$PH�T$XH��(��  + 0runtime.morestack_noctxt�fmt.Sprintln�$runtime.panicslice`P"".autotmp_0172type.int "".~r1@type.string"".args&type.[]interface {}"".entrytype.*"".EntryPWOP��,,( +I7Tgclocals·9f0d5ba6770c4a1ed4fa771547e96df1Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�""".StandardLogger H�H�\$� "".std "".~r0type.*"".LoggerTgclocals·a7a3692b8e27e823add69ec4239ba55fTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".SetOutput��eH� %H;aw���H��H�H�$H�<$��H�$,�H�H�$H�<$teH�$,H� Qj�YYH��u?H�H�$H�<$t$H�\$ H�\$H�\$(H�\$���H��É%�Ӑ�H��É%뒉%�j��� + 0runtime.morestack_noctxt: "".stdl$sync.(*Mutex).Lockz "".std�.sync.(*Mutex).Unlock·f�"runtime.deferproc� "".std�2runtime.writebarrieriface�&runtime.deferreturn�&runtime.deferreturn 0 "".outtype.io.Writer*0?:/0/0�"$ -+   5�Tgclocals·20671cc48303dfd2b9d73bba3d1850b7Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".SetFormatter��eH� %H;aw���H��H�H�$H�<$��H�$,�H�H�$H�<$tjH�$,H� Qj�YYH��uDH�H�$H�<$t)H�$H�\$ H�\$H�\$(H�\$���H��É%�ΐ�H��É%덉%�e��� + 0runtime.morestack_noctxt: "".stdl$sync.(*Mutex).Lockz "".std�.sync.(*Mutex).Unlock·f�"runtime.deferproc� "".std�2runtime.writebarrieriface�&runtime.deferreturn�&runtime.deferreturn 0"".formatter"type."".Formatter*0??/0/0�"2 -0   5�Tgclocals·20671cc48303dfd2b9d73bba3d1850b7Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".SetLevel��eH� %H;aw���H��H�H�$H�<$tfH�$,�H�H�$H�<$tAH�$,H� Qj�YYH��uH��l$@�k(��H��Ð�H��É%붉%� + 0runtime.morestack_noctxt: "".stdd$sync.(*Mutex).Lockr "".std�.sync.(*Mutex).Unlock·f�"runtime.deferproc� "".std�&runtime.deferreturn�&runtime.deferreturn"".leveltype."".Level*; +�@-  +1oTgclocals·5d05a78f811f5c3f62710534cdce0004Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".GetLevel��eH� %H;aw���H���D$H�H�$H�<$tfH�$,�H�H�$H�<$tAH�$,H� Qj�YYH��uH��k(@�l$��H��Ð�H��É%붉%� + 0runtime.morestack_noctxtD "".stdn$sync.(*Mutex).Lock| "".std�.sync.(*Mutex).Unlock·f�"runtime.deferproc� "".std�&runtime.deferreturn�&runtime.deferreturn "".~r0type."".Level*@ +�N-  +6jTgclocals·7c868751a5d2fdd881613692c78d6476Tgclocals·0115f8d53b75c1696444f08ad03251d9�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".AddHook��eH� %H;aw���H��H�H�$H�<$t~H�$,�H�H�$H�<$tYH�$,H� Qj�YYH��u3H�H�kH�,$H�\$ H�\$H�\$(H�\$���H��Ð�H��É%랉%�v��� + 0runtime.morestack_noctxt: "".stdd$sync.(*Mutex).Lockr "".std�.sync.(*Mutex).Unlock·f�"runtime.deferproc� "".std�""".LevelHooks.Add�&runtime.deferreturn�&runtime.deferreturn 0"".hooktype."".Hook*0;7/0 +/0�\-(  1�Tgclocals·20671cc48303dfd2b9d73bba3d1850b7Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".WithField��eH� %H;aw���H��0H�H�$H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$PH�\$ �H�\$(H�\$XH��0� + 0runtime.morestack_noctxt: "".std�,"".(*Logger).WithFieldP` "".~r2@type.*"".Entry"".value "type.interface {} "".keytype.string`F_p +rV +M#Tgclocals·66ae2244d17a3b89653cba445a520071Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".WithFields��eH� %H;aw���H��H�H�$H�\$ H�\$�H�\$H�\$(H��� + 0runtime.morestack_noctxt: "".std`."".(*Logger).WithFields 0 "".~r1type.*"".Entry"".fieldstype."".Fields0(/P �6 +/!Tgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Debug��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�$"".(*Logger).Debug0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Print��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�$"".(*Logger).Print0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Info��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�""".(*Logger).Info0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Warn��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�""".(*Logger).Warn0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Warning��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�("".(*Logger).Warning0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Error��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�$"".(*Logger).Error0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Panic��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�$"".(*Logger).Panic0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Fatal��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�$"".(*Logger).Fatal0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Debugf��eH� %H;aw���H��0H�H�$H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$PH�\$ H�\$XH�\$(�H��0� + 0runtime.morestack_noctxt: "".std�&"".(*Logger).DebugfP`"".args &type.[]interface {}"".formattype.string`F_p�B +WTgclocals·f271231f400e778e0f59be25f7a26a56Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Printf��eH� %H;aw���H��0H�H�$H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$PH�\$ H�\$XH�\$(�H��0� + 0runtime.morestack_noctxt: "".std�&"".(*Logger).PrintfP`"".args &type.[]interface {}"".formattype.string`F_p�B +WTgclocals·f271231f400e778e0f59be25f7a26a56Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Infof��eH� %H;aw���H��0H�H�$H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$PH�\$ H�\$XH�\$(�H��0� + 0runtime.morestack_noctxt: "".std�$"".(*Logger).InfofP`"".args &type.[]interface {}"".formattype.string`F_p�B +WTgclocals·f271231f400e778e0f59be25f7a26a56Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Warnf��eH� %H;aw���H��0H�H�$H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$PH�\$ H�\$XH�\$(�H��0� + 0runtime.morestack_noctxt: "".std�$"".(*Logger).WarnfP`"".args &type.[]interface {}"".formattype.string`F_p�B +WTgclocals·f271231f400e778e0f59be25f7a26a56Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Warningf��eH� %H;aw���H��0H�H�$H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$PH�\$ H�\$XH�\$(�H��0� + 0runtime.morestack_noctxt: "".std�*"".(*Logger).WarningfP`"".args &type.[]interface {}"".formattype.string`F_p�B +WTgclocals·f271231f400e778e0f59be25f7a26a56Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Errorf��eH� %H;aw���H��0H�H�$H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$PH�\$ H�\$XH�\$(�H��0� + 0runtime.morestack_noctxt: "".std�&"".(*Logger).ErrorfP`"".args &type.[]interface {}"".formattype.string`F_p�B +WTgclocals·f271231f400e778e0f59be25f7a26a56Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Panicf��eH� %H;aw���H��0H�H�$H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$PH�\$ H�\$XH�\$(�H��0� + 0runtime.morestack_noctxt: "".std�&"".(*Logger).PanicfP`"".args &type.[]interface {}"".formattype.string`F_p�B +WTgclocals·f271231f400e778e0f59be25f7a26a56Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Fatalf��eH� %H;aw���H��0H�H�$H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$PH�\$ H�\$XH�\$(�H��0� + 0runtime.morestack_noctxt: "".std�&"".(*Logger).FatalfP`"".args &type.[]interface {}"".formattype.string`F_p�B +WTgclocals·f271231f400e778e0f59be25f7a26a56Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Debugln��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�("".(*Logger).Debugln0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Println��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�("".(*Logger).Println0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Infoln��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�&"".(*Logger).Infoln0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Warnln��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�&"".(*Logger).Warnln0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Warningln��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�,"".(*Logger).Warningln0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Errorln��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�("".(*Logger).Errorln0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Panicln��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�("".(*Logger).Panicln0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�"".Fatalln��eH� %H;aw���H�� H�H�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + 0runtime.morestack_noctxt: "".std�("".(*Logger).Fatalln0@"".args&type.[]interface {}@2?P�. +C Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�*"".prefixFieldClashes� � eH� %H;aw���H��`H�H� H�kH�H�$H�\$hH�\$H�L$PH�L$H�l$XH�l$�H�L$ �\$(H��������H�H�+H�l$@H�kH�l$HH�H� H�kH�H�$H�\$hH�\$H�L$PH�L$H�l$XH�l$�H�\$ H���OH�+H�l$0H�kH�l$8H�H�$H�\$hH�\$H�\$@H�\$H�\$0H�\$�H�H� H�kH�H�$H�\$hH�\$H�L$PH�L$H�l$XH�l$�H�L$ �\$(H��������H�H�+H�l$@H�kH�l$HH�H� H�kH�H�$H�\$hH�\$H�L$PH�L$H�l$XH�l$�H�\$ H���FH�+H�l$0H�kH�l$8H�H�$H�\$hH�\$H�\$@H�\$H�\$0H�\$�H�H� H�kH�H�$H�\$hH�\$H�L$PH�L$H�l$XH�l$�H�L$ �\$(H��������H�H�+H�l$@H�kH�l$HH�H� H�kH�H�$H�\$hH�\$H�L$PH�L$H�l$XH�l$�H�\$ H��tDH�+H�l$0H�kH�l$8H�H�$H�\$hH�\$H�\$@H�\$H�\$0H�\$�H��`É븉�I����������@����������7���: + 0runtime.morestack_noctxt: go.string."time"Vtype."".Fields�4runtime.mapaccess2_faststr�.go.string."fields.time"� go.string."time"�type."".Fields�4runtime.mapaccess1_faststr�type."".Fields�$runtime.mapassign1�go.string."msg"�type."".Fields�4runtime.mapaccess2_faststr�,go.string."fields.msg"�go.string."msg"�type."".Fields�4runtime.mapaccess1_faststr�type."".Fields�$runtime.mapassign1�"go.string."level"�type."".Fields� 4runtime.mapaccess2_faststr� 0go.string."fields.level"� +"go.string."level"� +type."".Fields� +4runtime.mapaccess1_faststr� type."".Fields� $runtime.mapassign1�"".autotmp_0193$type.*interface {}"".autotmp_0192$type.*interface {}"".autotmp_0190"type.interface {}"".autotmp_0189type.string"".autotmp_0188type.string"".autotmp_0187type.string"".autotmp_0186"type.interface {}"".autotmp_0185type.string"".autotmp_0184type.string"".autotmp_0183type.string"".autotmp_0182_"type.interface {}"".autotmp_0181type.string"".autotmp_0180?type.string"".autotmp_0179type.string"".datatype."".Fields����4�XDP �P �P �*QqN<qN<qJ>Tgclocals·15395a9df917b4c9aa74d5c6c7e1ebf4Tgclocals·cfe802ef097eb87dc1d2f379757036b4�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/formatter.go�""".LevelHooks.Add��eH� %H�D$�H;Aw���H���H��$�H�$H��$�H�[(��H�T$H�D$H�L$H��$�H��$�H��$�H��$�1�H��$�H�D$HH��$�H��H�l$HH9��zH�D$X�(H�L$P@�l$G@�l$FH�H�$H��$�H�\$H�\$FH�\$�H�\$H���5H�H�KH�[H��$�H��$�H��$�H��H)�H��}FH�H�$H�T$`H�T$H�L$hH�L$H�D$pH�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$hH�D$pH��H�T$`H��Hk�H�H�$H��$�H�\$H��$�H�\$�H�\$`H�\$xH�\$hH��$�H�\$pH��$�H�H�$H��$�H�\$H�\$GH�\$H�\$xH�\$�H�D$XH��H�L$PH��H�l$HH9������H���É����� +*0runtime.morestack_noctxtt +�$type."".LevelHooks�$runtime.mapaccess1�type.[]"".Hook�"runtime.growslice�2runtime.writebarrieriface�$type."".LevelHooks�$runtime.mapassign10�"".autotmp_0204�type.[]"".Hook"".autotmp_0202�type.*"".Level"".autotmp_0201�type.int"".autotmp_0200�type.int"".autotmp_0199�type.[]"".Hook"".autotmp_0198�type.[]"".Hook"".autotmp_0197�type."".Level"".autotmp_0196�type."".Level"".autotmp_0195_type.[]"".Level"".autotmp_0194/type.[]"".Level"".hooktype."".Hook"".hooks$type."".LevelHooks"���� �&"~� :��U6Tgclocals·a02efc190d1c7709e4c72531a85b968dTgclocals·5347b08d42ef15c0183233bde05091ab�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/hooks.go�$"".LevelHooks.Fire��eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$���$��\$'H�H�$H��$�H�\$H�\$'H�\$�H�\$H��� H�H�CH�kH��$�H��$�H��$�H��$�1�H�D$xH�D$(H�T$pH��H�l$(H9���H�D$8H����H�H�hH�L$0H�T$`H�l$hH��$�H�\$H�l$HH�,$H�T$@H�Z ��H�L$H�D$H�D$XH��H�L$PtH��$�H��$�H�Ġ�H�D$8H�L$0H��H��H�l$(H9��i���HDŽ$�HDŽ$�H�ĠÉ�Q��������� + +*0runtime.morestack_noctxt�$type."".LevelHooks�$runtime.mapaccess1� +P�"".autotmp_0219type."".Hook"".autotmp_0218�type.*"".Hook"".autotmp_0217�type.int"".autotmp_0216�type.int"".autotmp_0214_type.[]"".Hook"".autotmp_0213/type.[]"".Hook"".autotmp_0212�type."".Level "".err�type.error"".hook�type."".Hook "".~r20type.error"".entry type.*"".Entry"".leveltype."".Level"".hooks$type."".LevelHooks&"����>��� 6:�; h��Tgclocals·7ce35767da505d40dfb8f85871f02969Tgclocals·7e4aab61b173caafc98b406c57151fa1�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/hooks.go�4"".(*JSONFormatter).Format��eH� %H��$����H;Aw���H��HDŽ$�HDŽ$�HDŽ$�HDŽ$�HDŽ$�H��$�H�]H��tH�H��H�H�$H��H��H�\$�H�\$H�\$@H��$�H�kH��$@1��H�H�$H�l$H��$@H�\$�H��$@1�H9���H��$HH����H� H�CH��$@H����H�+H��$�H�kH��$�H�L$HH�D$PH��$�H� $H��$�H�D$��\$H�H�$H��$�H�\$H��$�H�\$�H��$�H��$�H�T$H�T$hH�L$ H�L$p�\$(����H��$�H��$�H� $H�Z ��H�\$H��$�H�\$H��$�H�H�$H��$�H�\$�H�\$H��$�H�\$H��$�H�H�$H�\$@H�\$H��$�H�\$H��$�H�\$�H��$@H�$�H��$@1�H9��s���H�\$@H�$�H��$�H����H� H�CH��uH�H� H�CH�H�+H��$�H�sH��$�H��$�H���FH�wH�<$H�H�H�H�L$xH�L$H��$�H�D$ �H�\$(H��$�H�\$0H��$�H�H�$H��$�H�\$�H�\$H��$�H�\$H��$�H�H�$H�\$@H�\$H��$�H�\$H��$�H�\$�H�H�+H��$�H�kH��$�H�H�$H��$�H�\$H�|$�IH�D$0�H�\$H��$�H�\$H��$�H�H�$H�\$@H�\$H��$�H�\$H��$�H�\$�H�H�+H��$�H�kH��$�H��$��k(@�,$�H�\$H��$�H�\$H��$�H�H�$H��$�H�\$�H�\$H��$�H�\$H��$�H�H�$H�\$@H�\$H��$�H�\$H��$�H�\$�H�D$@H� H��$�H� $H��$�H�D$�H�l$H��$�H�L$H��$H�D$ H��$H�T$(H�t$0H��$�H��H��$�� H��$�H�H�CH��$�H����H��H��H��$(H��$0H��$8H�$H�t$�H�L$H�D$H��$(H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$(H�\$H��$0H�\$H��$8H�\$ �H�L$(H�D$0HDŽ$�HDŽ$�HDŽ$�H��$�H��$�H�ĐÉ� ���H��H��H)�H��}OH�H�$H��$H�T$H��$H�L$H��$ H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H� +� +H��$�H��$�H��$�HDŽ$�HDŽ$�H�ĐÉ%�����������a���H��$�H��$�H��$�H��$�H�L$XH��$�H�D$`H��$�H�H�$H�\$@H�\$H��$�H�\$H��$�H�\$�������e�����E���Z +00runtime.morestack_noctxt�type."".Fields�runtime.makemap�� runtime.duffzero�type."".Fields�&runtime.mapiterinit�$runtime.efacethash�type.error�$runtime.assertE2I2� +�type.string�runtime.convT2E�type."".Fields� $runtime.mapassign1� &runtime.mapiternext� +*"".prefixFieldClashes� +Jgo.string."2006-01-02T15:04:05Z07:00"� + go.string."time"�  time.Time.Format� type.string� runtime.convT2E� type."".Fields�$runtime.mapassign1�go.string."msg"�type.string�runtime.convT2E�type."".Fields�$runtime.mapassign1�"go.string."level"�"".Level.String�type.string�runtime.convT2E�type."".Fields�$runtime.mapassign1�type."".Fields�*encoding/json.Marshal�runtime.convI2E�2runtime.writebarrieriface�`go.string."Failed to marshal fields to JSON, %v"�fmt.Errorf�type.[]uint8�"runtime.growslice�type."".Fields�$runtime.mapassign1p�J"".autotmp_0250type.int"".autotmp_0249type.int"".autotmp_0248�type.[]uint8"".autotmp_0247"type.interface {}"".autotmp_0245�&type.[]interface {}"".autotmp_0244type.uint32"".autotmp_0242"type.interface {}"".autotmp_0241�"type.interface {}"".autotmp_0238�(type.[1]interface {}"".autotmp_0237type."".Fields"".autotmp_0236"type.interface {}"".autotmp_0235type.string"".autotmp_0234type.string"".autotmp_0233"type.interface {}"".autotmp_0232type.string"".autotmp_0231"type.interface {}"".autotmp_0230type.string"".autotmp_0229type.string"".autotmp_0228"type.interface {}"".autotmp_0227type.string"".autotmp_0226�"type.interface {}"".autotmp_0225�type.string"".autotmp_0224�type.string"".autotmp_0223�Btype.map.iter[string]interface {}"".autotmp_0222type."".Fields "".err�type.error"".serialized�type.[]uint8$"".timestampFormat�type.string"".v�"type.interface {}"".v�type.error"".v�"type.interface {}"".k�type.string"".data�type."".Fields "".~r2Ptype.error "".~r1 type.[]uint8"".entrytype.*"".Entry"".f,type.*"".JSONFormatter*%�� �������na>��� $���c��   ^ d�G}}4N! �7NMN37N*�v�� Tgclocals·91781b467bdd49442cfecbf49067c104Tgclocals·859165c97b106654e8e33715962a8293�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/json_formatter.go� "".New��eH� %H;aw���H��XH�H�$�H�\$H�\$0H�H�$H�D$�H�\$H�\$(H�H�$�H�L$H��H�L$ H���G1��H�1�H9���H� $H�<$��H� H�D$HH�D$H�L$PH�L$�H�1�H9�t|H�\$ H�$H�<$tcH�$H�L$0H�D$8H�D$H�L$@H�L$�H�\$ H�$H�<$t'H�$H�\$(H�\$�H�D$ �@(H�D$`H��XÉ%�Љ%�H�H�$H�H�\$H�H�\$�H�D$�R����%����H�H�$H�H�\$H�H�\$�H�L$ H�D$����������. + 0runtime.morestack_noctxt:*type."".TextFormatterL"runtime.newobjectn$type."".LevelHooks�runtime.makemap�type."".Logger�"runtime.newobject�� runtime.duffzero�4go.itab.*os.File.io.Writer�os.Stderr�2runtime.writebarrieriface�Lgo.itab.*"".TextFormatter."".Formatter�2runtime.writebarrieriface�.runtime.writebarrierptr�,type.*"".TextFormatter�"type."".Formatter�Lgo.itab.*"".TextFormatter."".Formatter� runtime.typ2Itab�type.*os.File�type.io.Writer�4go.itab.*os.File.io.Writer� runtime.typ2Itab� +"".autotmp_0265type.*uint8"".autotmp_0263otype.*"".Logger"".autotmp_0262_$type."".LevelHooks"".autotmp_0261O,type.*"".TextFormatter "".~r0type.*"".Logger������:\#./` +-2"%#_A$M]Tgclocals·e9c510091732f30fce387c9f1e977134Tgclocals·87b2493d48b5216b7adc3dac672105cb�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�,"".(*Logger).WithField��eH� %H;aw���H��HH�\$PH�\$0H�H�$H�D$�H�\$H�\$@H�H�$�H�L$H��H����1��H�L$8H� $H�<$��H�\$0H�\$�H�\$8H�$H�<$tYH�$H�\$@H�\$�H�\$8H�$H�\$XH�\$H�\$`H�\$H�\$hH�\$H�\$pH�\$ �H�\$(H�\$xH��HÉ%랉%�s�����Q��� + 0runtime.morestack_noctxtNtype."".Fieldsrruntime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�*"".(*Entry).WithField`�"".autotmp_0273type.*"".Entry"".autotmp_0272type.*"".Entry"".autotmp_0271type.*"".Entry"".autotmp_0270type."".Fields"".logger/type.*"".Logger "".~r2Ptype.*"".Entry"".value0"type.interface {} "".keytype.string"".loggertype.*"".Logger����$� t�8<$68Tgclocals·0a89f44bdb06c71b1e3fde611d9babf4Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�."".(*Logger).WithFields��eH� %H;aw���H��0H�\$8H�\$H�H�$H�D$�H�\$H�\$(H�H�$�H�L$H��H����1��H�L$ H� $H�<$tcH�\$H�\$�H�\$ H�$H�<$t;H�$H�\$(H�\$�H�\$ H�$H�\$@H�\$�H�\$H�\$HH��0É%뼉%딉�v��� + 0runtime.morestack_noctxtNtype."".Fieldsrruntime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�,"".(*Entry).WithFields0`"".autotmp_0277type.*"".Entry"".autotmp_0276type.*"".Entry"".autotmp_0275type.*"".Entry"".autotmp_0274type."".Fields"".logger/type.*"".Logger "".~r1 type.*"".Entry"".fieldstype."".Fields"".loggertype.*"".Logger`�_`&���88$:Tgclocals·ab01a2d55089ff50c402006df1039c39Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�&"".(*Logger).Debugf��eH� %H;aw���H��HH�D$P�X(����H�D$0H�H�$H�D$�H�\$H�\$@H�H�$�H�L$H��H����1��H�L$8H� $H�<$��H�\$0H�\$�H�\$8H�$H�<$tYH�$H�\$@H�\$�H�\$8H�$H�\$XH�\$H�\$`H�\$H�\$hH�\$H�\$pH�\$ H�\$xH�\$(�H��HÉ%랉%�s�����Q��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�$"".(*Entry).Debugf`�"".autotmp_0280type.*"".Entry"".autotmp_0279type.*"".Entry"".autotmp_0278type."".Fields"".logger/type.*"".Logger"".args0&type.[]interface {}"".formattype.string"".loggertype.*"".Logger����'�� �'E<$@1Tgclocals·fc96ae191c2547955912928601e85959Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�$"".(*Logger).Infof��eH� %H;aw���H��HH�D$P�X(����H�D$0H�H�$H�D$�H�\$H�\$@H�H�$�H�L$H��H����1��H�L$8H� $H�<$��H�\$0H�\$�H�\$8H�$H�<$tYH�$H�\$@H�\$�H�\$8H�$H�\$XH�\$H�\$`H�\$H�\$hH�\$H�\$pH�\$ H�\$xH�\$(�H��HÉ%랉%�s�����Q��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�""".(*Entry).Infof`�"".autotmp_0283type.*"".Entry"".autotmp_0282type.*"".Entry"".autotmp_0281type."".Fields"".logger/type.*"".Logger"".args0&type.[]interface {}"".formattype.string"".loggertype.*"".Logger����'�� �'E<$@1Tgclocals·fc96ae191c2547955912928601e85959Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�&"".(*Logger).Printf��eH� %H;aw���H��HH�\$PH�\$0H�H�$H�D$�H�\$H�\$@H�H�$�H�L$H��H����1��H�L$8H� $H�<$��H�\$0H�\$�H�\$8H�$H�<$tYH�$H�\$@H�\$�H�\$8H�$H�\$XH�\$H�\$`H�\$H�\$hH�\$H�\$pH�\$ H�\$xH�\$(�H��HÉ%랉%�s�����Q��� + 0runtime.morestack_noctxtNtype."".Fieldsrruntime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�$"".(*Entry).Printf`�"".autotmp_0286type.*"".Entry"".autotmp_0285type.*"".Entry"".autotmp_0284type."".Fields"".logger/type.*"".Logger"".args0&type.[]interface {}"".formattype.string"".loggertype.*"".Logger����$���$8<$@.Tgclocals·fc96ae191c2547955912928601e85959Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�$"".(*Logger).Warnf��eH� %H;aw���H��HH�D$P�X(����H�D$0H�H�$H�D$�H�\$H�\$@H�H�$�H�L$H��H����1��H�L$8H� $H�<$��H�\$0H�\$�H�\$8H�$H�<$tYH�$H�\$@H�\$�H�\$8H�$H�\$XH�\$H�\$`H�\$H�\$hH�\$H�\$pH�\$ H�\$xH�\$(�H��HÉ%랉%�s�����Q��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�""".(*Entry).Warnf`�"".autotmp_0289type.*"".Entry"".autotmp_0288type.*"".Entry"".autotmp_0287type."".Fields"".logger/type.*"".Logger"".args0&type.[]interface {}"".formattype.string"".loggertype.*"".Logger����'�� �'E<$@1Tgclocals·fc96ae191c2547955912928601e85959Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�*"".(*Logger).Warningf��eH� %H;aw���H��HH�D$P�X(����H�D$0H�H�$H�D$�H�\$H�\$@H�H�$�H�L$H��H����1��H�L$8H� $H�<$��H�\$0H�\$�H�\$8H�$H�<$tYH�$H�\$@H�\$�H�\$8H�$H�\$XH�\$H�\$`H�\$H�\$hH�\$H�\$pH�\$ H�\$xH�\$(�H��HÉ%랉%�s�����Q��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�""".(*Entry).Warnf`�"".autotmp_0292type.*"".Entry"".autotmp_0291type.*"".Entry"".autotmp_0290type."".Fields"".logger/type.*"".Logger"".args0&type.[]interface {}"".formattype.string"".loggertype.*"".Logger����'�� �'E<$@1Tgclocals·fc96ae191c2547955912928601e85959Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�&"".(*Logger).Errorf��eH� %H;aw���H��HH�D$P�X(����H�D$0H�H�$H�D$�H�\$H�\$@H�H�$�H�L$H��H����1��H�L$8H� $H�<$��H�\$0H�\$�H�\$8H�$H�<$tYH�$H�\$@H�\$�H�\$8H�$H�\$XH�\$H�\$`H�\$H�\$hH�\$H�\$pH�\$ H�\$xH�\$(�H��HÉ%랉%�s�����Q��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�$"".(*Entry).Errorf`�"".autotmp_0295type.*"".Entry"".autotmp_0294type.*"".Entry"".autotmp_0293type."".Fields"".logger/type.*"".Logger"".args0&type.[]interface {}"".formattype.string"".loggertype.*"".Logger����'�� �'E<$@1Tgclocals·fc96ae191c2547955912928601e85959Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�&"".(*Logger).Fatalf��eH� %H;aw���H��HH�D$P�X(����H�D$0H�H�$H�D$�H�\$H�\$@H�H�$�H�L$H��H����1��H�L$8H� $H�<$��H�\$0H�\$�H�\$8H�$H�<$tfH�$H�\$@H�\$�H�\$8H�$H�\$XH�\$H�\$`H�\$H�\$hH�\$H�\$pH�\$ H�\$xH�\$(�H�$�H��HÉ%둉%�f�����D��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�$"".(*Entry).Fatalf�os.Exit`�"".autotmp_0298type.*"".Entry"".autotmp_0297type.*"".Entry"".autotmp_0296type."".Fields"".logger/type.*"".Logger"".args0&type.[]interface {}"".formattype.string"".loggertype.*"".Logger����*�� � *E<$@ATgclocals·fc96ae191c2547955912928601e85959Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�&"".(*Logger).Panicf��eH� %H;aw���H��HH�D$P�X(����H�D$0H�H�$H�D$�H�\$H�\$@H�H�$�H�L$H��H����1��H�L$8H� $H�<$��H�\$0H�\$�H�\$8H�$H�<$tYH�$H�\$@H�\$�H�\$8H�$H�\$XH�\$H�\$`H�\$H�\$hH�\$H�\$pH�\$ H�\$xH�\$(�H��HÉ%랉%�s�����Q��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�$"".(*Entry).Panicf`�"".autotmp_0301type.*"".Entry"".autotmp_0300type.*"".Entry"".autotmp_0299type."".Fields"".logger/type.*"".Logger"".args0&type.[]interface {}"".formattype.string"".loggertype.*"".Logger����'�� �'E<$@1Tgclocals·fc96ae191c2547955912928601e85959Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�$"".(*Logger).Debug��eH� %H;aw���H��8H�D$@�X(����H�D$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tmH�\$ H�\$�H�\$(H�$H�<$tEH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H��8É%벉%늉�l��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�""".(*Entry).Debug@p "".autotmp_0304type.*"".Entry"".autotmp_0303type.*"".Entry"".autotmp_0302type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op�� �E8$,)Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�""".(*Logger).Info��eH� %H;aw���H��8H�D$@�X(����H�D$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tmH�\$ H�\$�H�\$(H�$H�<$tEH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H��8É%벉%늉�l��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr� "".(*Entry).Info@p "".autotmp_0307type.*"".Entry"".autotmp_0306type.*"".Entry"".autotmp_0305type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op�� �E8$,)Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�$"".(*Logger).Print��eH� %H;aw���H��8H�\$@H�\$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tmH�\$ H�\$�H�\$(H�$H�<$tEH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H��8É%벉%늉�l��� + 0runtime.morestack_noctxtNtype."".Fieldsrruntime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr� "".(*Entry).Info@p "".autotmp_0310type.*"".Entry"".autotmp_0309type.*"".Entry"".autotmp_0308type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op���88$,&Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�""".(*Logger).Warn��eH� %H;aw���H��8H�D$@�X(����H�D$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tmH�\$ H�\$�H�\$(H�$H�<$tEH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H��8É%벉%늉�l��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr� "".(*Entry).Warn@p "".autotmp_0313type.*"".Entry"".autotmp_0312type.*"".Entry"".autotmp_0311type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op�� �E8$,)Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�("".(*Logger).Warning��eH� %H;aw���H��8H�D$@�X(����H�D$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tmH�\$ H�\$�H�\$(H�$H�<$tEH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H��8É%벉%늉�l��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr� "".(*Entry).Warn@p "".autotmp_0316type.*"".Entry"".autotmp_0315type.*"".Entry"".autotmp_0314type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op�� �E8$,)Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�$"".(*Logger).Error��eH� %H;aw���H��8H�D$@�X(����H�D$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tmH�\$ H�\$�H�\$(H�$H�<$tEH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H��8É%벉%늉�l��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�""".(*Entry).Error@p "".autotmp_0319type.*"".Entry"".autotmp_0318type.*"".Entry"".autotmp_0317type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op�� �E8$,)Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�$"".(*Logger).Fatal��eH� %H;aw���H��8H�D$@�X(����H�D$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tzH�\$ H�\$�H�\$(H�$H�<$tRH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H�$�H��8É%륉%�z�����\��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�""".(*Entry).Fatal�os.Exit@p "".autotmp_0322type.*"".Entry"".autotmp_0321type.*"".Entry"".autotmp_0320type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op"�� � "E8$,9Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�$"".(*Logger).Panic��eH� %H;aw���H��8H�D$@�X(����H�D$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tmH�\$ H�\$�H�\$(H�$H�<$tEH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H��8É%벉%늉�l��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�""".(*Entry).Panic@p "".autotmp_0325type.*"".Entry"".autotmp_0324type.*"".Entry"".autotmp_0323type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op�� �E8$,)Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�("".(*Logger).Debugln��eH� %H;aw���H��8H�D$@�X(����H�D$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tmH�\$ H�\$�H�\$(H�$H�<$tEH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H��8É%벉%늉�l��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�&"".(*Entry).Debugln@p "".autotmp_0328type.*"".Entry"".autotmp_0327type.*"".Entry"".autotmp_0326type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op�� �E8$,)Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�&"".(*Logger).Infoln��eH� %H;aw���H��8H�D$@�X(����H�D$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tmH�\$ H�\$�H�\$(H�$H�<$tEH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H��8É%벉%늉�l��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�$"".(*Entry).Infoln@p "".autotmp_0331type.*"".Entry"".autotmp_0330type.*"".Entry"".autotmp_0329type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op�� �E8$,)Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�("".(*Logger).Println��eH� %H;aw���H��8H�\$@H�\$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tmH�\$ H�\$�H�\$(H�$H�<$tEH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H��8É%벉%늉�l��� + 0runtime.morestack_noctxtNtype."".Fieldsrruntime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�&"".(*Entry).Println@p "".autotmp_0334type.*"".Entry"".autotmp_0333type.*"".Entry"".autotmp_0332type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op���88$,&Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�&"".(*Logger).Warnln��eH� %H;aw���H��8H�D$@�X(����H�D$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tmH�\$ H�\$�H�\$(H�$H�<$tEH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H��8É%벉%늉�l��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�$"".(*Entry).Warnln@p "".autotmp_0337type.*"".Entry"".autotmp_0336type.*"".Entry"".autotmp_0335type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op�� �E8$,)Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�,"".(*Logger).Warningln��eH� %H;aw���H��8H�D$@�X(����H�D$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tmH�\$ H�\$�H�\$(H�$H�<$tEH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H��8É%벉%늉�l��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�$"".(*Entry).Warnln@p "".autotmp_0340type.*"".Entry"".autotmp_0339type.*"".Entry"".autotmp_0338type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op�� �E8$,)Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�("".(*Logger).Errorln��eH� %H;aw���H��8H�D$@�X(����H�D$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tmH�\$ H�\$�H�\$(H�$H�<$tEH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H��8É%벉%늉�l��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�&"".(*Entry).Errorln@p "".autotmp_0343type.*"".Entry"".autotmp_0342type.*"".Entry"".autotmp_0341type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op�� �E8$,)Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�("".(*Logger).Fatalln��eH� %H;aw���H��8H�D$@�X(����H�D$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tzH�\$ H�\$�H�\$(H�$H�<$tRH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H�$�H��8É%륉%�z�����\��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�&"".(*Entry).Fatalln�os.Exit@p "".autotmp_0346type.*"".Entry"".autotmp_0345type.*"".Entry"".autotmp_0344type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op"�� � "E8$,9Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�("".(*Logger).Panicln��eH� %H;aw���H��8H�D$@�X(����H�D$ H�H�$H�D$�H�\$H�\$0H�H�$�H�L$H��H����1��H�L$(H� $H�<$tmH�\$ H�\$�H�\$(H�$H�<$tEH�$H�\$0H�\$�H�\$(H�$H�\$HH�\$H�\$PH�\$H�\$XH�\$�H��8É%벉%늉�l��� + 0runtime.morestack_noctxthtype."".Fields�runtime.makemap�type."".Entry�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�&"".(*Entry).Panicln@p "".autotmp_0349type.*"".Entry"".autotmp_0348type.*"".Entry"".autotmp_0347type."".Fields"".logger/type.*"".Logger"".args&type.[]interface {}"".loggertype.*"".Loggerp�op�� �E8$,)Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·0442504e096b61648fffc20fe86cec66�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go�"".Level.String���\$��ws��uH�H�+H�l$H�kH�l$À�uH�H�+H�l$H�kH�l$À�uH�H�+H�l$H�kH�l$�H�H�+H�l$H�kH�l$À�uH�H�+H�l$H�kH�l$À�uH�H�+H�l$H�kH�l$À�u�H�H�+H�l$H�kH�l$�$"go.string."panic"`"go.string."fatal"�"go.string."error"�&go.string."unknown"�&go.string."warning"� go.string."info"�"go.string."debug"0 "".~r0type.string"".leveltype."".Level��:" +"Tgclocals·a73fd2a0c6f832642aa9216fd9c5e6beTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logrus.go�"".ParseLevel��eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$�H��$�H��$�H�L$PH� $H�\$XH�\$H�-L�D$L��H��H�H��H�L$PH�D$XH�\$ H���OH��u`H� $H�D$H�-L�D$L��H��H�H��H�L$PH�D$X�\$ ��t(Ƅ$�HDŽ$�HDŽ$�H�Ę�H��u`H� $H�D$H�-L�D$L��H��H�H��H�L$PH�D$X�\$ ��t(Ƅ$�HDŽ$�HDŽ$�H�Ę�H��uVH� $H�D$H�-L�D$L��H��H�H���\$ ��t(Ƅ$�HDŽ$�HDŽ$�H�Ę��D$?H��$�H�\$`H��$�H�\$hH�\$pH�H�CH�\$pH����H��H��H��$�H��$�H��$�H�H�$H�\$`H�\$�H�L$H�D$H��$�H�$H�L$@H�L$H�D$HH�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0�\$?��$�H��$�H��$�H�ĘÉ����H� $H�D$H�-L�D$L��H��H�H��H�L$PH�D$XH�\$ H����H��u`H� $H�D$H�-L�D$L��H��H�H��H�L$PH�D$X�\$ ��t(Ƅ$�HDŽ$�HDŽ$�H�Ę�H���,���H� $H�D$H�-L�D$L��H��H�H���\$ �������Ƅ$�HDŽ$�HDŽ$�H�Ę�H��u`H� $H�D$H�-L�D$L��H��H�H��H�L$PH�D$X�\$ ��t(Ƅ$�HDŽ$�HDŽ$�H�Ę�H���b���H� $H�D$H�-L�D$L��H��H�H���\$ ��u��/���2 +*0runtime.morestack_noctxt�"go.string."fatal"�"runtime.cmpstring�"go.string."debug"� runtime.eqstring�"go.string."error"� runtime.eqstring�"go.string."fatal"� runtime.eqstring�type.string�runtime.convT2E� 2runtime.writebarrieriface� Pgo.string."not a valid logrus Level: %q"� +fmt.Errorf� "go.string."panic"� "runtime.cmpstring�  go.string."info"�  runtime.eqstring�"go.string."panic"� runtime.eqstring� go.string."warn"� runtime.eqstring�&go.string."warning"� runtime.eqstringP�"".autotmp_0357�"type.interface {}"".autotmp_0355/&type.[]interface {}"".autotmp_0354�type.string"".autotmp_0352otype.string"".autotmp_0351O(type.[1]interface {}"".l�type."".Level "".~r20type.error "".~r1 type."".Level "".lvltype.stringf"����e��[��������c��e��C� PH:W>( >(4(�=>(<( +>(8 ,s��pY�`jTgclocals·db0f6b31ff49b3f025910ec03f9742faTgclocals·626b2db390378ab5b89c88b48426687f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logrus.go�"".IsTerminal��eH� %H�D$�H;Aw���H��H�H�|$P1��H�$6H�\$H�D$tH@H�\$PH�\$H�D$ H�D$(H�D$0�H�\$HH��tƄ$�H�Ę�Ƅ$��� + +*0runtime.morestack_noctxtJsyscall.Stdoutb� runtime.duffzero� syscall.Syscall6�"".termios�type."".Termios "".~r0type.bool"�m���"" E& +p0Tgclocals·a7a3692b8e27e823add69ec4239ba55fTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_notwindows.go�"".init·1��eH� %H;aw���H��@�H�$�L$H�D$H�H�$H�D$H�T$(H�T$�L$0�L$H�D$8H�D$ ���$�H��@� + 0runtime.morestack_noctxt6time.Now^ "".baseTimestamp�0runtime.writebarrierfat3�"".IsTerminal�"".isTerminal�"".autotmp_0361/type.time.Time�Z �6G +fTgclocals·3280bececceccd33cb74587feedb1f9fTgclocals·0528ab8f76149a707fd2f0025c2178a3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/text_formatter.go�"".miniTS��eH� %H;aw���H�� H�H�$��\$H�H�\$�H�l$I����&� .H��I��I��I��H��?I)�L�D$(H�� � + 0runtime.morestack_noctxt: "".baseTimestampN "".baseTimestampd "".baseTimestampxtime.Since@ "".~r0type.int@R?p +@V +;5Tgclocals·a7a3692b8e27e823add69ec4239ba55fTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/text_formatter.go�4"".(*TextFormatter).Format��eH� %H��$8���H;Aw���H��HHDŽ$`HDŽ$hHDŽ$pHDŽ$xHDŽ$�H��$XH�]H��tH�H��H�H�$H�D$H�D$�H�T$H�L$ H�D$(H��$�H��$�H��$�H��$XH�kH��$�1��H�H�$H�l$H��$�H�\$�H��$�1�H9��2H��$�H����H�+H�l$pH�kH�l$xH��$�H��$�H��$�H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H��$�H��Hk�H�H�$H�\$pH�\$H�\$xH�\$�H��$�H��$�H��$�H��$�H��$�H��$�H��$�H�$�H��$�1�H9������H��$P�]��u+H��$�H�$H��$�H�\$H��$�H�\$�H�H�$�H�|$H��H���[1��H�\$PH��$XH�kH�,$�H��$XH��$P�=�H�����������1�H�XH��uH�hH�H��H��H�H�����H�$H�\$PH�\$H�T$H��$�H�\$H��$�H�\$ H��$�H�\$(�H�\$PH�$�D$ +�H�L$PH�qH�QH�AH9�rVH� H��H)�H��H)�H��t H��H�H��H��$`H��$hH��$pHDŽ$xHDŽ$�H��H�� �X����H����H�jH�$H��H��H�H�H�H�hH�\$H��H��H�H��H�\$(H��$�H�\$0H��$�H�H�$H��$�H�\$�H�\$H�l$ H��H��H�H�H��$PH�$H�t$PH�t$H�5H�l$H��H�H��H��$X�j(@�,$�H�\$H��$�H�\$H��$�H�H�$H��$�H�\$�H�\$H�l$ H��H��H�H�H��$PH�$H�t$PH�t$H�5H�l$H��H�H��H�H�$H��$XH�\$H�|$�qH�D$0�H�\$H�l$ H��H��H�H�H��$PH�$H�t$PH�t$H�5H�l$H��H�H��H��$�H��$�H��$�H��$�1�H��$�H�D$@H��$�H��H�l$@H9��w���H�D$XH����H�H�@H�T$HH�L$`H�D$hH�H�$H��$XH�kH�l$H��$�H�L$H��$�H�D$�H�\$ H��tfH� H�kH��$PH�$H�\$PH�\$H�\$`H�\$H�\$hH�\$H��$�H�L$ H��$�H�l$(�H�D$XH�T$HH��H���!����떉�3����%������%����X������H������1����������������L +00runtime.morestack_noctxt�type.[]string�"runtime.makeslice�� runtime.duffzero�type."".Fields�&runtime.mapiterinit�type.[]string�"runtime.growslice�4runtime.writebarrierstring�&runtime.mapiternext� sort.Strings� "type.bytes.Buffer� +"runtime.newobject� +� runtime.duffzero� +*"".prefixFieldClashes� "".isTerminal� Jgo.string."2006-01-02T15:04:05Z07:00"� @"".(*TextFormatter).printColored� 2bytes.(*Buffer).WriteByte�$runtime.panicslice� time.Time.Format�type.string�runtime.convT2E� go.string."time"�D"".(*TextFormatter).appendKeyValue�"".Level.String�type.string�runtime.convT2E�"go.string."level"�D"".(*TextFormatter).appendKeyValue�type.string�runtime.convT2E�go.string."msg"�D"".(*TextFormatter).appendKeyValue�type."".Fields�4runtime.mapaccess1_faststr�D"".(*TextFormatter).appendKeyValuep�2"".autotmp_0386type.uint64"".autotmp_0385type.uint64"".autotmp_0383�type.string"".autotmp_0382�type.*string"".autotmp_0381type.int"".autotmp_0380type.int"".autotmp_0375type.int"".autotmp_0374type.[]string"".autotmp_0373�"type.interface {}"".autotmp_0372type.string"".autotmp_0371type.[]string"".autotmp_0370type.string"".autotmp_0369�type.string"".autotmp_0368�type.[]string"".autotmp_0367�Btype.map.iter[string]interface {}"".autotmp_0365�type.[]string"".autotmp_0364�type.int "".key�type.string"".b�$type.*bytes.Buffer"".k�type.string"".keys�type.[]string "".~r2Ptype.error "".~r1 type.[]uint8"".entrytype.*"".Entry"".f,type.*"".TextFormatter%�������xa]s�$+.% + ?s ��qc�  P�d�WAOC�� E +7W7q�[pTgclocals·71f03f9031d1479950f850ff028ecf79Tgclocals·cfaa0e57cb7ad41983cddbe4e85a52ca�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/text_formatter.go�@"".(*TextFormatter).printColored�(�'eH� %H��$����H;Aw���H���H��$��i(@���� @���e H�D$P�i(@�,$�H�L$H�D$H��$H� $H��$H�D$�H�t$PL��$�H��$�H�L$H�\$H����H��A�X����H��$�H�t$pH��$�H��$�H��$�H��$��H�$H�\$hH�H��$�1�H9��IH��$�1��H��$�H���!H��H��H��$ H��$(H��$0H�H�$H�\$pH�\$�H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$�H�L$H�D$H��$ H��H�$H��$�H�L$H��$�H�D$�H�H�$H�\$hH�\$�H�L$H�D$H��$ H�� H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$H�|$��H�D$0�H�L$H�D$H��$ H��0H�$H��$�H�L$H��$�H�D$�H��$�H��$�H��$�H�$H��$�H�L$H�H�l$H��H��H�H�H��$ H�\$ H��$(H�\$(H��$0H�\$0�H��$�H��$�H��$�H��$H1�H��$@H�D$XH��$8H��H�l$XH9���H�D$xH����H�H�@H�T$`H��$�H��$�H�H�$H��$�H�kH�l$H��$H�L$H��$H�D$�H�\$ H���YH� H�kH��$�H��$�H��$�H��$�H�\$PH�\$hH��$�H��$�H��$�H��$�H�H��$�1�H9���H��$P1��H��$PH����H��H��H��$ H��$(H��$0H�H�$H�\$hH�\$�H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$�H�L$H�D$H��$ H��H�$H��$�H�L$H��$�H�D$�H��$ H�� H�$H��$�H�\$H��$�H�\$�H��$�H��$�H��$�H�$H��$�H�L$H�H�l$H��H��H�H�H��$ H�\$ H��$(H�\$(H��$0H�\$0�H�D$xH�T$`H��H��H�l$XH9��{���H���É�m���H�H�$H�H�\$H�H�\$�H�\$H��$������������3����%�!���������H�H�$H�H�\$H�H�\$�H�\$H��$��}���H��$�H�t$pH��$�H��$�H��$�H��$�H��$�H����H�oH�<$H��H�H�H�I�hH�\$H��H��H�H��H�\$(H��$�H�\$0H��$�H�H��$�1�H9��-H��$�1��H��$�H���H��H��H��$ H��$(H��$0H�H�$H�\$pH�\$�H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$�H�L$H�D$H��$ H��H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$�H�L$H�D$H��$ H�� H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$H�|$��H�D$0�H�L$H�D$H��$ H��0H�$H��$�H�L$H��$�H�D$�H��$�H��$�H��$�H�$H��$�H�L$H�H�l$H��H��H�H�H��$ H�\$ H��$(H�\$(H��$0H�\$0�������%�@���������H�H�$H�H�\$H�H�\$�H�\$H��$�������5���� @�������H�D$P"����@���y���@��uH�D$P!�n���@��u�H�D$P%�Z���� +00runtime.morestack_noctxt�"".Level.String�strings.ToUpper�"".miniTS�>go.itab.*bytes.Buffer.io.Writer�� runtime.duffzero�type.int�runtime.convT2E�2runtime.writebarrieriface�type.string�runtime.convT2E�2runtime.writebarrieriface�type.int�runtime.convT2E� 2runtime.writebarrieriface� type.string� +runtime.convT2E� +2runtime.writebarrieriface� Tgo.string."\x1b[%dm%s\x1b[0m[%04d] %-44s "� fmt.Fprintf�type."".Fields�4runtime.mapaccess1_faststr�>go.itab.*bytes.Buffer.io.Writer�� runtime.duffzero�type.int�runtime.convT2E�2runtime.writebarrieriface�type.string�runtime.convT2E�2runtime.writebarrieriface�2runtime.writebarrieriface�Bgo.string." \x1b[%dm%s\x1b[0m=%v"�fmt.Fprintf�$type.*bytes.Buffer�type.io.Writer�>go.itab.*bytes.Buffer.io.Writer� runtime.typ2Itab�$type.*bytes.Buffer�type.io.Writer�>go.itab.*bytes.Buffer.io.Writer� runtime.typ2Itab� time.Time.Format�>go.itab.*bytes.Buffer.io.Writer�� runtime.duffzero�type.int�runtime.convT2E�2runtime.writebarrieriface�type.string�runtime.convT2E� 2runtime.writebarrieriface� type.string� runtime.convT2E�!2runtime.writebarrieriface�!type.string�"runtime.convT2E�#2runtime.writebarrieriface�$Pgo.string."\x1b[%dm%s\x1b[0m[%s] %-44s "�%fmt.Fprintf�%$type.*bytes.Buffer�%type.io.Writer�%>go.itab.*bytes.Buffer.io.Writer�& runtime.typ2Itab�&$runtime.panicslice`�`"".autotmp_0437"type.interface {}"".autotmp_0436"type.interface {}"".autotmp_0434&type.[]interface {}"".autotmp_0433type.*uint8"".autotmp_0432type.string"".autotmp_0431�type.*string"".autotmp_0430type.int"".autotmp_0429type.int"".autotmp_0428"type.interface {}"".autotmp_0427"type.interface {}"".autotmp_0426"type.interface {}"".autotmp_0425"type.interface {}"".autotmp_0424*type.*[4]interface {}"".autotmp_0423&type.[]interface {}"".autotmp_0422type.*uint8"".autotmp_0421"type.interface {}"".autotmp_0420"type.interface {}"".autotmp_0419"type.interface {}"".autotmp_0418�"type.interface {}"".autotmp_0416�&type.[]interface {}"".autotmp_0415�type.*uint8"".autotmp_0412type.string"".autotmp_0411type.int"".autotmp_0410$type.*bytes.Buffer"".autotmp_0409�(type.[3]interface {}"".autotmp_0408"type.interface {}"".autotmp_0407type.string"".autotmp_0406�type.[]string"".autotmp_0405�type.string"".autotmp_0404type.string"".autotmp_0403type.int"".autotmp_0402$type.*bytes.Buffer"".autotmp_0401(type.[4]interface {}"".autotmp_0400�type.int"".autotmp_0399�type.string"".autotmp_0398�type.int"".autotmp_0397�$type.*bytes.Buffer"".autotmp_0396(type.[4]interface {}"".autotmp_0395�type.string"".autotmp_0394�type.string"".v�"type.interface {}"".k�type.string"".levelText�type.string"".levelColor�type.int"".keys0type.[]string"".entry type.*"".Entry"".b$type.*bytes.Buffer"".f,type.*"".TextFormatter%�� ������- + + +e�cq�AM�T  +  +  NV������kV[ ��@]Tgclocals·40341f41d3f930f66258415e27eda61dTgclocals·3d6acc133f31046c5a57fcb873eb249f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/text_formatter.go�"".needsQuoting��eH� %H;aw���H��@H�\$HH�\$0H�\$PH�\$81�H�L$(H�\$0H�$H�\$8H�\$H�L$�H�L$�D$ H��t2��a|��z~Ń�A|��Z~���0|��9~���-t���.t��D$XH��@��D$XH��@� + 0runtime.morestack_noctxt�&runtime.stringiter20�"".autotmp_0456/type.int"".autotmp_0455type.string "".~r1 type.bool"".texttype.string�x� ��G( +  +MSTgclocals·d7e8a62d22b1cde6d92b17a55c33fe8fTgclocals·8d600a433c6aaa81a4fe446d95c5546b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/text_formatter.go�D"".(*TextFormatter).appendKeyValue��eH� %H��$x���H;Aw���H��H��$H�$H��$ H�\$H��$(H�\$�H��$H�$�D$=�H��$0H��$8H��$�H� $H��$�H�D$��\$���\���H�H�$H��$�H�\$H��$�H�\$�H�T$H�T$`H�L$ H�L$h�\$(����H�$H�L$�H��$H�T$`H�L$h�\$��t1H�,$H�T$H�L$�H��$H�$�D$ �H���H�l$XH��$�H��$�H�H�D$P1�H9��H��$�H�H�CH��$�H����H��H��H��$�H��$�H��$H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H�L$XH�D$PH��$�H�$H��$�H�L$H�H�l$H��H��H�H�H��$�H�\$ H��$�H�\$(H��$H�\$0����������H�H�$H�H�\$H�H�\$�H�\$H�\$P����H�H�$H��$�H�\$H��$�H�\$�H�T$H��$�H�L$ H��$��\$(����H� $H�Z ��H�L$H�D$H��$�H� $H��$�H�D$�H��$�\$��t(H� $H��$�H�\$H��$�H�\$�����H�L$XH�H�D$P1�H9��H��$�H�H�CH��$�H����H��H��H��$�H��$�H��$H��$�H�$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H�L$XH�D$PH��$�H�$H��$�H�L$H�H�l$H��H��H�H�H��$�H�\$ H��$�H�\$(H��$H�\$0����������H�H�$H�H�\$H�H�\$�H�\$H�\$P����H��$�H�\$pH��$�H�\$xH��$H�\$XH�H�D$P1�H9���H��$�H�H�CH��$�H����H��H��H��$�H��$H��$�H�$H�\$pH�\$H�\$xH�\$�H�L$XH�D$PH��$�H�$H��$�H�L$H��$�H�\$H��$�H�\$H��$H�\$ ��K�����b���H�H�$H�H�\$H�H�\$�H�\$H�\$P����R +00runtime.morestack_noctxt�6bytes.(*Buffer).WriteString�2bytes.(*Buffer).WriteByte�$runtime.efacethash�type.string�$runtime.assertE2T2�"".needsQuoting�6bytes.(*Buffer).WriteString�2bytes.(*Buffer).WriteByte�>go.itab.*bytes.Buffer.io.Writer�type.string�runtime.convT2E�2runtime.writebarrieriface�go.string."%q"� fmt.Fprintf� +$type.*bytes.Buffer� +type.io.Writer� +>go.itab.*bytes.Buffer.io.Writer� + runtime.typ2Itab� type.error� $runtime.assertE2I2� +� "".needsQuoting� 6bytes.(*Buffer).WriteString�>go.itab.*bytes.Buffer.io.Writer�runtime.convI2E�2runtime.writebarrieriface�go.string."%q"�fmt.Fprintf�$type.*bytes.Buffer�type.io.Writer�>go.itab.*bytes.Buffer.io.Writer� runtime.typ2Itab�>go.itab.*bytes.Buffer.io.Writer�2runtime.writebarrieriface�fmt.Fprint�$type.*bytes.Buffer�type.io.Writer�>go.itab.*bytes.Buffer.io.Writer� runtime.typ2Itab`�8"".autotmp_0482*type.*[1]interface {}"".autotmp_0481&type.[]interface {}"".autotmp_0480type.*uint8"".autotmp_0479"type.interface {}"".autotmp_0478*type.*[1]interface {}"".autotmp_0477&type.[]interface {}"".autotmp_0476type.*uint8"".autotmp_0475�"type.interface {}"".autotmp_0473/&type.[]interface {}"".autotmp_0472�type.*uint8"".autotmp_0469�"type.interface {}"".autotmp_0468$type.*bytes.Buffer"".autotmp_0467(type.[1]interface {}"".autotmp_0466$type.*bytes.Buffer"".autotmp_0465(type.[1]interface {}"".autotmp_0464type.bool"".autotmp_0462otype.string"".autotmp_0461�$type.*bytes.Buffer"".autotmp_0460O(type.[1]interface {}"".autotmp_0459type.bool"".value�"type.interface {}"".errmsg�type.string"".value�type.error"".value�type.string"".value@"type.interface {} "".key type.string"".b$type.*bytes.Buffer"".f,type.*"".TextFormatter%����� � h�%+�*�> R0# �  >�>TKDj/��495 +%5 ��4�O4Tgclocals·81e712ea8ab1acd9cb646b715ca7f2cdTgclocals·d0bfb266a2b14e29993ffd436ee6f54e�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/text_formatter.go�&"".(*Logger).Writer��eH� %H;aw���H��8�H�$H�\$H�\$ H�\$@H�$H�D$H� Qj�YYH�L$ H�H�H�$H�L$H�H�D$(H�D$H�T$0H�T$�H�\$ H�\$HH��8� + 0runtime.morestack_noctxt6io.Pipe|:"".(*Logger).writerScanner·f�runtime.newproc�*"".writerFinalizer·f�&type.*io.PipeWriter�2type.func(*io.PipeWriter)�(runtime.SetFinalizer p"".writer/&type.*io.PipeWriter "".~r0&type.*io.PipeWriter"".loggertype.*"".Loggerp)Jo +�<+[Tgclocals·2148c3737b2bb476685a1100a2e8343eTgclocals·61e2515c69061b8fed0e66ece719f936�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/writer.go�4"".(*Logger).writerScanner��eH� %H��$`���H;Aw���H�� H��$0H�\$HH�1�H9���H�L$HH��$�H�D$pH��$�H�L$xH�H�$H�D$H�D$�H�\$H��$H�\$ H��$H�\$(H��$H�H�$�H�|$H��H���1��H�L$@H� $H�<$��H�\$pH�\$H�\$xH�\$�H�\$@H�$H�<$��H�$H�H�\$�H�D$@H�@H�$H�<$�yH�$8H��$H�\$H��$H�\$H��$H�\$�H�\$@H�\$8H�\$8H�$�H�L$8�\$���UH�D$PH�D$XH���2H�i H�$H��H��H�H�H��H�L$H�D$ H��$�H��$�H�L$PH��$�H�D$XH��$�H��$�H�H�CH��$�H����H��H��H��$�H��$�H��$H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H��$(H�$H��$�H�\$H��$�H�\$H��$H�\$�������D���������H��H�D$`H�D$hH�L$0H����H�A`H�IhH��$�H��$�H�-H9��^H�$H�L$H�-H�l$H�-H�l$�H�T$0�\$ ���%1�1�H�L$`H�D$hH��$�H��H��$���H��$�H�H�CH��$�H����H��H��H��$�H��$�H��$H� $H�D$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H��$(H�4$H�5H�l$H��H�H�H��$�H�\$H��$�H�\$ H��$H�\$(�H��$0H�$�H�� É�,���H��t H�J`H�Bh���������\����%�{����%�=����%����������H�H�$H�H�\$H�H�\$�H�D$�&���> +00runtime.morestack_noctxtj@go.itab.*io.PipeReader.io.Reader�type.[]uint8�"runtime.makeslice�$type.bufio.Scanner�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface�$bufio.ScanLines·f�.runtime.writebarrierptr�2runtime.writebarrierslice�*bufio.(*Scanner).Scan�2runtime.slicebytetostring� type.string� runtime.convT2E� +2runtime.writebarrieriface� $"".(*Logger).Print�  io.EOF�  io.EOF�  io.EOF� runtime.ifaceeq�runtime.convI2E�2runtime.writebarrieriface�^go.string."Error while reading from Writer: %s"�&"".(*Logger).Errorf�,io.(*PipeReader).Close�&type.*io.PipeReader�type.io.Reader�@go.itab.*io.PipeReader.io.Reader� runtime.typ2Itab �."".autotmp_0521"type.interface {}"".autotmp_0520*type.*[1]interface {}"".autotmp_0519&type.[]interface {}"".autotmp_0518�type.error"".autotmp_0517�"type.interface {}"".autotmp_0515_&type.[]interface {}"".autotmp_0514�type.string"".autotmp_0513�&type.*bufio.Scanner"".autotmp_0512&type.*bufio.Scanner"".autotmp_0510�type.io.Reader"".autotmp_0509(type.[1]interface {}"".autotmp_0508�type.string"".autotmp_0507(type.[1]interface {}"".autotmp_0505/type.[]uint8"".autotmp_0504�&type.*io.PipeReader "".~r0�type.errorbufio.s·2�&type.*bufio.Scanner "".~r0�type.stringbufio.r·2�type.io.Reader "".err�type.error"".scanner�&type.*bufio.Scanner"".reader&type.*io.PipeReader"".loggertype.*"".Logger%������ +>&%�!���_B�7Fw�m����Tgclocals·4a3831d274d2be9675c43f86862b9a60Tgclocals·a266581f5aa078217f9a8ca87caf12e6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/writer.go�$"".writerFinalizer`ZeH� %H;aw���H��H�\$ H�$�H��� + 0runtime.morestack_noctxtH,io.(*PipeWriter).Close0"".writer&type.*io.PipeWriter0/0< +# Tgclocals·519efd86263089ddb84df3cfe7fd2992Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/writer.go�"".init��eH� %H;aw���H�����t���uH���� ��������������H�$H�H�$H�D$���H���. + 0runtime.morestack_noctxt:"".initdone·R"".initdone·p"runtime.throwinit�"".initdone·�bufio.init�strings.init�runtime.init�syscall.init�log.init�sync.init�$encoding/json.init�time.init�os.init�io.init�fmt.init�bytes.init� "".New� "".std�.runtime.writebarrierptr�"".init·1�"".initdone·   x�@�+, +7yTgclocals·3280bececceccd33cb74587feedb1f9fTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/writer.go�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go�(type..hash.[8]string��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�runtime.strhash@` "".autotmp_0532type.int"".autotmp_0531type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*[8]string`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�$type..eq.[8]string��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$`H����H��Hk�H�H�3H�KH�\$hH��tvH��Hk�H�H�H�CH9�uVH�t$HH�4$H�L$PH�L$H�T$8H�T$H�D$@H�D$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt� runtime.eqstring@�"".autotmp_0536?type.string"".autotmp_0535type.string"".autotmp_0534_type.int"".autotmp_0533Otype.int "".~r30type.bool"".s type.uintptr"".qtype.*[8]string"".ptype.*[8]string&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�4type..hash.[8]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_0539type.int"".autotmp_0538type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[8]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�0type..eq.[8]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_0543?"type.interface {}"".autotmp_0542"type.interface {}"".autotmp_0541_type.int"".autotmp_0540Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[8]interface {}"".p*type.*[8]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�$"".(*Level).String��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$@H�D$HH�\$81�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�\$8�+@�,$�H�L$H�D$H�L$@H�D$HH��0� + 0runtime.morestack_noctxt�$go.string."logrus"�"go.string."Level"�$go.string."String"�"runtime.panicwrap�"".Level.String0` "".~r0type.string""..thistype.*"".Level`�_�� �1Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�"".Hook.Fire��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�D$@H�D$HH�\$8H�\$H�\$0H�$H�\$(H�[ ��H�L$H�D$H�L$@H�D$HH�� � + 0runtime.morestack_noctxt� +P@ "".~r10type.error""..anon0 type.*"".Entry""..thistype."".Hook@^?�� +^"Tgclocals·32f137afc3f53351f1adc065fe3b9f83Tgclocals·3280bececceccd33cb74587feedb1f9f�"".Hook.Levels��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�D$8H�D$@H�D$HH�\$0H�$H�\$(H�[(��H�T$H�L$H�D$H�T$8H�L$@H�D$HH�� � + 0runtime.morestack_noctxt� +P@ "".~r0 type.[]"".Level""..thistype."".Hook@g?�� +]3Tgclocals·9edc1f6d8fc7336ae101b48cbf822a45Tgclocals·3280bececceccd33cb74587feedb1f9f�("".(*LevelHooks).Add��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�\$81�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�\$8H�+H�,$H�\$@H�\$H�\$HH�\$�H��0� + 0runtime.morestack_noctxt~$go.string."logrus"�,go.string."LevelHooks"�go.string."Add"�"runtime.panicwrap�""".LevelHooks.Add0`"".hooktype."".Hook""..this&type.*"".LevelHooks`�_� +� +}3Tgclocals·284bdeb7a59f773ab3ee5877f5a03aa1Tgclocals·3280bececceccd33cb74587feedb1f9f�*"".(*LevelHooks).Fire��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$PH�D$XH�\$81�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�\$8H�+H�,$�\$@�\$H�\$HH�\$�H�L$H�D$ H�L$PH�D$XH��0� + 0runtime.morestack_noctxt�$go.string."logrus"�,go.string."LevelHooks"� go.string."Fire"�"runtime.panicwrap�$"".LevelHooks.FireP` "".~r20type.error"".entry type.*"".Entry"".leveltype."".Level""..this&type.*"".LevelHooks`�_� � �QTgclocals·dd0b304762533d7aaaca928b9df4b371Tgclocals·3280bececceccd33cb74587feedb1f9f�&"".Formatter.Format��eH� %H;aw���H��8H�Y H��t H�|$@H9;uH�#H�D$XH�D$`H�D$hH�D$pH�D$xH�\$PH�\$H�\$HH�$H�\$@H�[ ��H�t$H�l$H�T$ H�L$(H�D$0H�t$XH�l$`H�T$hH�L$pH�D$xH��8� + 0runtime.morestack_noctxt� +�p "".~r2`type.error "".~r10type.[]uint8""..anon0 type.*"".Entry""..this"type."".Formatterp�o�� +yGTgclocals·334cb8bc6294eb0b97ffb9b2c8e3805fTgclocals·3280bececceccd33cb74587feedb1f9f�4type..hash.[1]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_0552type.int"".autotmp_0551type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[1]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�0type..eq.[1]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_0556?"type.interface {}"".autotmp_0555"type.interface {}"".autotmp_0554_type.int"".autotmp_0553Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[1]interface {}"".p*type.*[1]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�6type..hash."".TextFormatter��eH� %H;aw���H�� H�\$(H�$H�<$��H�D$H�\$8H�\$�H�D$H�\$(H�$H�<$tgH�$H�D$H�D$8H�D$�H�D$H�\$(H�$H�<$t,H�$H�D$H�D$8H�D$�H�\$H�\$@H�� É%�ˉ%됉%�W��� + + 0runtime.morestack_noctxt�runtime.memhash�runtime.strhash�runtime.memhash@@ "".autotmp_0559type.uintptr"".autotmp_0558type.uintptr "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p,type.*"".TextFormatter@�?@'�� A�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�2type..eq."".TextFormatter��eH� %H;aw���H��HH�\$PH�$H�<$�:H�\$XH�\$H�|$�H�D$�H�\$PH�$H�<$��H�\$XH�\$H�|$��H�D$��\$��u +�D$hH��H�H�\$PH����H�sH�KH�\$XH��tyH�SH�CH9�ubH�t$8H�4$H�L$@H�L$H�T$(H�T$H�D$0H�D$��\$ ��t,H�l$P�]L�D$XA�h@8�t +�D$hH��H��D$hH��H��D$hH��HÉ냉�i����%�,����%� +����%������%���� + + 0runtime.morestack_noctxt�$runtime.memequal32�$runtime.memequal32� runtime.eqstring@� "".autotmp_0562?type.string"".autotmp_0561type.string "".~r30type.bool"".s type.uintptr"".q,type.*"".TextFormatter"".p,type.*"".TextFormatter@������� �� ��G�� M�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�4type..hash.[4]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_0565type.int"".autotmp_0564type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[4]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�0type..eq.[4]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_0569?"type.interface {}"".autotmp_0568"type.interface {}"".autotmp_0567_type.int"".autotmp_0566Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[4]interface {}"".p*type.*[4]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�4type..hash.[3]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_0572type.int"".autotmp_0571type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[3]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�0type..eq.[3]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_0576?"type.interface {}"".autotmp_0575"type.interface {}"".autotmp_0574_type.int"".autotmp_0573Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[3]interface {}"".p*type.*[3]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go�$"".StdLogger.Fatal��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$0H�$H�\$(H�[ ��H�� � + 0runtime.morestack_noctxt� +P@""..anon0 &type.[]interface {}""..this"type."".StdLogger@L? +pp +`Tgclocals·32f137afc3f53351f1adc065fe3b9f83Tgclocals·3280bececceccd33cb74587feedb1f9f�&"".StdLogger.Fatalf��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�\$HH�\$H�\$PH�\$H�\$XH�\$H�\$`H�\$ H�\$hH�\$(H�\$@H�$H�\$8H�[(��H��0� + 0runtime.morestack_noctxt� +p`""..anon1@&type.[]interface {}""..anon0 type.string""..this"type."".StdLogger``_�� +t Tgclocals·a3682a93adc1ecf7106501ba903ce847Tgclocals·3280bececceccd33cb74587feedb1f9f�("".StdLogger.Fatalln��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$0H�$H�\$(H�[0��H�� � + 0runtime.morestack_noctxt� +P@""..anon0 &type.[]interface {}""..this"type."".StdLogger@L? +pp +`Tgclocals·32f137afc3f53351f1adc065fe3b9f83Tgclocals·3280bececceccd33cb74587feedb1f9f�$"".StdLogger.Panic��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$0H�$H�\$(H�[8��H�� � + 0runtime.morestack_noctxt� +P@""..anon0 &type.[]interface {}""..this"type."".StdLogger@L? +pp +`Tgclocals·32f137afc3f53351f1adc065fe3b9f83Tgclocals·3280bececceccd33cb74587feedb1f9f�&"".StdLogger.Panicf��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�\$HH�\$H�\$PH�\$H�\$XH�\$H�\$`H�\$ H�\$hH�\$(H�\$@H�$H�\$8H�[@��H��0� + 0runtime.morestack_noctxt� +p`""..anon1@&type.[]interface {}""..anon0 type.string""..this"type."".StdLogger``_�� +t Tgclocals·a3682a93adc1ecf7106501ba903ce847Tgclocals·3280bececceccd33cb74587feedb1f9f�("".StdLogger.Panicln��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$0H�$H�\$(H�[H��H�� � + 0runtime.morestack_noctxt� +P@""..anon0 &type.[]interface {}""..this"type."".StdLogger@L? +pp +`Tgclocals·32f137afc3f53351f1adc065fe3b9f83Tgclocals·3280bececceccd33cb74587feedb1f9f�$"".StdLogger.Print��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$0H�$H�\$(H�[P��H�� � + 0runtime.morestack_noctxt� +P@""..anon0 &type.[]interface {}""..this"type."".StdLogger@L? +pp +`Tgclocals·32f137afc3f53351f1adc065fe3b9f83Tgclocals·3280bececceccd33cb74587feedb1f9f�&"".StdLogger.Printf��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�\$HH�\$H�\$PH�\$H�\$XH�\$H�\$`H�\$ H�\$hH�\$(H�\$@H�$H�\$8H�[X��H��0� + 0runtime.morestack_noctxt� +p`""..anon1@&type.[]interface {}""..anon0 type.string""..this"type."".StdLogger``_�� +t Tgclocals·a3682a93adc1ecf7106501ba903ce847Tgclocals·3280bececceccd33cb74587feedb1f9f�("".StdLogger.Println��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�\$8H�\$H�\$@H�\$H�\$HH�\$H�\$0H�$H�\$(H�[`��H�� � + 0runtime.morestack_noctxt� +P@""..anon0 &type.[]interface {}""..this"type."".StdLogger@L? +p p +`Tgclocals·32f137afc3f53351f1adc065fe3b9f83Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·e475e3c2360b557d64285d9b9a4e506400 +�Tgclocals·a7c27d2bfcc924fa8a92b6b29b7218b100�Tgclocals·0719ac7e4405ec7094b2d696ead0af25((,.�Tgclocals·363b18caf0020ca418fd378dbb75c855((�"go.string.""0, "go.string.""�Tgclocals·44e348188e22fef6300f71ab26e45197 ��Tgclocals·896a3e2c9de7030cc72aa334f690557d  +�Tgclocals·396579fca70851935df9d21183ca82fd  +��Tgclocals·0723c8881b4d19cb48cb8887cfa073be  ���Tgclocals·cbbe1bd73f3c341fc477038dafd9ade4pp4�Z/�Z +�Tgclocals·fdf817463ca91d173b8e929c420286bd@@ + + + + + +�,4go.itab.*os.File.io.Writer�,>go.itab.*bytes.Buffer.io.Reader�Jgo.string."Failed to fire hook: %v\n"`RFailed to fire hook: %v + Jgo.string."Failed to fire hook: %v\n"�Rgo.string."Failed to obtain reader, %v\n"`ZFailed to obtain reader, %v + Rgo.string."Failed to obtain reader, %v\n"�Pgo.string."Failed to write to log, %v\n"`XFailed to write to log, %v + Pgo.string."Failed to write to log, %v\n"�Tgclocals·22d60cc41efa02a0ac67663f051098e8�� +*��<, +�< ,�Tgclocals·65a30d49934626502b3d799f3cf8d99e`` +&&&&&&&&&&�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·29f0050a5ee7c2b9348a75428171d7de �Tgclocals·9ff42bf311af152488d11f0f78c8d5ce  + +�Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6(( / �Tgclocals·e8d3240594e259421cd655d317fed5fe(( ����Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6(( / �Tgclocals·e8d3240594e259421cd655d317fed5fe(( ����Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·0a3395567ab7eee3bb936aced49af517 ��Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6(( / �Tgclocals·e8d3240594e259421cd655d317fed5fe(( ����Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·0a3395567ab7eee3bb936aced49af517 ��Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6(( / �Tgclocals·e8d3240594e259421cd655d317fed5fe(( ����Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6(( / �Tgclocals·e8d3240594e259421cd655d317fed5fe(( ����Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6(( / �Tgclocals·e8d3240594e259421cd655d317fed5fe(( ����Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6(( / �Tgclocals·b29a376724b9675f7c9e576a6dabc1e0(( + + +�Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6(( / �Tgclocals·b29a376724b9675f7c9e576a6dabc1e0(( + + +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6(( / �Tgclocals·b29a376724b9675f7c9e576a6dabc1e0(( + + +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6(( / �Tgclocals·b29a376724b9675f7c9e576a6dabc1e0(( + + +�Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6(( / �Tgclocals·b29a376724b9675f7c9e576a6dabc1e0(( + + +�Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6(( / �Tgclocals·b29a376724b9675f7c9e576a6dabc1e0(( + + +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·9f0d5ba6770c4a1ed4fa771547e96df1 +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a7a3692b8e27e823add69ec4239ba55f�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·20671cc48303dfd2b9d73bba3d1850b7 �Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·20671cc48303dfd2b9d73bba3d1850b7 �Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·5d05a78f811f5c3f62710534cdce0004�Tgclocals·0115f8d53b75c1696444f08ad03251d9�Tgclocals·7c868751a5d2fdd881613692c78d6476 �Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·20671cc48303dfd2b9d73bba3d1850b7 �Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·66ae2244d17a3b89653cba445a520071 +��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·06cab038d51064a089bda21fa03e00f7�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·f271231f400e778e0f59be25f7a26a56 +"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·f271231f400e778e0f59be25f7a26a56 +"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·f271231f400e778e0f59be25f7a26a56 +"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·f271231f400e778e0f59be25f7a26a56 +"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·f271231f400e778e0f59be25f7a26a56 +"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·f271231f400e778e0f59be25f7a26a56 +"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·f271231f400e778e0f59be25f7a26a56 +"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·f271231f400e778e0f59be25f7a26a56 +"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515� go.string."time"0*time go.string."time"�.go.string."fields.time"@8 fields.time .go.string."fields.time"�go.string."msg"0(msg go.string."msg"�,go.string."fields.msg"@6 +fields.msg ,go.string."fields.msg"�"go.string."level"0,level "go.string."level"�0go.string."fields.level"@: fields.level 0go.string."fields.level"�Tgclocals·cfe802ef097eb87dc1d2f379757036b4(( /�Tgclocals·15395a9df917b4c9aa74d5c6c7e1ebf4((�Tgclocals·5347b08d42ef15c0183233bde05091ab00  +�Tgclocals·a02efc190d1c7709e4c72531a85b968d00....�Tgclocals·7e4aab61b173caafc98b406c57151fa1 �Tgclocals·7ce35767da505d40dfb8f85871f02969  +&&�Jgo.string."2006-01-02T15:04:05Z07:00"`T2006-01-02T15:04:05Z07:00 Jgo.string."2006-01-02T15:04:05Z07:00"�`go.string."Failed to marshal fields to JSON, %v"pj$Failed to marshal fields to JSON, %v `go.string."Failed to marshal fields to JSON, %v"�Tgclocals·859165c97b106654e8e33715962a8293��T�Z<�Z�Z��Z��Z<�Z��<���Tgclocals·91781b467bdd49442cfecbf49067c104�� + + + + + + + + + + + + + +�,Lgo.itab.*"".TextFormatter."".Formatter�Tgclocals·87b2493d48b5216b7adc3dac672105cb@@ (* +�Tgclocals·e9c510091732f30fce387c9f1e977134@@�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·0a89f44bdb06c71b1e3fde611d9babf488 ������Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·ab01a2d55089ff50c402006df1039c3988 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·fc96ae191c2547955912928601e8595988 ������Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·fc96ae191c2547955912928601e8595988 ������Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·fc96ae191c2547955912928601e8595988 ������Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·fc96ae191c2547955912928601e8595988 ������Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·fc96ae191c2547955912928601e8595988 ������Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·fc96ae191c2547955912928601e8595988 ������Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·fc96ae191c2547955912928601e8595988 ������Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·fc96ae191c2547955912928601e8595988 ������Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·0442504e096b61648fffc20fe86cec6688"(�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�"go.string."debug"0,debug "go.string."debug"� go.string."info"0*info go.string."info"�&go.string."warning"00warning &go.string."warning"�"go.string."error"0,error "go.string."error"�"go.string."fatal"0,fatal "go.string."fatal"�"go.string."panic"0,panic "go.string."panic"�&go.string."unknown"00unknown &go.string."unknown"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a73fd2a0c6f832642aa9216fd9c5e6be� go.string."warn"0*warn go.string."warn"�Pgo.string."not a valid logrus Level: %q"`Znot a valid logrus Level: %q Pgo.string."not a valid logrus Level: %q"�Tgclocals·626b2db390378ab5b89c88b48426687f00 ���Tgclocals·db0f6b31ff49b3f025910ec03f9742fa00 +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a7a3692b8e27e823add69ec4239ba55f�Tgclocals·0528ab8f76149a707fd2f0025c2178a3�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a7a3692b8e27e823add69ec4239ba55f�Tgclocals·cfaa0e57cb7ad41983cddbe4e85a52ca�� +>�j�j �j * +�Tgclocals·71f03f9031d1479950f850ff028ecf79`` + + + + + + + + + + +�,>go.itab.*bytes.Buffer.io.Writer�Tgo.string."\x1b[%dm%s\x1b[0m[%04d] %-44s "`R[%dm%s[%04d] %-44s  Tgo.string."\x1b[%dm%s\x1b[0m[%04d] %-44s "�Pgo.string."\x1b[%dm%s\x1b[0m[%s] %-44s "PN[%dm%s[%s] %-44s  Pgo.string."\x1b[%dm%s\x1b[0m[%s] %-44s "�Bgo.string." \x1b[%dm%s\x1b[0m=%v"@@ [%dm%s=%v Bgo.string." \x1b[%dm%s\x1b[0m=%v"�Tgclocals·3d6acc133f31046c5a57fcb873eb249f�� R �(������ �(�����������*�����Tgclocals·40341f41d3f930f66258415e27eda61dpp �������������Tgclocals·8d600a433c6aaa81a4fe446d95c5546b �Tgclocals·d7e8a62d22b1cde6d92b17a55c33fe8f �go.string."%q"0&%q go.string."%q"�Tgclocals·d0bfb266a2b14e29993ffd436ee6f54e�� .�  +����� +���Tgclocals·81e712ea8ab1acd9cb646b715ca7f2cdpp ************�Tgclocals·61e2515c69061b8fed0e66ece719f936 �Tgclocals·2148c3737b2bb476685a1100a2e8343e �,@go.itab.*io.PipeReader.io.Reader�^go.string."Error while reading from Writer: %s"ph#Error while reading from Writer: %s ^go.string."Error while reading from Writer: %s"�Tgclocals·a266581f5aa078217f9a8ca87caf12e6�� <� � � �����Tgclocals·4a3831d274d2be9675c43f86862b9a60pp  + + + + + + + + + + + +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·519efd86263089ddb84df3cfe7fd2992�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·3280bececceccd33cb74587feedb1f9f�* "".stdtype.*"".Logger�* "".baseTimestamp0type.time.Time�,"".isTerminaltype.bool�,"".initdone·type.uint8�"".NewEntry·f"".NewEntry�$runtime.makemap·fruntime.makemap�(runtime.newobject·f"runtime.newobject�4runtime.writebarrierptr·f.runtime.writebarrierptr�,runtime.throwreturn·f&runtime.throwreturn�*"".(*Entry).Reader·f$"".(*Entry).Reader�8runtime.writebarrierslice·f2runtime.writebarrierslice�*"".(*Entry).String·f$"".(*Entry).String�*runtime.panicslice·f$runtime.panicslice�8runtime.slicebytetostring·f2runtime.slicebytetostring�0"".(*Entry).WithField·f*"".(*Entry).WithField�*runtime.mapassign1·f$runtime.mapassign1�2"".(*Entry).WithFields·f,"".(*Entry).WithFields�,runtime.mapiterinit·f&runtime.mapiterinit�,runtime.mapiternext·f&runtime.mapiternext�$"".(*Entry).log·f"".(*Entry).log�time.Now·ftime.Now�6runtime.writebarrierfat3·f0runtime.writebarrierfat3�:runtime.writebarrierstring·f4runtime.writebarrierstring�*"".LevelHooks.Fire·f$"".LevelHooks.Fire�*sync.(*Mutex).Lock·f$sync.(*Mutex).Lock�&runtime.typ2Itab·f runtime.typ2Itab�$runtime.convI2E·fruntime.convI2E�8runtime.writebarrieriface·f2runtime.writebarrieriface�fmt.Fprintf·ffmt.Fprintf�.sync.(*Mutex).Unlock·f(sync.(*Mutex).Unlock�(runtime.deferproc·f"runtime.deferproc�,runtime.deferreturn·f&runtime.deferreturn�io.Copy·fio.Copy�$runtime.gopanic·fruntime.gopanic�("".(*Entry).Debug·f""".(*Entry).Debug�fmt.Sprint·ffmt.Sprint�("".(*Entry).Print·f""".(*Entry).Print�&"".(*Entry).Info·f "".(*Entry).Info�&"".(*Entry).Warn·f "".(*Entry).Warn�,"".(*Entry).Warning·f&"".(*Entry).Warning�("".(*Entry).Error·f""".(*Entry).Error�("".(*Entry).Fatal·f""".(*Entry).Fatal�os.Exit·fos.Exit�("".(*Entry).Panic·f""".(*Entry).Panic�$runtime.convT2E·fruntime.convT2E�*"".(*Entry).Debugf·f$"".(*Entry).Debugf�fmt.Sprintf·ffmt.Sprintf�("".(*Entry).Infof·f""".(*Entry).Infof�*"".(*Entry).Printf·f$"".(*Entry).Printf�("".(*Entry).Warnf·f""".(*Entry).Warnf�."".(*Entry).Warningf·f("".(*Entry).Warningf�*"".(*Entry).Errorf·f$"".(*Entry).Errorf�*"".(*Entry).Fatalf·f$"".(*Entry).Fatalf�*"".(*Entry).Panicf·f$"".(*Entry).Panicf�,"".(*Entry).Debugln·f&"".(*Entry).Debugln�0"".(*Entry).sprintlnn·f*"".(*Entry).sprintlnn�*"".(*Entry).Infoln·f$"".(*Entry).Infoln�,"".(*Entry).Println·f&"".(*Entry).Println�*"".(*Entry).Warnln·f$"".(*Entry).Warnln�0"".(*Entry).Warningln·f*"".(*Entry).Warningln�,"".(*Entry).Errorln·f&"".(*Entry).Errorln�,"".(*Entry).Fatalln·f&"".(*Entry).Fatalln�,"".(*Entry).Panicln·f&"".(*Entry).Panicln�fmt.Sprintln·ffmt.Sprintln�("".StandardLogger·f""".StandardLogger�"".SetOutput·f"".SetOutput�$"".SetFormatter·f"".SetFormatter�"".SetLevel·f"".SetLevel�"".GetLevel·f"".GetLevel�"".AddHook·f"".AddHook�("".LevelHooks.Add·f""".LevelHooks.Add�"".WithField·f"".WithField�2"".(*Logger).WithField·f,"".(*Logger).WithField� "".WithFields·f"".WithFields�4"".(*Logger).WithFields·f."".(*Logger).WithFields�"".Debug·f"".Debug�*"".(*Logger).Debug·f$"".(*Logger).Debug�"".Print·f"".Print�*"".(*Logger).Print·f$"".(*Logger).Print�"".Info·f"".Info�("".(*Logger).Info·f""".(*Logger).Info�"".Warn·f"".Warn�("".(*Logger).Warn·f""".(*Logger).Warn�"".Warning·f"".Warning�."".(*Logger).Warning·f("".(*Logger).Warning�"".Error·f"".Error�*"".(*Logger).Error·f$"".(*Logger).Error�"".Panic·f"".Panic�*"".(*Logger).Panic·f$"".(*Logger).Panic�"".Fatal·f"".Fatal�*"".(*Logger).Fatal·f$"".(*Logger).Fatal�"".Debugf·f"".Debugf�,"".(*Logger).Debugf·f&"".(*Logger).Debugf�"".Printf·f"".Printf�,"".(*Logger).Printf·f&"".(*Logger).Printf�"".Infof·f"".Infof�*"".(*Logger).Infof·f$"".(*Logger).Infof�"".Warnf·f"".Warnf�*"".(*Logger).Warnf·f$"".(*Logger).Warnf�"".Warningf·f"".Warningf�0"".(*Logger).Warningf·f*"".(*Logger).Warningf�"".Errorf·f"".Errorf�,"".(*Logger).Errorf·f&"".(*Logger).Errorf�"".Panicf·f"".Panicf�,"".(*Logger).Panicf·f&"".(*Logger).Panicf�"".Fatalf·f"".Fatalf�,"".(*Logger).Fatalf·f&"".(*Logger).Fatalf�"".Debugln·f"".Debugln�."".(*Logger).Debugln·f("".(*Logger).Debugln�"".Println·f"".Println�."".(*Logger).Println·f("".(*Logger).Println�"".Infoln·f"".Infoln�,"".(*Logger).Infoln·f&"".(*Logger).Infoln�"".Warnln·f"".Warnln�,"".(*Logger).Warnln·f&"".(*Logger).Warnln�"".Warningln·f"".Warningln�2"".(*Logger).Warningln·f,"".(*Logger).Warningln�"".Errorln·f"".Errorln�."".(*Logger).Errorln·f("".(*Logger).Errorln�"".Panicln·f"".Panicln�."".(*Logger).Panicln·f("".(*Logger).Panicln�"".Fatalln·f"".Fatalln�."".(*Logger).Fatalln·f("".(*Logger).Fatalln�0"".prefixFieldClashes·f*"".prefixFieldClashes�:runtime.mapaccess2_faststr·f4runtime.mapaccess2_faststr�:runtime.mapaccess1_faststr·f4runtime.mapaccess1_faststr�*runtime.mapaccess1·f$runtime.mapaccess1�(runtime.growslice·f"runtime.growslice�:"".(*JSONFormatter).Format·f4"".(*JSONFormatter).Format�*runtime.efacethash·f$runtime.efacethash�*runtime.assertE2I2·f$runtime.assertE2I2�&time.Time.Format·f time.Time.Format�$"".Level.String·f"".Level.String�0encoding/json.Marshal·f*encoding/json.Marshal�fmt.Errorf·ffmt.Errorf�"".New·f "".New� "".ParseLevel·f"".ParseLevel�(runtime.cmpstring·f"runtime.cmpstring�&runtime.eqstring·f runtime.eqstring� "".IsTerminal·f"".IsTerminal�&syscall.Syscall6·f syscall.Syscall6�"".init·1·f"".init·1�"".miniTS·f"".miniTS�time.Since·ftime.Since�:"".(*TextFormatter).Format·f4"".(*TextFormatter).Format�(runtime.makeslice·f"runtime.makeslice�sort.Strings·fsort.Strings�F"".(*TextFormatter).printColored·f@"".(*TextFormatter).printColored�J"".(*TextFormatter).appendKeyValue·fD"".(*TextFormatter).appendKeyValue�8bytes.(*Buffer).WriteByte·f2bytes.(*Buffer).WriteByte�$strings.ToUpper·fstrings.ToUpper�$"".needsQuoting·f"".needsQuoting�,runtime.stringiter2·f&runtime.stringiter2� &type..alg.[8]string0bruntime.gcbits.0x48484848484848480000000000000000P*go.string."[8]string"p.go.weak.type.*[8]string�"runtime.zerovalue�type.string�type.[]string�>go.typelink.[8]string/[8]stringtype.[8]string�4go.string."[]interface {}"@>[]interface {} 4go.string."[]interface {}"�&type.[]interface {}��p��/ � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P4go.string."[]interface {}"p8go.weak.type.*[]interface {}�"runtime.zerovalue�"type.interface {}�Rgo.typelink.[]interface {}/[]interface {}&type.[]interface {}�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�*logrus.Fields 4go.string."*logrus.Fields"�type.*"".Fields���Xr�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."*logrus.Fields"p0go.weak.type.**"".Fields�"runtime.zerovalue�type."".Fields�2go.string."logrus.Fields"@< logrus.Fields 2go.string."logrus.Fields"�$go.string."Fields"0.Fields $go.string."Fields"�type."".Fields�����i5 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P2go.string."logrus.Fields"ptype.*"".Fields�"runtime.zerovalue�type.string�"type.interface {}�Ftype.map.bucket[string]interface {}�@type.map.hdr[string]interface {}`�type."".Fields�$go.string."Fields"�"go.importpath."".��type."".Fields�2go.string."*logrus.Level"@< *logrus.Level 2go.string."*logrus.Level"�$go.string."logrus"0.logrus $go.string."logrus"�"go.string."Level"0,Level "go.string."Level"�$go.string."String"0.String $go.string."String"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Lgo.string."func(*logrus.Level) string"`Vfunc(*logrus.Level) string Lgo.string."func(*logrus.Level) string"�6type.func(*"".Level) string����x�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PLgo.string."func(*logrus.Level) string"pHgo.weak.type.*func(*"".Level) string�"runtime.zerovalue��6type.func(*"".Level) string��6type.func(*"".Level) string�type.*"".Level�type.string�2go.string."func() string"@< func() string 2go.string."func() string"�$type.func() string���m�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P2go.string."func() string"p6go.weak.type.*func() string�"runtime.zerovalue��$type.func() string��$type.func() string�type.string�type.*"".Level���l�U6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P2go.string."*logrus.Level"p.go.weak.type.**"".Level�"runtime.zerovalue�type."".Level`�type.*"".Level��type.*"".Level�$go.string."String"�$type.func() string�6type.func(*"".Level) string�$"".(*Level).String�$"".(*Level).String�^runtime.gcbits.0x000000000000000000000000000000 �0go.string."logrus.Level"@: logrus.Level 0go.string."logrus.Level"�Jgo.string."func(logrus.Level) string"`Tfunc(logrus.Level) string Jgo.string."func(logrus.Level) string"�4type.func("".Level) string��N��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PJgo.string."func(logrus.Level) string"pFgo.weak.type.*func("".Level) string�"runtime.zerovalue��4type.func("".Level) string��4type.func("".Level) string�type."".Level�type.string�type."".Level���I � @ runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P0go.string."logrus.Level"ptype.*"".Level�"runtime.zerovalue`�type."".Level�"go.string."Level"�"go.importpath."".��type."".Level�$go.string."String"�$type.func() string�4type.func("".Level) string�$"".(*Level).String�"".Level.String�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·32f137afc3f53351f1adc065fe3b9f83 ++�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·9edc1f6d8fc7336ae101b48cbf822a45 + �2go.string."*logrus.Entry"@< *logrus.Entry 2go.string."*logrus.Entry"�`go.string."func(*logrus.Entry, ...interface {})"pj$func(*logrus.Entry, ...interface {}) `go.string."func(*logrus.Entry, ...interface {})"�Jtype.func(*"".Entry, ...interface {})���]�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P`go.string."func(*logrus.Entry, ...interface {})"p\go.weak.type.*func(*"".Entry, ...interface {})�"runtime.zerovalue��Jtype.func(*"".Entry, ...interface {})��Jtype.func(*"".Entry, ...interface {})�type.*"".Entry�&type.[]interface {}�pgo.string."func(*logrus.Entry, string, ...interface {})"�z,func(*logrus.Entry, string, ...interface {}) pgo.string."func(*logrus.Entry, string, ...interface {})"�Ztype.func(*"".Entry, string, ...interface {})����{�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ppgo.string."func(*logrus.Entry, string, ...interface {})"plgo.weak.type.*func(*"".Entry, string, ...interface {})�"runtime.zerovalue��Ztype.func(*"".Entry, string, ...interface {})��Ztype.func(*"".Entry, string, ...interface {})�type.*"".Entry�type.string�&type.[]interface {}�lgo.string."func(*logrus.Entry) (*bytes.Buffer, error)"�v*func(*logrus.Entry) (*bytes.Buffer, error) lgo.string."func(*logrus.Entry) (*bytes.Buffer, error)"�Vtype.func(*"".Entry) (*bytes.Buffer, error)���j]�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Plgo.string."func(*logrus.Entry) (*bytes.Buffer, error)"phgo.weak.type.*func(*"".Entry) (*bytes.Buffer, error)�"runtime.zerovalue��Vtype.func(*"".Entry) (*bytes.Buffer, error)��Vtype.func(*"".Entry) (*bytes.Buffer, error)�type.*"".Entry�$type.*bytes.Buffer�type.error�^go.string."func(*logrus.Entry) (string, error)"ph#func(*logrus.Entry) (string, error) ^go.string."func(*logrus.Entry) (string, error)"�Htype.func(*"".Entry) (string, error)����}3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P^go.string."func(*logrus.Entry) (string, error)"pZgo.weak.type.*func(*"".Entry) (string, error)�"runtime.zerovalue��Htype.func(*"".Entry) (string, error)��Htype.func(*"".Entry) (string, error)�type.*"".Entry�type.string�type.error��go.string."func(*logrus.Entry, string, interface {}) *logrus.Entry"��7func(*logrus.Entry, string, interface {}) *logrus.Entry �go.string."func(*logrus.Entry, string, interface {}) *logrus.Entry"�htype.func(*"".Entry, string, interface {}) *"".Entry���4�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*logrus.Entry, string, interface {}) *logrus.Entry"pzgo.weak.type.*func(*"".Entry, string, interface {}) *"".Entry�"runtime.zerovalue��htype.func(*"".Entry, string, interface {}) *"".Entry��htype.func(*"".Entry, string, interface {}) *"".Entry�type.*"".Entry�type.string�"type.interface {}�type.*"".Entry�xgo.string."func(*logrus.Entry, logrus.Fields) *logrus.Entry"��0func(*logrus.Entry, logrus.Fields) *logrus.Entry xgo.string."func(*logrus.Entry, logrus.Fields) *logrus.Entry"�Rtype.func(*"".Entry, "".Fields) *"".Entry��O\�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pxgo.string."func(*logrus.Entry, logrus.Fields) *logrus.Entry"pdgo.weak.type.*func(*"".Entry, "".Fields) *"".Entry�"runtime.zerovalue��Rtype.func(*"".Entry, "".Fields) *"".Entry��Rtype.func(*"".Entry, "".Fields) *"".Entry�type.*"".Entry�type."".Fields�type.*"".Entry�jgo.string."func(*logrus.Entry, logrus.Level, string)"�t)func(*logrus.Entry, logrus.Level, string) jgo.string."func(*logrus.Entry, logrus.Level, string)"�Ltype.func(*"".Entry, "".Level, string)������3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pjgo.string."func(*logrus.Entry, logrus.Level, string)"p^go.weak.type.*func(*"".Entry, "".Level, string)�"runtime.zerovalue��Ltype.func(*"".Entry, "".Level, string)��Ltype.func(*"".Entry, "".Level, string)�type.*"".Entry�type."".Level�type.string�ngo.string."func(*logrus.Entry, ...interface {}) string"�x+func(*logrus.Entry, ...interface {}) string ngo.string."func(*logrus.Entry, ...interface {}) string"�Xtype.func(*"".Entry, ...interface {}) string������3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pngo.string."func(*logrus.Entry, ...interface {}) string"pjgo.weak.type.*func(*"".Entry, ...interface {}) string�"runtime.zerovalue��Xtype.func(*"".Entry, ...interface {}) string��Xtype.func(*"".Entry, ...interface {}) string�type.*"".Entry�&type.[]interface {}�type.string�"go.string."Debug"0,Debug "go.string."Debug"�Bgo.string."func(...interface {})"PLfunc(...interface {}) Bgo.string."func(...interface {})"�4type.func(...interface {})���3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."func(...interface {})"pFgo.weak.type.*func(...interface {})�"runtime.zerovalue��4type.func(...interface {})��4type.func(...interface {})�&type.[]interface {}�$go.string."Debugf"0.Debugf $go.string."Debugf"�Rgo.string."func(string, ...interface {})"`\func(string, ...interface {}) Rgo.string."func(string, ...interface {})"�Dtype.func(string, ...interface {})����@�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PRgo.string."func(string, ...interface {})"pVgo.weak.type.*func(string, ...interface {})�"runtime.zerovalue��Dtype.func(string, ...interface {})��Dtype.func(string, ...interface {})�type.string�&type.[]interface {}�&go.string."Debugln"00Debugln &go.string."Debugln"�"go.string."Error"0,Error "go.string."Error"�$go.string."Errorf"0.Errorf $go.string."Errorf"�&go.string."Errorln"00Errorln &go.string."Errorln"�"go.string."Fatal"0,Fatal "go.string."Fatal"�$go.string."Fatalf"0.Fatalf $go.string."Fatalf"�&go.string."Fatalln"00Fatalln &go.string."Fatalln"� go.string."Info"0*Info go.string."Info"�"go.string."Infof"0,Infof "go.string."Infof"�$go.string."Infoln"0.Infoln $go.string."Infoln"�"go.string."Panic"0,Panic "go.string."Panic"�$go.string."Panicf"0.Panicf $go.string."Panicf"�&go.string."Panicln"00Panicln &go.string."Panicln"�"go.string."Print"0,Print "go.string."Print"�$go.string."Printf"0.Printf $go.string."Printf"�&go.string."Println"00Println &go.string."Println"�$go.string."Reader"0.Reader $go.string."Reader"�Rgo.string."func() (*bytes.Buffer, error)"`\func() (*bytes.Buffer, error) Rgo.string."func() (*bytes.Buffer, error)"�Dtype.func() (*bytes.Buffer, error)����(3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PRgo.string."func() (*bytes.Buffer, error)"pVgo.weak.type.*func() (*bytes.Buffer, error)�"runtime.zerovalue��Dtype.func() (*bytes.Buffer, error)��Dtype.func() (*bytes.Buffer, error)�$type.*bytes.Buffer�type.error�Dgo.string."func() (string, error)"PNfunc() (string, error) Dgo.string."func() (string, error)"�6type.func() (string, error)����u�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PDgo.string."func() (string, error)"pHgo.weak.type.*func() (string, error)�"runtime.zerovalue��6type.func() (string, error)��6type.func() (string, error)�type.string�type.error� go.string."Warn"0*Warn go.string."Warn"�"go.string."Warnf"0,Warnf "go.string."Warnf"�&go.string."Warning"00Warning &go.string."Warning"�(go.string."Warningf"@2Warningf (go.string."Warningf"�*go.string."Warningln"@4 Warningln *go.string."Warningln"�$go.string."Warnln"0.Warnln $go.string."Warnln"�*go.string."WithField"@4 WithField *go.string."WithField"�hgo.string."func(string, interface {}) *logrus.Entry"�r(func(string, interface {}) *logrus.Entry hgo.string."func(string, interface {}) *logrus.Entry"�Rtype.func(string, interface {}) *"".Entry���À%3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Phgo.string."func(string, interface {}) *logrus.Entry"pdgo.weak.type.*func(string, interface {}) *"".Entry�"runtime.zerovalue��Rtype.func(string, interface {}) *"".Entry��Rtype.func(string, interface {}) *"".Entry�type.string�"type.interface {}�type.*"".Entry�,go.string."WithFields"@6 +WithFields ,go.string."WithFields"�Zgo.string."func(logrus.Fields) *logrus.Entry"pd!func(logrus.Fields) *logrus.Entry Zgo.string."func(logrus.Fields) *logrus.Entry"�[]logrus.Level 4go.string."[]logrus.Level"�type.[]"".Level��Ec�� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P4go.string."[]logrus.Level"p0go.weak.type.*[]"".Level�"runtime.zerovalue�type."".Level�Jgo.typelink.[]logrus.Level/[]"".Leveltype.[]"".Level�Bgo.string."func() []logrus.Level"PLfunc() []logrus.Level Bgo.string."func() []logrus.Level"�,type.func() []"".Level��l%��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."func() []logrus.Level"p>go.weak.type.*func() []"".Level�"runtime.zerovalue��,type.func() []"".Level��,type.func() []"".Level�type.[]"".Level�0go.string."*logrus.Hook"@: *logrus.Hook 0go.string."*logrus.Hook"�type.*"".Hook�� ��>6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P0go.string."*logrus.Hook"p,go.weak.type.**"".Hook�"runtime.zerovalue�type."".Hook�bruntime.gcbits.0x8c000000000000000000000000000000 ��.go.string."logrus.Hook"@8 logrus.Hook .go.string."logrus.Hook"� go.string."Fire"0*Fire go.string."Fire"�$go.string."Levels"0.Levels $go.string."Levels"� go.string."Hook"0*Hook go.string."Hook"�type."".Hook��ۍ�T � runtime.algarray0bruntime.gcbits.0x8c000000000000000000000000000000P.go.string."logrus.Hook"ptype.*"".Hook�"runtime.zerovalue��type."".Hook� go.string."Fire"�4type.func(*"".Entry) error�$go.string."Levels"�,type.func() []"".Level`�type."".Hook� go.string."Hook"�"go.importpath."".��type."".Hook�2go.string."[]logrus.Hook"@< []logrus.Hook 2go.string."[]logrus.Hook"�type.[]"".Hook���φ$ � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P2go.string."[]logrus.Hook"p.go.weak.type.*[]"".Hook�"runtime.zerovalue�type."".Hook�Fgo.typelink.[]logrus.Hook/[]"".Hooktype.[]"".Hook�6go.string."[8]logrus.Level"@@[8]logrus.Level 6go.string."[8]logrus.Level"� type.[8]"".Level���-�.� � runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P6go.string."[8]logrus.Level"p2go.weak.type.*[8]"".Level�"runtime.zerovalue�type."".Level�type.[]"".Level�Ngo.typelink.[8]logrus.Level/[8]"".Level type.[8]"".Level�6go.string."[][]logrus.Hook"@@[][]logrus.Hook 6go.string."[][]logrus.Hook"� type.[][]"".Hook��׀%� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P6go.string."[][]logrus.Hook"p2go.weak.type.*[][]"".Hook�"runtime.zerovalue�type.[]"".Hook�Ngo.typelink.[][]logrus.Hook/[][]"".Hook type.[][]"".Hook�bruntime.gcbits.0x48844448844448844448844400000000 H�DH�DH�DH�D�8go.string."[8][]logrus.Hook"PB[8][]logrus.Hook 8go.string."[8][]logrus.Hook"�"type.[8][]"".Hook���$�� � runtime.algarray0bruntime.gcbits.0x48844448844448844448844400000000P8go.string."[8][]logrus.Hook"p4go.weak.type.*[8][]"".Hook�"runtime.zerovalue�type.[]"".Hook� type.[][]"".Hook�Rgo.typelink.[8][]logrus.Hook/[8][]"".Hook"type.[8][]"".Hook�dgo.string."*map.bucket[logrus.Level][]logrus.Hook"pn&*map.bucket[logrus.Level][]logrus.Hook dgo.string."*map.bucket[logrus.Level][]logrus.Hook"�Ftype.*map.bucket["".Level][]"".Hook���Ҿ%6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pdgo.string."*map.bucket[logrus.Level][]logrus.Hook"pXgo.weak.type.**map.bucket["".Level][]"".Hook�"runtime.zerovalue�Dtype.map.bucket["".Level][]"".Hook�,Ltype..gc.map.bucket["".Level][]"".Hook8�Ttype..gcprog.map.bucket["".Level][]"".HookeY�eY�%�bgo.string."map.bucket[logrus.Level][]logrus.Hook"pl%map.bucket[logrus.Level][]logrus.Hook bgo.string."map.bucket[logrus.Level][]logrus.Hook"�Dtype.map.bucket["".Level][]"".Hook�����"Y� � runtime.algarray0Ltype..gc.map.bucket["".Level][]"".Hook@Ttype..gcprog.map.bucket["".Level][]"".HookPbgo.string."map.bucket[logrus.Level][]logrus.Hook"pVgo.weak.type.*map.bucket["".Level][]"".Hook�"runtime.zerovalue��Dtype.map.bucket["".Level][]"".Hook� go.string."keys"� type.[8]"".Level�$go.string."values"�"type.[8][]"".Hook�(go.string."overflow"�Ftype.*map.bucket["".Level][]"".Hook�\go.string."map.hdr[logrus.Level][]logrus.Hook"pf"map.hdr[logrus.Level][]logrus.Hook \go.string."map.hdr[logrus.Level][]logrus.Hook"�>type.map.hdr["".Level][]"".Hook��0�-/�  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000P\go.string."map.hdr[logrus.Level][]logrus.Hook"pPgo.weak.type.*map.hdr["".Level][]"".Hook�"runtime.zerovalue��>type.map.hdr["".Level][]"".Hook�&go.string."buckets"�Ftype.*map.bucket["".Level][]"".Hook�,go.string."oldbuckets"�Ftype.*map.bucket["".Level][]"".Hook�type.map.hdr["".Level][]"".Hook`�$type."".LevelHooks�,go.string."LevelHooks"�"go.importpath."".��$type."".LevelHooks�go.string."Add"�$type.func("".Hook)�Btype.func("".LevelHooks, "".Hook)�""".LevelHooks.Add�""".LevelHooks.Add� go.string."Fire"�Htype.func("".Level, *"".Entry) error�ftype.func("".LevelHooks, "".Level, *"".Entry) error�$"".LevelHooks.Fire�$"".LevelHooks.Fire�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·334cb8bc6294eb0b97ffb9b2c8e3805f+�&go.string."[]uint8"00[]uint8 &go.string."[]uint8"�type.[]uint8���~.8 � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P&go.string."[]uint8"p*go.weak.type.*[]uint8�"runtime.zerovalue�type.uint8�6go.typelink.[]uint8/[]uint8type.[]uint8�`go.string."func(*logrus.Entry) ([]uint8, error)"pj$func(*logrus.Entry) ([]uint8, error) `go.string."func(*logrus.Entry) ([]uint8, error)"�Jtype.func(*"".Entry) ([]uint8, error)��O"%�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P`go.string."func(*logrus.Entry) ([]uint8, error)"p\go.weak.type.*func(*"".Entry) ([]uint8, error)�"runtime.zerovalue��Jtype.func(*"".Entry) ([]uint8, error)��Jtype.func(*"".Entry) ([]uint8, error)�type.*"".Entry�type.[]uint8�type.error�:go.string."*logrus.Formatter"PD*logrus.Formatter :go.string."*logrus.Formatter"�$type.*"".Formatter��:0x6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."*logrus.Formatter"p6go.weak.type.**"".Formatter�"runtime.zerovalue�"type."".Formatter�8go.string."logrus.Formatter"PBlogrus.Formatter 8go.string."logrus.Formatter"�$go.string."Format"0.Format $go.string."Format"�*go.string."Formatter"@4 Formatter *go.string."Formatter"�"type."".Formatter��־�� � runtime.algarray0bruntime.gcbits.0x8c000000000000000000000000000000P8go.string."logrus.Formatter"p$type.*"".Formatter�"runtime.zerovalue��"type."".Formatter�$go.string."Format"�Jtype.func(*"".Entry) ([]uint8, error)`�"type."".Formatter�*go.string."Formatter"�"go.importpath."".��"type."".Formatter�bruntime.gcbits.0x8cc848c4888c44000000000000000000 ��HĈ�D�2go.string."logrus.Logger"@< logrus.Logger 2go.string."logrus.Logger"�go.string."Out"0(Out go.string."Out"�"go.string."Hooks"0,Hooks "go.string."Hooks"�go.string."mu"0&mu go.string."mu"�$go.string."Logger"0.Logger $go.string."Logger"�type."".Logger��8D�k(,* � runtime.algarray0bruntime.gcbits.0x8cc848c4888c44000000000000000000P2go.string."logrus.Logger"ptype.*"".Logger�"runtime.zerovalue��type."".Logger�go.string."Out"�type.io.Writer�"go.string."Hooks"�$type."".LevelHooks�*go.string."Formatter"�"type."".Formatter�"go.string."Level"�type."".Level�go.string."mu"�"go.importpath."".�type.sync.Mutex`�type."".Logger�$go.string."Logger"�"go.importpath."".��type."".Logger�4go.string."*logrus.Logger"@>*logrus.Logger 4go.string."*logrus.Logger"�bgo.string."func(*logrus.Logger, ...interface {})"pl%func(*logrus.Logger, ...interface {}) bgo.string."func(*logrus.Logger, ...interface {})"�Ltype.func(*"".Logger, ...interface {})��[R��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pbgo.string."func(*logrus.Logger, ...interface {})"p^go.weak.type.*func(*"".Logger, ...interface {})�"runtime.zerovalue��Ltype.func(*"".Logger, ...interface {})��Ltype.func(*"".Logger, ...interface {})�type.*"".Logger�&type.[]interface {}�rgo.string."func(*logrus.Logger, string, ...interface {})"�|-func(*logrus.Logger, string, ...interface {}) rgo.string."func(*logrus.Logger, string, ...interface {})"�\type.func(*"".Logger, string, ...interface {})��y���3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Prgo.string."func(*logrus.Logger, string, ...interface {})"pngo.weak.type.*func(*"".Logger, string, ...interface {})�"runtime.zerovalue��\type.func(*"".Logger, string, ...interface {})��\type.func(*"".Logger, string, ...interface {})�type.*"".Logger�type.string�&type.[]interface {}��go.string."func(*logrus.Logger, string, interface {}) *logrus.Entry"��8func(*logrus.Logger, string, interface {}) *logrus.Entry �go.string."func(*logrus.Logger, string, interface {}) *logrus.Entry"�jtype.func(*"".Logger, string, interface {}) *"".Entry�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*logrus.Logger, string, interface {}) *logrus.Entry"p|go.weak.type.*func(*"".Logger, string, interface {}) *"".Entry�"runtime.zerovalue��jtype.func(*"".Logger, string, interface {}) *"".Entry��jtype.func(*"".Logger, string, interface {}) *"".Entry�type.*"".Logger�type.string�"type.interface {}�type.*"".Entry�zgo.string."func(*logrus.Logger, logrus.Fields) *logrus.Entry"��1func(*logrus.Logger, logrus.Fields) *logrus.Entry zgo.string."func(*logrus.Logger, logrus.Fields) *logrus.Entry"�Ttype.func(*"".Logger, "".Fields) *"".Entry���&��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pzgo.string."func(*logrus.Logger, logrus.Fields) *logrus.Entry"pfgo.weak.type.*func(*"".Logger, "".Fields) *"".Entry�"runtime.zerovalue��Ttype.func(*"".Logger, "".Fields) *"".Entry��Ttype.func(*"".Logger, "".Fields) *"".Entry�type.*"".Logger�type."".Fields�type.*"".Entry�^go.string."func(*logrus.Logger) *io.PipeWriter"ph#func(*logrus.Logger) *io.PipeWriter ^go.string."func(*logrus.Logger) *io.PipeWriter"�Htype.func(*"".Logger) *io.PipeWriter���v&O3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P^go.string."func(*logrus.Logger) *io.PipeWriter"pZgo.weak.type.*func(*"".Logger) *io.PipeWriter�"runtime.zerovalue��Htype.func(*"".Logger) *io.PipeWriter��Htype.func(*"".Logger) *io.PipeWriter�type.*"".Logger�&type.*io.PipeWriter�`go.string."func(*logrus.Logger, *io.PipeReader)"pj$func(*logrus.Logger, *io.PipeReader) `go.string."func(*logrus.Logger, *io.PipeReader)"�Jtype.func(*"".Logger, *io.PipeReader)��X2��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P`go.string."func(*logrus.Logger, *io.PipeReader)"p\go.weak.type.*func(*"".Logger, *io.PipeReader)�"runtime.zerovalue��Jtype.func(*"".Logger, *io.PipeReader)��Jtype.func(*"".Logger, *io.PipeReader)�type.*"".Logger�&type.*io.PipeReader�$go.string."Writer"0.Writer $go.string."Writer"�Bgo.string."func() *io.PipeWriter"PLfunc() *io.PipeWriter Bgo.string."func() *io.PipeWriter"�4type.func() *io.PipeWriter���Y�h3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."func() *io.PipeWriter"pFgo.weak.type.*func() *io.PipeWriter�"runtime.zerovalue��4type.func() *io.PipeWriter��4type.func() *io.PipeWriter�&type.*io.PipeWriter�2go.string."writerScanner"@< writerScanner 2go.string."writerScanner"�@go.string."func(*io.PipeReader)"PJfunc(*io.PipeReader) @go.string."func(*io.PipeReader)"�2type.func(*io.PipeReader)����83 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P@go.string."func(*io.PipeReader)"pDgo.weak.type.*func(*io.PipeReader)�"runtime.zerovalue��2type.func(*io.PipeReader)��2type.func(*io.PipeReader)�&type.*io.PipeReader�type.*"".Logger��9�8�6� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."*logrus.Logger"p0go.weak.type.**"".Logger�"runtime.zerovalue�type."".Logger`�type.*"".Logger��type.*"".Logger�"go.string."Debug"�4type.func(...interface {})�Ltype.func(*"".Logger, ...interface {})�$"".(*Logger).Debug�$"".(*Logger).Debug�$go.string."Debugf"�Dtype.func(string, ...interface {})�\type.func(*"".Logger, string, ...interface {})�&"".(*Logger).Debugf�&"".(*Logger).Debugf�&go.string."Debugln"�4type.func(...interface {})�Ltype.func(*"".Logger, ...interface {})�("".(*Logger).Debugln�("".(*Logger).Debugln�"go.string."Error"�4type.func(...interface {})�Ltype.func(*"".Logger, ...interface {})�$"".(*Logger).Error�$"".(*Logger).Error�$go.string."Errorf"�Dtype.func(string, ...interface {})�\type.func(*"".Logger, string, ...interface {})�&"".(*Logger).Errorf�&"".(*Logger).Errorf�&go.string."Errorln"�4type.func(...interface {})�Ltype.func(*"".Logger, ...interface {})�("".(*Logger).Errorln�("".(*Logger).Errorln�"go.string."Fatal"�4type.func(...interface {})�Ltype.func(*"".Logger, ...interface {})�$"".(*Logger).Fatal�$"".(*Logger).Fatal�$go.string."Fatalf"�Dtype.func(string, ...interface {})�\type.func(*"".Logger, string, ...interface {})�&"".(*Logger).Fatalf�&"".(*Logger).Fatalf�&go.string."Fatalln"�4type.func(...interface {})�Ltype.func(*"".Logger, ...interface {})�("".(*Logger).Fatalln�("".(*Logger).Fatalln� go.string."Info"�4type.func(...interface {})� Ltype.func(*"".Logger, ...interface {})� """.(*Logger).Info� """.(*Logger).Info� "go.string."Infof"� Dtype.func(string, ...interface {})� \type.func(*"".Logger, string, ...interface {})� $"".(*Logger).Infof� +$"".(*Logger).Infof� +$go.string."Infoln"� +4type.func(...interface {})� +Ltype.func(*"".Logger, ...interface {})� +&"".(*Logger).Infoln� +&"".(*Logger).Infoln� +"go.string."Panic"� 4type.func(...interface {})� Ltype.func(*"".Logger, ...interface {})� $"".(*Logger).Panic� $"".(*Logger).Panic� $go.string."Panicf"� Dtype.func(string, ...interface {})� \type.func(*"".Logger, string, ...interface {})� &"".(*Logger).Panicf� &"".(*Logger).Panicf� &go.string."Panicln"� 4type.func(...interface {})� Ltype.func(*"".Logger, ...interface {})� ("".(*Logger).Panicln� ("".(*Logger).Panicln� "go.string."Print"� 4type.func(...interface {})� Ltype.func(*"".Logger, ...interface {})� $"".(*Logger).Print� $"".(*Logger).Print� $go.string."Printf"�Dtype.func(string, ...interface {})�\type.func(*"".Logger, string, ...interface {})�&"".(*Logger).Printf�&"".(*Logger).Printf�&go.string."Println"�4type.func(...interface {})�Ltype.func(*"".Logger, ...interface {})�("".(*Logger).Println�("".(*Logger).Println� go.string."Warn"�4type.func(...interface {})�Ltype.func(*"".Logger, ...interface {})�""".(*Logger).Warn�""".(*Logger).Warn�"go.string."Warnf"�Dtype.func(string, ...interface {})�\type.func(*"".Logger, string, ...interface {})�$"".(*Logger).Warnf�$"".(*Logger).Warnf�&go.string."Warning"�4type.func(...interface {})�Ltype.func(*"".Logger, ...interface {})�("".(*Logger).Warning�("".(*Logger).Warning�(go.string."Warningf"�Dtype.func(string, ...interface {})�\type.func(*"".Logger, string, ...interface {})�*"".(*Logger).Warningf�*"".(*Logger).Warningf�*go.string."Warningln"�4type.func(...interface {})�Ltype.func(*"".Logger, ...interface {})�,"".(*Logger).Warningln�,"".(*Logger).Warningln�$go.string."Warnln"�4type.func(...interface {})�Ltype.func(*"".Logger, ...interface {})�&"".(*Logger).Warnln�&"".(*Logger).Warnln�*go.string."WithField"�Rtype.func(string, interface {}) *"".Entry�jtype.func(*"".Logger, string, interface {}) *"".Entry�,"".(*Logger).WithField�,"".(*Logger).WithField�,go.string."WithFields"�go.typelink.[]uintptr/[]uintptrtype.[]uintptr�,go.string."[4]uintptr"@6 +[4]uintptr ,go.string."[4]uintptr"�type.[4]uintptr�� l<�  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P,go.string."[4]uintptr"p0go.weak.type.*[4]uintptr�"runtime.zerovalue�type.uintptr�type.[]uintptr�Bgo.typelink.[4]uintptr/[4]uintptrtype.[4]uintptr�bruntime.gcbits.0x88888844440000000000000000000000 ���DD�Pgo.string."map.iter[string]interface {}"`Zmap.iter[string]interface {} Pgo.string."map.iter[string]interface {}"�go.string."key"0(key go.string."key"�go.string."val"0(val go.string."val"�go.string."t"0$t go.string."t"�go.string."h"0$h go.string."h"� go.string."bptr"0*bptr go.string."bptr"�"go.string."other"0,other "go.string."other"�Btype.map.iter[string]interface {}��Pm8� (0( � runtime.algarray0bruntime.gcbits.0x88888844440000000000000000000000PPgo.string."map.iter[string]interface {}"pTgo.weak.type.*map.iter[string]interface {}�"runtime.zerovalue��Btype.map.iter[string]interface {}�go.string."key"�type.*string�go.string."val"�$type.*interface {}�go.string."t"�type.*uint8�go.string."h"�Btype.*map.hdr[string]interface {}�&go.string."buckets"�Htype.*map.bucket[string]interface {}� go.string."bptr"�Htype.*map.bucket[string]interface {}�"go.string."other"�type.[4]uintptr�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�go.weak.type.**"".JSONFormatter�"runtime.zerovalue�*type."".JSONFormatter`�,type.*"".JSONFormatter��,type.*"".JSONFormatter�$go.string."Format"�Jtype.func(*"".Entry) ([]uint8, error)�ptype.func(*"".JSONFormatter, *"".Entry) ([]uint8, error)�4"".(*JSONFormatter).Format�4"".(*JSONFormatter).Format�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�>type..hashfunc."".TextFormatter6type..hash."".TextFormatter�:type..eqfunc."".TextFormatter2type..eq."".TextFormatter�4type..alg."".TextFormatter >type..hashfunc."".TextFormatter:type..eqfunc."".TextFormatter�Bgo.string."*logrus.TextFormatter"PL*logrus.TextFormatter Bgo.string."*logrus.TextFormatter"��go.string."func(*logrus.TextFormatter, *logrus.Entry) ([]uint8, error)"��;func(*logrus.TextFormatter, *logrus.Entry) ([]uint8, error) �go.string."func(*logrus.TextFormatter, *logrus.Entry) ([]uint8, error)"�ptype.func(*"".TextFormatter, *"".Entry) ([]uint8, error)����D�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*logrus.TextFormatter, *logrus.Entry) ([]uint8, error)"p�go.weak.type.*func(*"".TextFormatter, *"".Entry) ([]uint8, error)�"runtime.zerovalue��ptype.func(*"".TextFormatter, *"".Entry) ([]uint8, error)��ptype.func(*"".TextFormatter, *"".Entry) ([]uint8, error)�,type.*"".TextFormatter�type.*"".Entry�type.[]uint8�type.error��go.string."func(*logrus.TextFormatter, *bytes.Buffer, string, interface {})"��@func(*logrus.TextFormatter, *bytes.Buffer, string, interface {}) �go.string."func(*logrus.TextFormatter, *bytes.Buffer, string, interface {})"��type.func(*"".TextFormatter, *bytes.Buffer, string, interface {})��j��-3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*logrus.TextFormatter, *bytes.Buffer, string, interface {})"p�go.weak.type.*func(*"".TextFormatter, *bytes.Buffer, string, interface {})�"runtime.zerovalue���type.func(*"".TextFormatter, *bytes.Buffer, string, interface {})���type.func(*"".TextFormatter, *bytes.Buffer, string, interface {})�,type.*"".TextFormatter�$type.*bytes.Buffer�type.string�"type.interface {}��go.string."func(*logrus.TextFormatter, *bytes.Buffer, *logrus.Entry, []string)"��Cfunc(*logrus.TextFormatter, *bytes.Buffer, *logrus.Entry, []string) �go.string."func(*logrus.TextFormatter, *bytes.Buffer, *logrus.Entry, []string)"��type.func(*"".TextFormatter, *bytes.Buffer, *"".Entry, []string)����ly3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*logrus.TextFormatter, *bytes.Buffer, *logrus.Entry, []string)"p�go.weak.type.*func(*"".TextFormatter, *bytes.Buffer, *"".Entry, []string)�"runtime.zerovalue���type.func(*"".TextFormatter, *bytes.Buffer, *"".Entry, []string)���type.func(*"".TextFormatter, *bytes.Buffer, *"".Entry, []string)�,type.*"".TextFormatter�$type.*bytes.Buffer�type.*"".Entry�type.[]string�4go.string."appendKeyValue"@>appendKeyValue 4go.string."appendKeyValue"�jgo.string."func(*bytes.Buffer, string, interface {})"�t)func(*bytes.Buffer, string, interface {}) jgo.string."func(*bytes.Buffer, string, interface {})"�\type.func(*bytes.Buffer, string, interface {})��g|�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pjgo.string."func(*bytes.Buffer, string, interface {})"pngo.weak.type.*func(*bytes.Buffer, string, interface {})�"runtime.zerovalue��\type.func(*bytes.Buffer, string, interface {})��\type.func(*bytes.Buffer, string, interface {})�$type.*bytes.Buffer�type.string�"type.interface {}�0go.string."printColored"@: printColored 0go.string."printColored"�pgo.string."func(*bytes.Buffer, *logrus.Entry, []string)"�z,func(*bytes.Buffer, *logrus.Entry, []string) pgo.string."func(*bytes.Buffer, *logrus.Entry, []string)"�Ztype.func(*bytes.Buffer, *"".Entry, []string)��� ��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ppgo.string."func(*bytes.Buffer, *logrus.Entry, []string)"plgo.weak.type.*func(*bytes.Buffer, *"".Entry, []string)�"runtime.zerovalue��Ztype.func(*bytes.Buffer, *"".Entry, []string)��Ztype.func(*bytes.Buffer, *"".Entry, []string)�$type.*bytes.Buffer�type.*"".Entry�type.[]string�,type.*"".TextFormatter���l:�62 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."*logrus.TextFormatter"p>go.weak.type.**"".TextFormatter�"runtime.zerovalue�*type."".TextFormatter`�,type.*"".TextFormatter��,type.*"".TextFormatter�$go.string."Format"�Jtype.func(*"".Entry) ([]uint8, error)�ptype.func(*"".TextFormatter, *"".Entry) ([]uint8, error)�4"".(*TextFormatter).Format�4"".(*TextFormatter).Format�4go.string."appendKeyValue"�"go.importpath."".�\type.func(*bytes.Buffer, string, interface {})��type.func(*"".TextFormatter, *bytes.Buffer, string, interface {})�D"".(*TextFormatter).appendKeyValue�D"".(*TextFormatter).appendKeyValue�0go.string."printColored"�"go.importpath."".�Ztype.func(*bytes.Buffer, *"".Entry, []string)��type.func(*"".TextFormatter, *bytes.Buffer, *"".Entry, []string)�@"".(*TextFormatter).printColored�@"".(*TextFormatter).printColored�bruntime.gcbits.0x84440000000000000000000000000000 �D�@go.string."logrus.TextFormatter"PJlogrus.TextFormatter @go.string."logrus.TextFormatter"�.go.string."ForceColors"@8 ForceColors .go.string."ForceColors"�2go.string."DisableColors"@< DisableColors 2go.string."DisableColors"�8go.string."DisableTimestamp"PBDisableTimestamp 8go.string."DisableTimestamp"�2go.string."FullTimestamp"@< FullTimestamp 2go.string."FullTimestamp"�4go.string."DisableSorting"@>DisableSorting 4go.string."DisableSorting"�2go.string."TextFormatter"@< TextFormatter 2go.string."TextFormatter"�*type."".TextFormatter�� {,��, 4type..alg."".TextFormatter0bruntime.gcbits.0x84440000000000000000000000000000P@go.string."logrus.TextFormatter"p,type.*"".TextFormatter�"runtime.zerovalue��*type."".TextFormatter�.go.string."ForceColors"�type.bool�2go.string."DisableColors"�type.bool�8go.string."DisableTimestamp"�type.bool�2go.string."FullTimestamp"�type.bool�6go.string."TimestampFormat"�type.string�4go.string."DisableSorting"�type.bool`�*type."".TextFormatter�2go.string."TextFormatter"�"go.importpath."".��*type."".TextFormatter�*go.string."[20]uint8"@4 [20]uint8 *go.string."[20]uint8"�type.[20]uint8��~���  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P*go.string."[20]uint8"p.go.weak.type.*[20]uint8�"runtime.zerovalue�type.uint8�type.[]uint8�>go.typelink.[20]uint8/[20]uint8type.[20]uint8�(go.string."[4]uint8"@2[4]uint8 (go.string."[4]uint8"�type.[4]uint8���B� � runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P(go.string."[4]uint8"p,go.weak.type.*[4]uint8�"runtime.zerovalue�type.uint8�type.[]uint8�:go.typelink.[4]uint8/[4]uint8type.[4]uint8�6go.string."*logrus.Termios"@@*logrus.Termios 6go.string."*logrus.Termios"� type.*"".Termios���P��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."*logrus.Termios"p2go.weak.type.**"".Termios�"runtime.zerovalue�type."".Termios�4go.string."logrus.Termios"@>logrus.Termios 4go.string."logrus.Termios"�"go.string."Iflag"0,Iflag "go.string."Iflag"�"go.string."Oflag"0,Oflag "go.string."Oflag"�"go.string."Cflag"0,Cflag "go.string."Cflag"�"go.string."Lflag"0,Lflag "go.string."Lflag"�go.string."Cc"0&Cc go.string."Cc"�*go.string."Pad_cgo_0"@4 Pad_cgo_0 *go.string."Pad_cgo_0"�$go.string."Ispeed"0.Ispeed $go.string."Ispeed"�$go.string."Ospeed"0.Ospeed $go.string."Ospeed"�&go.string."Termios"00Termios &go.string."Termios"�type."".Termios��H�ۆi� 48@4  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P4go.string."logrus.Termios"p type.*"".Termios�"runtime.zerovalue��type."".Termios�"go.string."Iflag"�type.uint64�"go.string."Oflag"�type.uint64�"go.string."Cflag"�type.uint64�"go.string."Lflag"�type.uint64�go.string."Cc"�type.[20]uint8�*go.string."Pad_cgo_0"�type.[4]uint8�$go.string."Ispeed"�type.uint64�$go.string."Ospeed"�type.uint64`�type."".Termios�&go.string."Termios"�"go.importpath."".��type."".Termios�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236� +__.PKGDEF 0 0 0 644 19986 ` +go object darwin amd64 go1.4.2 X:precisestack + +$$ +package opts + import net "net" + import runtime "runtime" + import bufio "bufio" + import volume "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume" + import ulimit "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit" + import os "os" + import strings "strings" + import fmt "fmt" + import regexp "regexp" + import parsers "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers" + import path "path" + import syntax "regexp/syntax" // indirect + type @"regexp/syntax".InstOp uint8 + func (@"regexp/syntax".i·2 @"regexp/syntax".InstOp) String () (? string) { if uint(@"regexp/syntax".i·2) >= uint(len(@"regexp/syntax".instOpNames)) { return "" }; return @"regexp/syntax".instOpNames[@"regexp/syntax".i·2] } + type @"regexp/syntax".Inst struct { Op @"regexp/syntax".InstOp; Out uint32; Arg uint32; Rune []rune } + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") MatchEmptyWidth (@"regexp/syntax".before·3 rune, @"regexp/syntax".after·4 rune) (? bool) + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") MatchRune (@"regexp/syntax".r·3 rune) (? bool) + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") MatchRunePos (@"regexp/syntax".r·3 rune) (? int) + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") String () (? string) + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") @"regexp/syntax".op () (? @"regexp/syntax".InstOp) + type @"regexp/syntax".EmptyOp uint8 + type @"regexp/syntax".Prog struct { Inst []@"regexp/syntax".Inst; Start int; NumCap int } + func (@"regexp/syntax".p·3 *@"regexp/syntax".Prog "esc:0x0") Prefix () (@"regexp/syntax".prefix·1 string, @"regexp/syntax".complete·2 bool) + func (@"regexp/syntax".p·2 *@"regexp/syntax".Prog "esc:0x0") StartCond () (? @"regexp/syntax".EmptyOp) + func (@"regexp/syntax".p·2 *@"regexp/syntax".Prog "esc:0x0") String () (? string) + func (@"regexp/syntax".p·3 *@"regexp/syntax".Prog "esc:0x1") @"regexp/syntax".skipNop (@"regexp/syntax".pc·4 uint32) (? *@"regexp/syntax".Inst, ? uint32) + type @"regexp".onePassInst struct { ? @"regexp/syntax".Inst; Next []uint32 } + type @"regexp".onePassProg struct { Inst []@"regexp".onePassInst; Start int; NumCap int } + import sync "sync" // indirect + type @"sync".Mutex struct { @"sync".state int32; @"sync".sema uint32 } + func (@"sync".m·1 *@"sync".Mutex) Lock () + func (@"sync".m·1 *@"sync".Mutex) Unlock () + type @"regexp".thread struct { @"regexp".inst *@"regexp/syntax".Inst; @"regexp".cap []int } + type @"regexp".entry struct { @"regexp".pc uint32; @"regexp".t *@"regexp".thread } + type @"regexp".queue struct { @"regexp".sparse []uint32; @"regexp".dense []@"regexp".entry } + type @"regexp".inputBytes struct { @"regexp".str []byte } + func (@"regexp".i·2 *@"regexp".inputBytes "esc:0x0") @"regexp".canCheckPrefix () (? bool) { return true } + func (@"regexp".i·2 *@"regexp".inputBytes "esc:0x0") @"regexp".context (@"regexp".pos·3 int) (? @"regexp/syntax".EmptyOp) + func (@"regexp".i·2 *@"regexp".inputBytes "esc:0x0") @"regexp".hasPrefix (@"regexp".re·3 *@"regexp".Regexp "esc:0x0") (? bool) + func (@"regexp".i·2 *@"regexp".inputBytes "esc:0x0") @"regexp".index (@"regexp".re·3 *@"regexp".Regexp "esc:0x0", @"regexp".pos·4 int) (? int) + func (@"regexp".i·3 *@"regexp".inputBytes "esc:0x0") @"regexp".step (@"regexp".pos·4 int) (? rune, ? int) + type @"regexp".inputString struct { @"regexp".str string } + func (@"regexp".i·2 *@"regexp".inputString "esc:0x0") @"regexp".canCheckPrefix () (? bool) { return true } + func (@"regexp".i·2 *@"regexp".inputString "esc:0x0") @"regexp".context (@"regexp".pos·3 int) (? @"regexp/syntax".EmptyOp) + func (@"regexp".i·2 *@"regexp".inputString "esc:0x0") @"regexp".hasPrefix (@"regexp".re·3 *@"regexp".Regexp "esc:0x0") (? bool) + func (@"regexp".i·2 *@"regexp".inputString "esc:0x0") @"regexp".index (@"regexp".re·3 *@"regexp".Regexp "esc:0x0", @"regexp".pos·4 int) (? int) + func (@"regexp".i·3 *@"regexp".inputString "esc:0x0") @"regexp".step (@"regexp".pos·4 int) (? rune, ? int) + import io "io" // indirect + type @"io".RuneReader interface { ReadRune() (@"io".r rune, @"io".size int, @"io".err error) } + type @"regexp".inputReader struct { @"regexp".r @"io".RuneReader; @"regexp".atEOT bool; @"regexp".pos int } + func (@"regexp".i·2 *@"regexp".inputReader "esc:0x0") @"regexp".canCheckPrefix () (? bool) { return false } + func (@"regexp".i·2 *@"regexp".inputReader "esc:0x0") @"regexp".context (@"regexp".pos·3 int) (? @"regexp/syntax".EmptyOp) { return @"regexp/syntax".EmptyOp(0x0) } + func (@"regexp".i·2 *@"regexp".inputReader "esc:0x0") @"regexp".hasPrefix (@"regexp".re·3 *@"regexp".Regexp "esc:0x0") (? bool) { return false } + func (@"regexp".i·2 *@"regexp".inputReader "esc:0x0") @"regexp".index (@"regexp".re·3 *@"regexp".Regexp "esc:0x0", @"regexp".pos·4 int) (? int) { return -0x1 } + func (@"regexp".i·3 *@"regexp".inputReader) @"regexp".step (@"regexp".pos·4 int) (? rune, ? int) + type @"regexp".input interface { @"regexp".canCheckPrefix() (? bool); @"regexp".context(@"regexp".pos int) (? @"regexp/syntax".EmptyOp); @"regexp".hasPrefix(@"regexp".re *@"regexp".Regexp) (? bool); @"regexp".index(@"regexp".re *@"regexp".Regexp, @"regexp".pos int) (? int); @"regexp".step(@"regexp".pos int) (@"regexp".r rune, @"regexp".width int) } + type @"regexp".machine struct { @"regexp".re *@"regexp".Regexp; @"regexp".p *@"regexp/syntax".Prog; @"regexp".op *@"regexp".onePassProg; @"regexp".q0 @"regexp".queue; @"regexp".q1 @"regexp".queue; @"regexp".pool []*@"regexp".thread; @"regexp".matched bool; @"regexp".matchcap []int; @"regexp".inputBytes @"regexp".inputBytes; @"regexp".inputString @"regexp".inputString; @"regexp".inputReader @"regexp".inputReader } + func (@"regexp".m·2 *@"regexp".machine) @"regexp".add (@"regexp".q·3 *@"regexp".queue, @"regexp".pc·4 uint32, @"regexp".pos·5 int, @"regexp".cap·6 []int "esc:0x0", @"regexp".cond·7 @"regexp/syntax".EmptyOp, @"regexp".t·8 *@"regexp".thread) (? *@"regexp".thread) + func (@"regexp".m·2 *@"regexp".machine) @"regexp".alloc (@"regexp".i·3 *@"regexp/syntax".Inst) (? *@"regexp".thread) + func (@"regexp".m·1 *@"regexp".machine) @"regexp".clear (@"regexp".q·2 *@"regexp".queue) + func (@"regexp".m·1 *@"regexp".machine) @"regexp".free (@"regexp".t·2 *@"regexp".thread) { @"regexp".m·1.@"regexp".inputBytes.@"regexp".str = nil; @"regexp".m·1.@"regexp".inputString.@"regexp".str = ""; @"regexp".m·1.@"regexp".inputReader.@"regexp".r = nil; @"regexp".m·1.@"regexp".pool = append(@"regexp".m·1.@"regexp".pool, @"regexp".t·2) } + func (@"regexp".m·1 *@"regexp".machine) @"regexp".init (@"regexp".ncap·2 int) + func (@"regexp".m·2 *@"regexp".machine) @"regexp".match (@"regexp".i·3 @"regexp".input, @"regexp".pos·4 int) (? bool) + func (@"regexp".m·2 *@"regexp".machine "esc:0x2") @"regexp".newInputBytes (@"regexp".b·3 []byte) (? @"regexp".input) { @"regexp".m·2.@"regexp".inputBytes.@"regexp".str = @"regexp".b·3; return &@"regexp".m·2.@"regexp".inputBytes } + func (@"regexp".m·2 *@"regexp".machine "esc:0x2") @"regexp".newInputReader (@"regexp".r·3 @"io".RuneReader) (? @"regexp".input) { @"regexp".m·2.@"regexp".inputReader.@"regexp".r = @"regexp".r·3; @"regexp".m·2.@"regexp".inputReader.@"regexp".atEOT = false; @"regexp".m·2.@"regexp".inputReader.@"regexp".pos = 0x0; return &@"regexp".m·2.@"regexp".inputReader } + func (@"regexp".m·2 *@"regexp".machine "esc:0x2") @"regexp".newInputString (@"regexp".s·3 string) (? @"regexp".input) { @"regexp".m·2.@"regexp".inputString.@"regexp".str = @"regexp".s·3; return &@"regexp".m·2.@"regexp".inputString } + func (@"regexp".m·2 *@"regexp".machine) @"regexp".onepass (@"regexp".i·3 @"regexp".input, @"regexp".pos·4 int) (? bool) + func (@"regexp".m·1 *@"regexp".machine) @"regexp".step (@"regexp".runq·2 *@"regexp".queue, @"regexp".nextq·3 *@"regexp".queue, @"regexp".pos·4 int, @"regexp".nextPos·5 int, @"regexp".c·6 rune, @"regexp".nextCond·7 @"regexp/syntax".EmptyOp) + type @"regexp".Regexp struct { @"regexp".expr string; @"regexp".prog *@"regexp/syntax".Prog; @"regexp".onepass *@"regexp".onePassProg; @"regexp".prefix string; @"regexp".prefixBytes []byte; @"regexp".prefixComplete bool; @"regexp".prefixRune rune; @"regexp".prefixEnd uint32; @"regexp".cond @"regexp/syntax".EmptyOp; @"regexp".numSubexp int; @"regexp".subexpNames []string; @"regexp".longest bool; @"regexp".mu @"sync".Mutex; @"regexp".machine []*@"regexp".machine } + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") Expand (@"regexp".dst·3 []byte "esc:0x2", @"regexp".template·4 []byte "esc:0x0", @"regexp".src·5 []byte "esc:0x0", @"regexp".match·6 []int "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") ExpandString (@"regexp".dst·3 []byte "esc:0x2", @"regexp".template·4 string, @"regexp".src·5 string "esc:0x0", @"regexp".match·6 []int "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) Find (@"regexp".b·3 []byte) (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) FindAll (@"regexp".b·3 []byte, @"regexp".n·4 int) (? [][]byte) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllIndex (@"regexp".b·3 []byte, @"regexp".n·4 int) (? [][]int) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllString (@"regexp".s·3 string, @"regexp".n·4 int) (? []string) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllStringIndex (@"regexp".s·3 string, @"regexp".n·4 int) (? [][]int) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllStringSubmatch (@"regexp".s·3 string, @"regexp".n·4 int) (? [][]string) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllStringSubmatchIndex (@"regexp".s·3 string, @"regexp".n·4 int) (? [][]int) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllSubmatch (@"regexp".b·3 []byte, @"regexp".n·4 int) (? [][][]byte) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllSubmatchIndex (@"regexp".b·3 []byte, @"regexp".n·4 int) (? [][]int) + func (@"regexp".re·2 *@"regexp".Regexp) FindIndex (@"regexp".b·3 []byte) (@"regexp".loc·1 []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindReaderIndex (@"regexp".r·3 @"io".RuneReader) (@"regexp".loc·1 []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindReaderSubmatchIndex (@"regexp".r·3 @"io".RuneReader) (? []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindString (@"regexp".s·3 string) (? string) + func (@"regexp".re·2 *@"regexp".Regexp) FindStringIndex (@"regexp".s·3 string) (@"regexp".loc·1 []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindStringSubmatch (@"regexp".s·3 string) (? []string) + func (@"regexp".re·2 *@"regexp".Regexp) FindStringSubmatchIndex (@"regexp".s·3 string) (? []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindSubmatch (@"regexp".b·3 []byte) (? [][]byte) + func (@"regexp".re·2 *@"regexp".Regexp) FindSubmatchIndex (@"regexp".b·3 []byte) (? []int) + func (@"regexp".re·3 *@"regexp".Regexp "esc:0x1") LiteralPrefix () (@"regexp".prefix·1 string, @"regexp".complete·2 bool) { return @"regexp".re·3.@"regexp".prefix, @"regexp".re·3.@"regexp".prefixComplete } + func (@"regexp".re·1 *@"regexp".Regexp "esc:0x0") Longest () { @"regexp".re·1.@"regexp".longest = true } + func (@"regexp".re·2 *@"regexp".Regexp) Match (@"regexp".b·3 []byte) (? bool) + func (@"regexp".re·2 *@"regexp".Regexp) MatchReader (@"regexp".r·3 @"io".RuneReader) (? bool) + func (@"regexp".re·2 *@"regexp".Regexp) MatchString (@"regexp".s·3 string) (? bool) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") NumSubexp () (? int) { return @"regexp".re·2.@"regexp".numSubexp } + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAll (@"regexp".src·3 []byte, @"regexp".repl·4 []byte "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllFunc (@"regexp".src·3 []byte, @"regexp".repl·4 func(? []byte) (? []byte) "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllLiteral (@"regexp".src·3 []byte, @"regexp".repl·4 []byte "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllLiteralString (@"regexp".src·3 string, @"regexp".repl·4 string "esc:0x0") (? string) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllString (@"regexp".src·3 string, @"regexp".repl·4 string) (? string) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllStringFunc (@"regexp".src·3 string, @"regexp".repl·4 func(? string) (? string) "esc:0x0") (? string) + func (@"regexp".re·2 *@"regexp".Regexp) Split (@"regexp".s·3 string, @"regexp".n·4 int) (? []string) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x1") String () (? string) { return @"regexp".re·2.@"regexp".expr } + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x1") SubexpNames () (? []string) { return @"regexp".re·2.@"regexp".subexpNames } + func (@"regexp".re·1 *@"regexp".Regexp) @"regexp".allMatches (@"regexp".s·2 string, @"regexp".b·3 []byte, @"regexp".n·4 int, @"regexp".deliver·5 func(? []int) "esc:0x0") + func (@"regexp".re·2 *@"regexp".Regexp) @"regexp".doExecute (@"regexp".r·3 @"io".RuneReader, @"regexp".b·4 []byte, @"regexp".s·5 string, @"regexp".pos·6 int, @"regexp".ncap·7 int) (? []int) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") @"regexp".expand (@"regexp".dst·3 []byte "esc:0x2", @"regexp".template·4 string, @"regexp".bsrc·5 []byte "esc:0x0", @"regexp".src·6 string "esc:0x0", @"regexp".match·7 []int "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) @"regexp".get () (? *@"regexp".machine) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") @"regexp".pad (@"regexp".a·3 []int "esc:0x2") (? []int) + func (@"regexp".re·1 *@"regexp".Regexp) @"regexp".put (@"regexp".z·2 *@"regexp".machine) + func (@"regexp".re·2 *@"regexp".Regexp) @"regexp".replaceAll (@"regexp".bsrc·3 []byte, @"regexp".src·4 string, @"regexp".nmatch·5 int, @"regexp".repl·6 func(@"regexp".dst []byte, @"regexp".m []int) (? []byte) "esc:0x0") (? []byte) + var @"".EnvironmentVariableRegexp *@"regexp".Regexp + func @"".ParseEnvFile (@"".filename·3 string) (? []string, ? error) + type @"".ErrBadEnvVariable struct { @"".msg string } + func (@"".e·2 @"".ErrBadEnvVariable) Error () (? string) + var @"".DefaultHost string + type @"net".IPMask []byte + func (@"net".m·3 @"net".IPMask "esc:0x0") Size () (@"net".ones·1 int, @"net".bits·2 int) + func (@"net".m·2 @"net".IPMask "esc:0x0") String () (? string) + type @"net".IP []byte + func (@"net".ip·2 @"net".IP "esc:0x0") DefaultMask () (? @"net".IPMask) + func (@"net".ip·2 @"net".IP "esc:0x0") Equal (@"net".x·3 @"net".IP "esc:0x0") (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsGlobalUnicast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsInterfaceLocalMulticast () (? bool) { return len(@"net".ip·2) == 0x10 && @"net".ip·2[0x0] == byte(0xFF) && @"net".ip·2[0x1] & byte(0xF) == byte(0x1) } + func (@"net".ip·2 @"net".IP "esc:0x0") IsLinkLocalMulticast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsLinkLocalUnicast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsLoopback () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsMulticast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsUnspecified () (? bool) + func (@"net".ip·3 @"net".IP "esc:0x0") MarshalText () (? []byte, ? error) + func (@"net".ip·2 @"net".IP "esc:0x0") Mask (@"net".mask·3 @"net".IPMask "esc:0x0") (? @"net".IP) + func (@"net".ip·2 @"net".IP "esc:0x0") String () (? string) + func (@"net".ip·2 @"net".IP "esc:0x2") To16 () (? @"net".IP) + func (@"net".ip·2 @"net".IP "esc:0x2") To4 () (? @"net".IP) + func (@"net".ip·2 *@"net".IP "esc:0x0") UnmarshalText (@"net".text·3 []byte "esc:0x0") (? error) + type @"".IpOpt struct { ? *@"net".IP } + func (@"".o·2 *@"".IpOpt "esc:0x0") Set (@"".val·3 string) (? error) + func (@"".o·2 *@"".IpOpt "esc:0x0") String () (? string) + func @"".NewIpOpt (@"".ref·2 *@"net".IP, @"".defaultVal·3 string) (? *@"".IpOpt) + var @"".DefaultHTTPHost string + var @"".DefaultHTTPPort int + var @"".DefaultUnixSocket string + type @"".ValidatorFctType func(@"".val string) (? string, ? error) + type @"".ListOpts struct { @"".values *[]string; @"".validator @"".ValidatorFctType } + func (@"".opts·1 *@"".ListOpts) Delete (@"".key·2 string "esc:0x0") + func (@"".opts·2 *@"".ListOpts "esc:0x0") Get (@"".key·3 string "esc:0x0") (? bool) + func (@"".opts·2 *@"".ListOpts "esc:0x1") GetAll () (? []string) { return *@"".opts·2.@"".values } + func (@"".opts·2 *@"".ListOpts "esc:0x0") GetMap () (? map[string]struct {}) + func (@"".opts·2 *@"".ListOpts "esc:0x0") Len () (? int) { return len(*@"".opts·2.@"".values) } + func (@"".opts·2 *@"".ListOpts) Set (@"".value·3 string) (? error) + func (@"".opts·2 *@"".ListOpts) String () (? string) + func @"".NewListOpts (@"".validator·2 @"".ValidatorFctType) (? @"".ListOpts) + func @"".NewListOptsRef (@"".values·2 *[]string, @"".validator·3 @"".ValidatorFctType) (? *@"".ListOpts) { return (&@"".ListOpts{ @"".values:@"".values·2, @"".validator:@"".validator·3 }) } + type @"".MapOpts struct { @"".values map[string]string; @"".validator @"".ValidatorFctType } + func (@"".opts·2 *@"".MapOpts "esc:0x0") Set (@"".value·3 string) (? error) + func (@"".opts·2 *@"".MapOpts) String () (? string) + func @"".NewMapOpts (@"".values·2 map[string]string, @"".validator·3 @"".ValidatorFctType) (? *@"".MapOpts) { if @"".values·2 == nil { @"".values·2 = make(map[string]string, 0x0) }; return (&@"".MapOpts{ @"".values:@"".values·2, @"".validator:@"".validator·3 }) } + type @"".ValidatorFctListType func(@"".val string) (? []string, ? error) + func @"".ValidateAttach (@"".val·3 string "esc:0x2") (? string, ? error) + func @"".ValidateLink (@"".val·3 string) (? string, ? error) + func @"".ValidateDevice (@"".val·3 string) (? string, ? error) + func @"".ValidatePath (@"".val·3 string) (? string, ? error) + func @"".ValidateEnv (@"".val·3 string) (? string, ? error) + func @"".ValidateIPAddress (@"".val·3 string) (? string, ? error) + func @"".ValidateMACAddress (@"".val·3 string "esc:0x2") (? string, ? error) + func @"".ValidateDNSSearch (@"".val·3 string) (? string, ? error) + func @"".ValidateExtraHost (@"".val·3 string) (? string, ? error) + func @"".ValidateLabel (@"".val·3 string) (? string, ? error) + func @"".ValidateHost (@"".val·3 string) (? string, ? error) + type @"github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit".Rlimit struct { Type int "json:\"type,omitempty\""; Hard uint64 "json:\"hard,omitempty\""; Soft uint64 "json:\"soft,omitempty\"" } + type @"github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit".Ulimit struct { Name string; Hard int64; Soft int64 } + func (@"github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit".u·3 *@"github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit".Ulimit) GetRlimit () (? *@"github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit".Rlimit, ? error) + func (@"github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit".u·2 *@"github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit".Ulimit) String () (? string) + type @"".UlimitOpt struct { @"".values *map[string]*@"github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit".Ulimit } + func (@"".o·2 *@"".UlimitOpt "esc:0x0") GetList () (? []*@"github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit".Ulimit) + func (@"".o·2 *@"".UlimitOpt "esc:0x0") Set (@"".val·3 string) (? error) + func (@"".o·2 *@"".UlimitOpt "esc:0x0") String () (? string) + func @"".NewUlimitOpt (@"".ref·2 *map[string]*@"github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit".Ulimit) (? *@"".UlimitOpt) { if @"".ref·2 == nil { @"".ref·2 = (&map[string]*@"github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit".Ulimit{ }) }; return (&@"".UlimitOpt{ @"".values:@"".ref·2 }) } + func @"".init () + var @"regexp/syntax".instOpNames []string + +$$ +_go_.6 0 0 0 644 198312 ` +go object darwin amd64 go1.4.2 X:precisestack + +! +go13ldbufio.a +fmt.aos.aregexp.astrings.a +net.a path.a�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers.a�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume.a�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.a�"".ParseEnvFile�7�6eH� %H��$h���H;Aw���H��HDŽ$0HDŽ$8HDŽ$@HDŽ$HHDŽ$PH��$ H�$H��$(H�\$�H�T$H�L$H�\$ H��$�H��H��$���H�H�$�H�t$H��tb1�1�H��$�H��$0H��$�H��$8H��$�H��$@H��$�H��$HH��$�H��$P��H��É�H�T$PH�$H� Qj�YYH���] H�H�$�H�\$H���7 1�1�H��$hH��$pH��$xH�\$PH�\$`H�1�H9��� H�L$`H��$H��$�H��$ H��$�H�H�$H�D$H�D$�H�\$H��$�H�\$ H��$�H�\$(H��$�H�H�$�H�|$H��H���6 1��H�L$XH� $H�<$� H��$�H�\$H��$�H�\$�H�\$XH�$H�<$�� +H�$H�H�\$�H�D$XH�@H�$H�<$�� +H�$8H��$�H�\$H��$�H�\$H��$�H�\$�H�\$XH�\$HH�\$HH�$�H�T$H�\$���8 H�D$hH�D$pH��� H�j H�$H��H��H�H�H��L�L$H�t$ L��$H��$L�L$hL��$�H�t$pH��$�H����L��$�H�H�;H��$�H�CH��$�H��$�H9��yH9��wL��$H��$H9��WL� $H�D$H�|$H�D$�L��$�H��$��\$ ���!H��<� L� $H�t$H�H�l$H��H��H�H�H�D$ �H�T$(H�L$0H�D$8H��$�H��$�H��H��$���H�,$H��H��H�H�H�H�\$H�H�\$�H�T$ H�L$(H�H�$H��$�H�T$H��$�H�L$�H��$�H��$��\$����H��$HH��$PH��$XH�H�CH��$XH����H��H��H��$�H��$�H��$�H�H�$H��$HH�\$�H�T$H�L$H��$�H�$H��$�H�T$H��$H�L$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�T$(H�L$0H��$8H�H�CH��$H��$8H��$H��$@H�H�$�H�t$H����1�1�H��$�H��$0H��$�H��$8H��$�H��$@H�H�$H�H�\$H�H�\$H��$8H�\$�H�\$ H��$HH�\$(H��$P��H��É�h�����H���H��$�H���qH��$HH��$PH��$�1��H��$�H���9H��H��H��$�H��$�H��$�H�H�$H��$HH�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$H�D$�H�H�$H��$�H��$���H��H�\$�H�L$H�D$H��$�H��H�$H��$�H�L$H��$H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�\$(H��$H�\$0H��$H��$hH��$pH��$xH��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��H��$�H��$�H��$�H��Hk�H�H�$H��$H�\$H��$H�\$�H��$�H��$�H��$�H��$hH��$pH��$x����� ������H��$�H�$H��$�H�\$�H�\$H��$HH�\$H��$PH��$�H�$H��$�H�\$�H�\$H��$(H�\$H��$0H��$�1��H��$�H���H��H��H��$�H��$�H��$�H�H�$H��$HH�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$H�D$�H�H�$H��$(H�\$�H�L$H�D$H��$�H��H�$H��$�H�L$H��$H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�\$(H��$H�\$0H��$H��$hH��$pH��$xH��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��H��$�H��$�H��$�H��Hk�H�H�$H��$H�\$H��$H�\$�H��$�H��$�H��$�H��$hH��$pH��$x������������ ����1������� ������H��H�D$xHDŽ$�H�T$@H����H�J`H�RhH��$�H��$�H�-H9���H� $H�T$H�-H�l$H�-H�l$�H�t$@�\$ ��t_1�1�H��$hH��$0H��$pH��$8H��$xH��$@H�T$xH��$HH��$�H��$P��H���H��t +H�V`H�Nh땉���)����%�b����%�$����%�����������H�H�$H�H�\$H�H�\$�H�D$������������H��ä +00runtime.morestack_noctxt�os.Open�type.[0]string�"runtime.newobject�&runtime.deferreturn�&os.(*File).Close·f�"runtime.deferproc�type.[0]string�"runtime.newobject�4go.itab.*os.File.io.Reader�type.[]uint8�"runtime.makeslice�$type.bufio.Scanner�"runtime.newobject�� runtime.duffzero� 2runtime.writebarrieriface� +$bufio.ScanLines·f� +.runtime.writebarrierptr� 2runtime.writebarrierslice� *bufio.(*Scanner).Scan� 2runtime.slicebytetostring�go.string."#"� runtime.eqstring�go.string."="�strings.SplitN�"".whiteSpaces�"".whiteSpaces� strings.TrimLeft�8"".EnvironmentVariableRegexp�8regexp.(*Regexp).MatchString�type.string�runtime.convT2E�2runtime.writebarrieriface�zgo.string."variable '%s' is not a valid environment variable"�fmt.Sprintf�type.[0]string�"runtime.newobject�2type."".ErrBadEnvVariable�type.error�Dgo.itab."".ErrBadEnvVariable.error�runtime.convT2I�&runtime.deferreturn�� runtime.duffzero�type.string�runtime.convT2E�2runtime.writebarrieriface�type.string�runtime.convT2E� 2runtime.writebarrieriface� "go.string."%s=%s"�!fmt.Sprintf�"type.[]string�#"runtime.growslice�$4runtime.writebarrierstring�%$runtime.panicindex�&"strings.TrimSpace�'os.Getenv�'� runtime.duffzero�(type.string�(runtime.convT2E�)2runtime.writebarrieriface�)type.string�*runtime.convT2E�+2runtime.writebarrieriface�+"go.string."%s=%s"�,fmt.Sprintf�-type.[]string�."runtime.growslice�/4runtime.writebarrierstring�0$runtime.panicindex�0$runtime.panicslice�2 io.EOF�2 io.EOF�2 io.EOF�2runtime.ifaceeq�4&runtime.deferreturn�5type.*os.File�5type.io.Reader�64go.itab.*os.File.io.Reader�6 runtime.typ2Itab�6&runtime.deferreturnp��"".autotmp_0060�type.error"".autotmp_0059type.uint64"".autotmp_0058type.uint64"".autotmp_0057type.int"".autotmp_0056type.int"".autotmp_0055type.[]string"".autotmp_0054"type.interface {}"".autotmp_0053"type.interface {}"".autotmp_0052*type.*[2]interface {}"".autotmp_0051&type.[]interface {}"".autotmp_0050type.uint64"".autotmp_0049type.uint64"".autotmp_0048type.int"".autotmp_0047type.int"".autotmp_0046type.[]string"".autotmp_0045"type.interface {}"".autotmp_0044"type.interface {}"".autotmp_0042&type.[]interface {}"".autotmp_0041type.*[0]string"".autotmp_0040type.[]string"".autotmp_0039�"type.interface {}"".autotmp_0037�&type.[]interface {}"".autotmp_0034type.string"".autotmp_0032�type.string"".autotmp_0031�&type.*bufio.Scanner"".autotmp_0030&type.*bufio.Scanner"".autotmp_0028�type.io.Reader"".autotmp_0027type.*[0]string"".autotmp_0025�type.[]string"".autotmp_0024type.[]string"".autotmp_0023type.string"".autotmp_0022�type.string"".autotmp_0021type.string"".autotmp_0020(type.[2]interface {}"".autotmp_0019�type.[]string"".autotmp_0018type.string"".autotmp_0017type.string"".autotmp_0016?(type.[2]interface {}"".autotmp_0015type.int"".autotmp_0014�2type."".ErrBadEnvVariable"".autotmp_0013type.string"".autotmp_0012�type.string"".autotmp_0011�(type.[1]interface {}"".autotmp_0009type.string"".autotmp_0008type.[]string"".autotmp_0005type.int"".autotmp_0004type.int"".autotmp_0002otype.[]uint8"".autotmp_0001�type.*os.File "".~r0�type.errorbufio.s·2�&type.*bufio.Scanner"strings.prefix·3�type.stringstrings.s·2�type.string "".~r0�type.stringbufio.r·2�type.io.Reader"".variable�type.string"".data�type.[]string"".line�type.string"".scanner�&type.*bufio.Scanner"".lines�type.[]string "".err�type.error +"".fh�type.*os.File "".~r2Ptype.error "".~r1 type.[]string"".filenametype.stringN%����� ��� ���� +��(a5�#;�!a�DOH���#$ *�-]�z9d%�7Lw��4�vQx�ez~]:'8~�~]A9�1 +> +h(Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ip.go�"".NewListOpts��eH� %H;aw���H��0H�D$@H�D$HH�H�$�H�D$H�L$8H�L$H�D$ H�D$(H�L$ H�L$H� $H�<$t\H�D$�H�\$H�$H�<$t9H�$H�\$H�\$�H�\$H��tH�+H�l$@H�kH�l$HH��0É��%뾉%� + 0runtime.morestack_noctxt^type.[]stringp"runtime.newobject�.runtime.writebarrierptr�.runtime.writebarrierptr0` "".autotmp_0111 type."".ListOpts"".autotmp_0110/"type.*"".ListOpts"".autotmp_0109"type.*"".ListOpts"".validator?0type."".ValidatorFctType "".~r1 type."".ListOpts"".validator0type."".ValidatorFctType`�_`�L,�7@$ETgclocals·6d340c3bdac448a6ef1256f331f68dd3Tgclocals·9307bf1379da22b408b9b243276c0115�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�""".NewListOptsRef��eH� %H;aw���H��H�H�$�H�D$H�D$H�$H�<$tKH�\$ H�\$�H�\$H�$H�<$t#H�$H�\$(H�\$�H�\$H�\$0H��É%�ԉ%� + 0runtime.morestack_noctxt: type."".ListOptsL"runtime.newobject�.runtime.writebarrierptr�.runtime.writebarrierptr00"".autotmp_0112"type.*"".ListOpts "".~r2 "type.*"".ListOpts"".validator0type."".ValidatorFctType"".valuestype.*[]string0f/0�VX%$WTgclocals·bd51743682bd6c0f7b9f2e8e6dffed99Tgclocals·e1ae6533a9e39048ba0735a2264ce16a�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�*"".(*ListOpts).String��eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$�H��$�H�+H���H�]H�\$pH�]H�\$xH�]H��$�H�\$HH�H�CH�\$HH����H��H��H�\$XH�T$`H�L$hH�H�$H�\$pH�\$�H�L$H�D$H�\$XH�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H�H�\$XH�\$H�\$`H�\$H�\$hH�\$ �H�L$(H�D$0H��$�H��$�H�ĈÉ�<����E����� +*0runtime.morestack_noctxt�type.[]string�runtime.convT2E�2runtime.writebarrieriface�go.string."%v"�fmt.Sprintf0� "".autotmp_0118�"type.interface {}"".autotmp_0116_&type.[]interface {}"".autotmp_0114/type.[]string"".autotmp_0113(type.[1]interface {} "".~r0type.string"".opts"type.*"".ListOpts"����� d:��d:Tgclocals·6d340c3bdac448a6ef1256f331f68dd3Tgclocals·7876b70d8da64fa07ca2fd3ecc71f905�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�$"".(*ListOpts).Set��eH� %H�D$�H;Aw���H��H��$�HDŽ$�HDŽ$�H�^1�H9���H��$�H�$H��$�H�\$H�VH���H��$�H�l$H�T$H�D$ H�L$(H�L$XH��H�D$PtH��$�H��$�H�Đ�H�l$@H��$�H�T$HH��$�H�.H���/H�UH�MH�]H�T$xH��$�H��$�H��H)�H��}FH�H�$H�T$`H�T$H�L$hH�L$H�D$pH�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$hH�D$pH��H�T$`H��Hk�H�H�$H��$�H�\$H��$�H�\$�H�T$`H�L$hH�D$pH��$�H�+H�,$H�<$tIH�T$xH�T$H��$�H�L$H��$�H�D$�HDŽ$�HDŽ$�H�ĐÉ%뮉E����� +*0runtime.morestack_noctxt� +�type.[]string�"runtime.growslice�4runtime.writebarrierstring�2runtime.writebarriersliceP�"".autotmp_0124_type.[]string"".autotmp_0123/type.[]string"".autotmp_0122type.[]string "".errtype.error"".v�type.string "".~r10type.error"".valuetype.string"".opts"type.*"".ListOpts("��������(pBC � q�NATgclocals·61fa3b017c2e156e481b3d912c20f49bTgclocals·1509598f597bd125bdfd9d44972821c7�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�*"".(*ListOpts).Delete� � eH� %H�D$�H;Aw���H���H��$�H��$�H�+H����H�UH�MH�]H��$�1�H��$�H�L$HH��$�H�l$HH9���H�T$XH����H�2H�JH�D$PH�D$@H�t$pH�L$xH9��?H�t$`H�4$H�L$hH�L$H��$�H�l$H�|$�L��$�H��$�H�T$XH�D$P�\$ ����I�+H�MH�D$@H9���I�+L�UI��H�D$@I��H��I�+H�}I�+H�UH9���H��I�+H�MH)�H��H)�H��H��t H��H��H�H��H��$�H��H��$�L��$�L��L��$�L��$�L��L��$�H��$�H�L��$�L)�H��~[H�H�$H��$�H�t$L�L$L�D$H�D$ �L��$�H��$�H�t$(H�\$0H��$�H�\$8H��$�L��H��H��$�Hk�H�H�,$H��$�H�\$H��H��H�\$�H��$�H��$�H��$�H��$�H�H��$�H��$�H��$�H��$�H�+H�,$H�<$t4H��$�H�t$H��$�H�T$H��$�H�D$�H���É%��� � H��H��H�l$HH9��y���H���É�y����E�)��� +*0runtime.morestack_noctxt� runtime.eqstring�type.[]string�"runtime.growslice� runtime.memmove� 2runtime.writebarrierslice� $runtime.panicslice� $runtime.panicslice0�,"".autotmp_0152type.uint64"".autotmp_0151type.uint64"".autotmp_0150type.int"".autotmp_0149type.uintptr"".autotmp_0148type.int"".autotmp_0147�type.[]string"".autotmp_0146type.uintptr"".autotmp_0144type.uint64"".autotmp_0143type.uint64"".autotmp_0142type.int"".autotmp_0141�type.[]string"".autotmp_0138_type.[]string"".autotmp_0137�type.string"".autotmp_0136�type.*string"".autotmp_0135�type.int"".autotmp_0134�type.int"".autotmp_0133type.[]string"".autotmp_0132/type.[]string"".k�type.string"".i�type.int "".keytype.string"".opts"type.*"".ListOpts&"����3���*�*x[�  ��f}ZTgclocals·2018557e3ee0abccf2865b16663e690bTgclocals·3548d93f69958cd35a7e42f3f32eff97�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�*"".(*ListOpts).GetMap��eH� %H�D$�H;Aw���H��H�H�$H�D$�H�\$H�\$0H��$�H�+H����H�UH�EH�]H��$�1�H�D$xH�D$ H�T$pH��H�l$ H9���H�D$8H����H�H�hH�L$(H�T$PH�l$XH�T$@H�T$`H�l$HH�l$hH�\$ H�H�$H�\$0H�\$H�\$`H�\$H�\$ H�\$�H�D$8H�L$(H��H��H�l$ H9��u���H�\$0H��$�H�ĈÉ�h����E���� +*0runtime.morestack_noctxtJ2type.map[string]struct {}nruntime.makemap�2type.map[string]struct {}�$runtime.mapassign1 �"".autotmp_0169otype.string"".autotmp_0168�type.*string"".autotmp_0167�type.int"".autotmp_0166�type.int"".autotmp_0165�type.struct {}"".autotmp_0164Otype.string"".autotmp_0163/type.[]string"".k�type.string "".ret�2type.map[string]struct {} "".~r02type.map[string]struct {}"".opts"type.*"".ListOpts"����� �"#pG6�ITgclocals·2148c3737b2bb476685a1100a2e8343eTgclocals·895437610367c247af3cb64952bed446�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�*"".(*ListOpts).GetAll`^H�\$H�+H��tH�]H�\$H�]H�\$H�]H�\$ ÉE��@ "".~r0type.[]string"".opts"type.*"".ListOpts00�0Tgclocals·13d3af77a5bf02af6db4588efb2ea811Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�$"".(*ListOpts).Get��eH� %H;aw���H��xH��$�H��$�H�+H����H�MH�EH�]H�\$p1�H�D$hH�D$(H�L$`H�l$(H9���H�L$8H����H�1H�AH�T$0H�t$PH�D$XH9�uSH�t$@H�4$H�D$HH�D$H��$�H�l$H�|$�H��$�H�T$0H�L$8�\$ ��t Ƅ$�H��x�H��H��H�l$(H9��n���Ƅ$�H��xÉ�i����E�"��� + 0runtime.morestack_noctxt� runtime.eqstring@�"".autotmp_0174Otype.string"".autotmp_0173type.*string"".autotmp_0172�type.int"".autotmp_0171�type.int"".autotmp_0170/type.[]string"".kotype.string "".~r10type.bool "".keytype.string"".opts"type.*"".ListOpts&����!��� �"jK  +  �jTgclocals·9ff42bf311af152488d11f0f78c8d5ceTgclocals·4398bb51467914f29637b614067b995f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�$"".(*ListOpts).Len@$H�\$H�+H�]H�\$�  "".~r0type.int"".opts"type.*"".ListOpts  � Tgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�""".(*MapOpts).Set��eH� %H�D$�H;Aw���H��H��$�H��$�H��$�HDŽ$�HDŽ$�H�Z1�H9�t^H� $H�D$H�RH���H�l$H�T$H�D$ H�L$(H�L$XH��H�D$PtH��$�H��$�H�Ĉ�H�l$@H��H�T$HH��H��$�H� $H��$�H�D$H�H�l$H��H��H�H�H�D$ �H��$�H�T$(H�D$0H�L$8H��$�H��uqH�D$`H�D$hH�H�$H�mH�l$H�T$pH��H�D$xv4H�T$H�\$`H�\$�HDŽ$�HDŽ$�H�Ĉ�� H�H�$H�mH�l$H��v/H�T$H��H�T$pH��H�D$xvH��H�\$��� �  +*0runtime.morestack_noctxt� +�go.string."="�strings.SplitN�,type.map[string]string�$runtime.mapassign1�$runtime.panicindex�,type.map[string]string�$runtime.mapassign1�$runtime.panicindex�$runtime.panicindexP�"".autotmp_0178Otype.string"".vals/type.[]string "".errotype.error"".v�type.string "".~r10type.error"".valuetype.string"".opts type.*"".MapOpts("�������Y�<�R + \J @m�gTgclocals·61fa3b017c2e156e481b3d912c20f49bTgclocals·729deb178891d0cb0d4c5b2058f91105�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�("".(*MapOpts).String��eH� %H;aw���H��pHDŽ$�HDŽ$�H�D$xH�(H�D$HH�H�@H�D$HH����H��H��H�T$`H�L$hH�D$XH�$H�H�D$8H�D$H�l$@H�l$�H�H�,$H��H��H�H�H�\$XH�\$H�\$`H�\$H�\$hH�\$ �H�L$(H�D$0H��$�H��$�H��pÉ�a��� + 0runtime.morestack_noctxt�,type.map[string]string�2runtime.writebarrieriface�go.string."%v"�fmt.Sprintf0�"".autotmp_0182/&type.[]interface {}"".autotmp_0179O(type.[1]interface {} "".~r0type.string"".opts type.*"".MapOpts���� ��2��8/Tgclocals·6d340c3bdac448a6ef1256f331f68dd3Tgclocals·403a8d79fd24b295e8557f6970497aa3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�"".NewMapOpts��eH� %H;aw���H�� H�\$(1�H9�u#H�H�$H�D$�H�\$H�\$(H�H�$�H�D$H�D$H�$H�<$tKH�\$(H�\$�H�\$H�$H�<$t#H�$H�\$0H�\$�H�\$H�\$8H�� É%�ԉ%� + 0runtime.morestack_noctxtR,type.map[string]stringvruntime.makemap�type."".MapOpts�"runtime.newobject�.runtime.writebarrierptr�.runtime.writebarrierptr0@"".autotmp_0190 type.*"".MapOpts "".~r2  type.*"".MapOpts"".validator0type."".ValidatorFctType"".values,type.map[string]string@�?@ �� #X :>XTgclocals·bd51743682bd6c0f7b9f2e8e6dffed99Tgclocals·e1ae6533a9e39048ba0735a2264ce16a�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�""".ValidateAttach��eH� %H�D$�H;Aw���H���HDŽ$�HDŽ$�HDŽ$�HDŽ$�H��$�H�$H��$�H�\$�L�L$L�D$L�L$pL�L$`L�D$xL�D$hH��$�H���~H�-H��H���H��H��H��$�1�H��$�H�D$@H��$�H��H�l$@H9���H�L$HH���H�1H�AH�T$8H�t$pH�t$PH�D$xH�D$XI9�ufL� $L�D$H�t$H�D$�L�L$`L�D$hH�T$8H�L$H�\$ ��t0L��$�L��$�HDŽ$�HDŽ$�H����H��H��H�l$@H9��Q���H�H�,$H��H��H�H�H�\$H�H�CH�C�H�L$(H�D$0H��$�H��$�H��$�H��$�H��$�H��$�H���É�������{��� +*0runtime.morestack_noctxt�strings.ToLower�""".statictmp_0194�� runtime.duffcopy� runtime.eqstring�lgo.string."valid streams are STDIN, STDOUT and STDERR"�fmt.Errorf`�"".autotmp_0200type.string"".autotmp_0199�type.*string"".autotmp_0198type.int"".autotmp_0197type.int"".autotmp_0196_type.[3]string"".autotmp_0192�type.[]string"".autotmp_0191�type.string "".str�type.string"".s�type.string "".~r2@type.error "".~r1 type.string "".valtype.string("��������&�R<�@0 +x k��`Tgclocals·6d3fa487f5e45db9cb9199d2a5e0e216Tgclocals·5de38da5eeb0729bf417a80c29b78c42�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�"".ValidateLink��eH� %H;aw���H��@H�D$XH�D$`H�D$hH�D$pH�\$HH�$H�\$PH�\$�H�l$HH�T$PH�D$0H�L$8H��tH�l$XH�T$`H�D$hH�L$pH��@�H�l$XH�T$`H�D$hH�D$pH��@� + 0runtime.morestack_noctxt��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers.ParseLink`� "".~r2@type.error "".~r1 type.string "".valtype.string�n� ��>2' +Q_Tgclocals·13c015770347481bee7a16dde25a3e2fTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�""".ValidateDevice��eH� %H;aw���H��8H�D$PH�D$XH�D$`H�D$hH�\$@H�$H�\$HH�\$�D$�H�l$H�T$ H�L$(H�D$0H�l$PH�T$XH�L$`H�D$hH��8� + 0runtime.morestack_noctxt�"".validatePath`p "".~r2@type.error "".~r1 type.string "".valtype.stringpmo � �>R +V:Tgclocals·13c015770347481bee7a16dde25a3e2fTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�"".ValidatePath��eH� %H;aw���H��8H�D$PH�D$XH�D$`H�D$hH�\$@H�$H�\$HH�\$�D$�H�l$H�T$ H�L$(H�D$0H�l$PH�T$XH�L$`H�D$hH��8� + 0runtime.morestack_noctxt�"".validatePath`p "".~r2@type.error "".~r1 type.string "".valtype.stringpmo � �>R +V:Tgclocals·13c015770347481bee7a16dde25a3e2fTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�"".validatePath�@�@eH� %H��$����H;Aw���H��HDŽ$�HDŽ$�HDŽ$�HDŽ$�HDŽ$�HDŽ$�H�D$xHDŽ$�H��$�H�$H��$�H�t$H�5H�l$H��H�H��H��$�H��$�H�\$ H���;H��$�H��$�H��$�H�H�CH��$�H����H��H��H��$ H��$(H��$0H�H�$H��$�H�\$�H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$ H�\$H��$(H�\$H��$0H�\$ �H�L$(H�D$0H��$�H��$�H��$�H��$�H��$�H��$�H�ĈÉ�����H�$H�L$H�H�l$H��H��H�H�H�D$ �H��$�H��$�H�L$(H�D$0H�T$8H��$H��$H��H��$�~ H�YH���;H��$�H��$�H��$�H�H�CH��$�H����H��H��H��$ H��$(H��$0H�H�$H��$�H�\$�H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$ H�\$H��$(H�\$H��$0H�\$ �H�L$(H�D$0H��$�H��$�H��$�H��$�H��$�H��$�H�ĈÉ�����H���%H��H���H� H�kH��$�H� $H��$�H�l$�H�t$H�l$H��$�H��$�H��$�H�T$HH��$�H�D$PH����H�������/��H��<�KH��$�H��$�H��$�H��$�H��$�H�H�CH��$�H����H��H��H��$ H��$(H��$0H�H�$H��$�H�\$�H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$ H�\$H��$(H�\$H��$0H�\$ �H�L$(H�D$0H��$�H��$�H��$�H��$�H��$�H��$�H�ĈÉ�����H��$�H��$�HDŽ$�HDŽ$�H�Ĉ�1��v���� � H���H��H����H��H� H�kH�L$XH�l$`H�H�$H�H�\$H��$�H�L$H��$�H�l$�H�\$ �+@�l$GH�L$XH�D$`H�H�$H�H�\$H��$�H�L$H��$�H�D$�H��$H��$H�\$ �+@�l$F�|$G�1H�L$XH�D$`H�H�$H�H�\$H��$�H�L$H��$�H�D$�H��$H��$H�\$ �+@����1��\$F��$���<��H����H�H�FH��H����H��H�+H�l$xH�kH��$�H��$�H� $H��$�H�D$�H�\$H��$�H�\$H��$�H�\$xH��$�H��$�H��$�H��$81��H��$8H���.H��H��H��$ H��$(H��$0H�H�$H��$�H�\$�H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$�H�L$H�D$H��$ H��H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$ H�\$H��$(H�\$H��$0H�\$ �H�t$(H�l$0H��$�H��$������������� � H��H����H��H� H�kH��$�H� $H��$�H�l$�H�\$H��$�H�\$H��$�H��$81��H��$8H���DH��H��H��$ H��$(H��$0H�H�$H��$H��$��H�\$�H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$�H�L$H�D$H��$ H��H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$ H�\$H��$(H�\$H��$0H�\$ �H�t$(H�l$0H��$�H��$��4���� ������ H���'���� H������H��H����H��H�+H��$�H�kH��$�H��H����H�� H�+H�l$xH�kH��$�H��H���rH�� H� H�kH�L$hH�l$pH�H�$H�H�\$H��$�H�L$H��$�H�l$�H�\$ �+@�l$FH�L$hH�D$pH�H�$H�H�\$H��$�H�L$H��$�H�D$�H�\$ �+@�l$E�|$F��H�L$hH�D$pH�H�$H�H�\$H��$�H�L$H��$�H�D$�H�\$ �+@���d1��\$E��$��P<�HH�\$xH��$�H��$�H��$�H��$�H�H�CH��$�H����H��H��H��$ H��$(H��$0H�H�$H��$�H�\$�H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$ H�\$H��$(H�\$H��$0H�\$ �H�L$(H�D$0H��$�H��$�H��$�H��$�H��$�H��$�H�ĈÉ�����H��$�H��$�H��$�H��$�H�\$xH��$�H��$�H��$�H��$X1��H��$XH����H��H��H��$ H��$(H��$0H�H�$H��$H��$�KH�\$�H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$�H�L$H�D$H��$ H��H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$�H�L$H�D$H��$ H�� H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$ H�\$H��$(H�\$H��$0H�\$ �H�t$(H�l$0H��$�H��$��P���� ��_���H������� � � � � +00runtime.morestack_noctxt�go.string.":"�strings.Count�type.string�runtime.convT2E�2runtime.writebarrieriface�Lgo.string."bad format for volumes: %s"�fmt.Errorf�go.string.":"�strings.SplitN� type.string� runtime.convT2E� 2runtime.writebarrieriface� Lgo.string."bad format for volumes: %s"� fmt.Errorf�path.Clean�type.string�runtime.convT2E�2runtime.writebarrieriface�Lgo.string."%s is not an absolute path"�fmt.Errorf�$runtime.panicindex�$runtime.panicindex�(type.map[string]bool��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume.roModes�4runtime.mapaccess1_faststr�(type.map[string]bool��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume.rwModes�4runtime.mapaccess1_faststr�(type.map[string]bool��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume.rwModes�4runtime.mapaccess1_faststr�path.Clean�� runtime.duffzero� type.string� runtime.convT2E�!2runtime.writebarrieriface�!type.string�"runtime.convT2E�"2runtime.writebarrieriface�#"go.string."%s:%s"�$fmt.Sprintf�$$runtime.panicindex�$$runtime.panicindex�%path.Clean�&� runtime.duffzero�'type.string�'runtime.convT2E�(2runtime.writebarrieriface�(type.string�)runtime.convT2E�*2runtime.writebarrieriface�*"go.string."%s:%s"�+fmt.Sprintf�+$runtime.panicindex�+$runtime.panicindex�,$runtime.panicindex�.(type.map[string]bool�.�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume.roModes�.4runtime.mapaccess1_faststr�/(type.map[string]bool�/�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume.rwModes�04runtime.mapaccess1_faststr�0(type.map[string]bool�0�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume.rwModes�14runtime.mapaccess1_faststr�4type.string�4runtime.convT2E�52runtime.writebarrieriface�5Rgo.string."bad mount mode specified : %s"�6fmt.Errorf�8� runtime.duffzero�9type.string�:runtime.convT2E�:2runtime.writebarrieriface�;type.string�;runtime.convT2E�<2runtime.writebarrieriface�<type.string�<runtime.convT2E�=2runtime.writebarrieriface�=(go.string."%s:%s:%s"�>fmt.Sprintf�?$runtime.panicindex�?$runtime.panicindex�?$runtime.panicindex�?$runtime.panicindex�?$runtime.panicindexp��"".autotmp_0278"type.interface {}"".autotmp_0277*type.*[1]interface {}"".autotmp_0276&type.[]interface {}"".autotmp_0275type.bool"".autotmp_0274"type.interface {}"".autotmp_0273"type.interface {}"".autotmp_0272"type.interface {}"".autotmp_0270&type.[]interface {}"".autotmp_0269"type.interface {}"".autotmp_0268*type.*[1]interface {}"".autotmp_0267&type.[]interface {}"".autotmp_0266type.bool"".autotmp_0265type.bool"".autotmp_0264"type.interface {}"".autotmp_0263"type.interface {}"".autotmp_0262*type.*[2]interface {}"".autotmp_0261&type.[]interface {}"".autotmp_0260"type.interface {}"".autotmp_0259"type.interface {}"".autotmp_0257&type.[]interface {}"".autotmp_0256type.bool"".autotmp_0255�type.bool"".autotmp_0254type.int"".autotmp_0253"type.interface {}"".autotmp_0252*type.*[1]interface {}"".autotmp_0251&type.[]interface {}"".autotmp_0250�"type.interface {}"".autotmp_0248�&type.[]interface {}"".autotmp_0247type.error"".autotmp_0246type.string"".autotmp_0245(type.[1]interface {}"".autotmp_0244type.int"".autotmp_0243type.string"".autotmp_0242type.string"".autotmp_0241type.string"".autotmp_0240_(type.[3]interface {}"".autotmp_0239type.error"".autotmp_0238type.string"".autotmp_0237(type.[1]interface {}"".autotmp_0236type.bool"".autotmp_0235type.string"".autotmp_0234type.bool"".autotmp_0233type.string"".autotmp_0232type.bool"".autotmp_0231type.string"".autotmp_0230type.string"".autotmp_0229type.string"".autotmp_0228(type.[2]interface {}"".autotmp_0227type.string"".autotmp_0226�type.string"".autotmp_0225type.string"".autotmp_0224�(type.[2]interface {}"".autotmp_0223�type.bool"".autotmp_0222�type.string"".autotmp_0220�type.string"".autotmp_0219�type.bool"".autotmp_0218type.string"".autotmp_0217�type.string"".autotmp_0216type.int"".autotmp_0215type.error"".autotmp_0214type.string"".autotmp_0213(type.[1]interface {}"".autotmp_0210�type.string"".autotmp_0209�(type.[1]interface {}path.path·2�type.string�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume.mode·3�type.string�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume.mode·3�type.string"".splited�type.[]string"".mode�type.string "".containerPath�type.string "".~r3Ptype.error "".~r20type.string("".validateMountMode type.bool "".valtype.stringR%����������6������� ��UP�T(� +8&I�0' +�%��  +(%���%&!���vv�v��v�k����+?��$���v��$Tgclocals·17574d085f5f5b0763ac1aaf01ce4b67Tgclocals·468082ab545c1bae5a803029688d85e2�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�"".ValidateEnv��eH� %H�D$�H;Aw���H���HDŽ$HDŽ$HDŽ$HDŽ$H��$�H�$H��$�H�t$H�5H�l$H��H�H��H�T$ H�L$(H�D$0H��$�H��~@H��$�H��$H��$�H��$HDŽ$HDŽ$H����H�H�$H��$�H��H��$���H�l$H��H��H�H��H��$�H��$��\$����H�T$xH��$�H��$�H�H�CH��$�H���VH��H��H��$�H��$�H��$�H�H�$H�\$xH�\$�H�L$H�D$H��$�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H�\$XH�H�CH�L$hH�L$XH�D$pH�D$`H��$�H��$H��$�H��$H�H�$H�H�\$H�H�\$H�\$XH�\$�H�\$ H��$H�\$(H��$H���É����H�$H�L$�H��$�H��$��\$��u0H��$H��$HDŽ$HDŽ$H����H�T$xH��$�H�$H�L$�H�\$H�\$HH�\$H�\$PH��$�1��H��$�H���7H��H��H��$�H��$�H��$�H�H�$H�\$xH�\$�H�L$H�D$H��$�H�$H�L$8H�L$H�D$@H�D$�H�H�$H�\$HH�\$�H�L$H�D$H��$�H��H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$H��$HDŽ$HDŽ$H���É������ 6 +*0runtime.morestack_noctxt�go.string."="�strings.Split�8"".EnvironmentVariableRegexp�8regexp.(*Regexp).MatchString�type.string�runtime.convT2E�2runtime.writebarrieriface�zgo.string."variable '%s' is not a valid environment variable"�fmt.Sprintf� 2type."".ErrBadEnvVariable� type.error� +Dgo.itab."".ErrBadEnvVariable.error� +runtime.convT2I� "".doesEnvExist� os.Getenv� � runtime.duffzero�type.string�runtime.convT2E�2runtime.writebarrieriface�type.string�runtime.convT2E�2runtime.writebarrieriface�"go.string."%s=%s"�fmt.Sprintf�$runtime.panicindex`�$"".autotmp_0318"type.interface {}"".autotmp_0317"type.interface {}"".autotmp_0315&type.[]interface {}"".autotmp_0314�"type.interface {}"".autotmp_0312�&type.[]interface {}"".autotmp_0311type.string"".autotmp_0310�type.string"".autotmp_0309type.string"".autotmp_0308?(type.[2]interface {}"".autotmp_0307type.bool"".autotmp_0306�2type."".ErrBadEnvVariable"".autotmp_0305�type.string"".autotmp_0304�type.string"".autotmp_0303�(type.[1]interface {} "".arrotype.[]string "".~r2@type.error "".~r1 type.string "".valtype.stringB"�������^������ 0�RH@W�(0� 0~�p�7eu�F Tgclocals·b343c92068d41d468064df311efb05d1Tgclocals·e77b4954b1067dbe825673e24b2082a5�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�("".ValidateIPAddress� �eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$�HDŽ$�HDŽ$�H��$�H�$H��$�H�\$�H�L$H�D$H�L$hH� $H�D$pH�D$�H�D$H�T$H�L$ H��$�H��$�H��H�D$xtMH�$H�T$H�L$�H�L$H�D$ H��$�H��$�HDŽ$�HDŽ$�H�Ĩ�H��$�H�\$HH��$�H�\$PH�\$XH�H�CH�\$XH����H��H��H��$�H��$�H��$�H�H�$H�\$HH�\$�H�L$H�D$H��$�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0HDŽ$�HDŽ$�H��$�H��$�H�ĨÉ���� +*0runtime.morestack_noctxt�"strings.TrimSpace�net.ParseIP�net.IP.String�type.string�runtime.convT2E�2runtime.writebarrieriface�Fgo.string."%s is not an ip address"�fmt.Errorf`�"".autotmp_0333�"type.interface {}"".autotmp_0331/&type.[]interface {}"".autotmp_0329�type.string"".autotmp_0328�(type.[1]interface {}"".autotmp_0327type.string"".autotmp_0325type.string +"".ip_type.net.IP "".~r2@type.error "".~r1 type.string "".valtype.string("���������R_ M�k�pOTgclocals·6d3fa487f5e45db9cb9199d2a5e0e216Tgclocals·0d8996d96a62b65d7a617f192875fe09�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�*"".ValidateMACAddress��eH� %H;aw���H��HH�D$`H�D$hH�D$pH�D$xH�\$PH�$H�\$XH�\$�H�L$H�D$H�L$8H� $H�D$@H�D$�H�D$(H�L$0H��t!H�D$`H�D$hH�D$pH�L$xH��H�H�\$PH�\$`H�\$XH�\$hH�D$pH�D$xH��H� + 0runtime.morestack_noctxt�"strings.TrimSpace�net.ParseMAC`�"".autotmp_0337type.string "".~r2@type.error "".~r1 type.string "".valtype.string ����*� ��>D!7 Q�Tgclocals·13c015770347481bee7a16dde25a3e2fTgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�("".ValidateDNSSearch��eH� %H;aw���H��0H�D$HH�D$PH�D$XH�D$`H�\$8H�$H�t$@H�t$H�5H�l$H��H�H��H�L$ H�D$(H��ucH�L$8H� $H�D$@H�D$H�-L�D$L��H��H�H��H�L$8H�D$@�\$ ��t!H�L$HH�D$PH�D$XH�D$`H��0�H�L$8H� $H�D$@H�D$�H�l$H�T$H�L$ H�D$(H�l$HH�T$PH�L$XH�D$`H��0� + 0runtime.morestack_noctxt�go.string." "�strings.Trim�go.string."."� runtime.eqstring�""".validateDomain``"".autotmp_0339type.string "".~r2@type.error "".~r1 type.string "".valtype.string`�_`D_��>}!T d�Tgclocals·13c015770347481bee7a16dde25a3e2fTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�""".validateDomain��eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$�HDŽ$�HDŽ$�H�H�$H��$�H�\$H��$�H�\$�H��$�H��$�H�\$H�\$ H���H�T$HH�L$PH�\$XH�H�CH�\$XH����H��H��H��$�H��$�H��$�H�H�$H�\$HH�\$�H�L$H�D$H��$�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0HDŽ$�HDŽ$�H��$�H��$�H�ĘÉ����H�$H�L$�H�\$H�l$H��H��H�H�H�H�H�$�H�L$ H�D$(H�T$0H�L$hH�T$xH�D$pH����H��H����H��H�kH���}cH��H��vSH��H�,$H��H��H�H�H��H�\$H��$�H�\$ H��$�HDŽ$�HDŽ$�H�Ę�� H��$�H�\$HH��$�H�\$PH�\$XH�H�CH�\$XH����H��H��H��$�H��$�H��$�H�H�$H�\$HH�\$�H�L$H�D$H��$�H�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0HDŽ$�HDŽ$�H��$�H��$�H�ĘÉ����� ( +*0runtime.morestack_noctxt�"".alphaRegexp�6regexp.(*Regexp).FindString�type.string�runtime.convT2E�2runtime.writebarrieriface�Hgo.string."%s is not a valid domain"�fmt.Errorf�2runtime.stringtoslicebyte�"".domainRegexp�:regexp.(*Regexp).FindSubmatch� 2runtime.slicebytetostring� +$runtime.panicindex� type.string� runtime.convT2E� 2runtime.writebarrieriface� Hgo.string."%s is not a valid domain"�fmt.Errorf�$runtime.panicindex`� "".autotmp_0356"type.interface {}"".autotmp_0355*type.*[1]interface {}"".autotmp_0354&type.[]interface {}"".autotmp_0353�"type.interface {}"".autotmp_0351/&type.[]interface {}"".autotmp_0350type.error"".autotmp_0349type.string"".autotmp_0348(type.[1]interface {}"".autotmp_0347type.int"".autotmp_0346type.int"".autotmp_0343�type.string"".autotmp_0342(type.[1]interface {} +"".ns_type.[][]uint8 "".~r2@type.error "".~r1 type.string "".valtype.string6"�����������(�RN�M-c�&w�pO�pFTgclocals·6d3fa487f5e45db9cb9199d2a5e0e216Tgclocals·e11ef9888c395c84aa28a6aa44bae264�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�("".ValidateExtraHost��eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$�HDŽ$�HDŽ$�H��$�H�$H��$�H�t$H�5H�l$H��H�H�H�D$ �H�T$(H�D$0H�L$8H��$�H��$�H��$�H����H����H�jH����H��H����H��H�,$H��H��H�H��H�D$ H�\$(H�\$HH��H�D$@�1H�\$pH�H�CH�\$pH���H��H��H��$�H��$�H��$�H�H�$H��$�H��$���H��H�\$�H�L$H�D$H��$�H�$H�L$PH�L$H�D$XH�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0HDŽ$�HDŽ$�H��$�H��$�H�İ�� ������H��$�H��$�H��$�H��$�HDŽ$�HDŽ$�H�İ�� H��$�H�\$`H��$�H�\$hH�\$pH�H�CH�\$pH����H��H��H��$�H��$�H��$�H�H�$H�\$`H�\$�H�L$H�D$H��$�H�$H�L$PH�L$H�D$XH�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0HDŽ$�HDŽ$�H��$�H��$�H�İÉ����� $ +*0runtime.morestack_noctxt�go.string.":"�strings.SplitN�("".ValidateIPAddress�type.string�runtime.convT2E�2runtime.writebarrieriface�\go.string."invalid IP address in add-host: %q"�fmt.Errorf�$runtime.panicindex� +$runtime.panicindex� type.string� runtime.convT2E� 2runtime.writebarrieriface� Ngo.string."bad format for add-host: %q"� fmt.Errorf�$runtime.panicindex`�"".autotmp_0376"type.interface {}"".autotmp_0375*type.*[1]interface {}"".autotmp_0374&type.[]interface {}"".autotmp_0373�"type.interface {}"".autotmp_0371_&type.[]interface {}"".autotmp_0370type.error"".autotmp_0369(type.[1]interface {}"".autotmp_0367�type.string"".autotmp_0366(type.[1]interface {} "".err�type.error "".arr/type.[]string "".~r2@type.error "".~r1 type.string "".valtype.string4"����M������,�RY*B�@�,�m�p�pF Tgclocals·4a0bb136639836c86d1f426111a5a477Tgclocals·0a70b462877543c2d66d492afda34c99�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go� "".ValidateLabel��eH� %H;aw���H��HDŽ$�HDŽ$�HDŽ$�HDŽ$�H��$�H�$H��$�H�t$H�5H�l$H��H�H��H��$�H��$�H�\$ H��� H�T$HH�L$PH�\$XH�H�CH�\$XH����H��H��H�\$hH�T$pH�L$xH�H�$H�\$HH�\$�H�L$H�D$H�\$hH�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H�H�\$hH�\$H�\$pH�\$H�\$xH�\$ �H�L$(H�D$0HDŽ$�HDŽ$�H��$�H��$�H�ĀÉ�$���H��$�H��$�HDŽ$�HDŽ$�H�Ā� + 0runtime.morestack_noctxt�go.string."="�strings.Count�type.string�runtime.convT2E�2runtime.writebarrieriface�Hgo.string."bad attribute format: %s"�fmt.Errorf`�"".autotmp_0389�"type.interface {}"".autotmp_0387/&type.[]interface {}"".autotmp_0385otype.string"".autotmp_0384O(type.[1]interface {} "".~r2@type.error "".~r1 type.string "".valtype.string ����6� ��MP�:y�d�Tgclocals·6d3fa487f5e45db9cb9199d2a5e0e216Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�"".ValidateHost��eH� %H;aw���H��PH�D$hH�D$pH�D$xHDŽ$�H�H�$H�H�\$H�H�\$H�H�\$H�\$XH�\$ H�\$`H�\$(�H�l$0H�T$8H�D$@H�L$HH��t&H�\$XH�\$hH�\$`H�\$pH�D$xH��$�H��P�H�l$hH�T$pH�D$xHDŽ$�H��P� + 0runtime.morestack_noctxt�$"".DefaultHTTPHost�$"".DefaultHTTPHost�("".DefaultUnixSocket�("".DefaultUnixSocket��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers.ParseHost`� "".~r2@type.error "".~r1 type.string "".valtype.string ����#���A\&' �lTgclocals·13c015770347481bee7a16dde25a3e2fTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�"".doesEnvExist��eH� %H�D$�H;Aw���H����H�$H�D$H�L$H��$�H��$�H��$�H��$�1�H��$�H�D$@H��$�H��H�l$@H9���H�D$PH���H�H�xH�L$HH�T$hH�|$pH�T$XH�$H�|$`H�|$H�H�|$H��H�H�H�D$ �H��$�H�T$(H�L$0H�D$8H��$�H�T$xH��H��$�vH� +H�BH9�uDH�L$hH� $H�D$pH�D$H��$�H�l$H�t$��\$ ��tƄ$�H����H�D$PH�L$HH��H��H�l$@H9�����Ƅ$�H����� ������ +*0runtime.morestack_noctxtFos.Environ�go.string."="�strings.SplitN� runtime.eqstring�$runtime.panicindex0�"".autotmp_0400type.string"".autotmp_0399�type.string"".autotmp_0398�type.*string"".autotmp_0397�type.int"".autotmp_0396�type.int"".autotmp_0394_type.[]string"".autotmp_0393/type.[]string"".parts�type.[]string"".entry�type.string "".~r1 type.bool"".nametype.string&"����.���*�"�SS "��Tgclocals·d7e8a62d22b1cde6d92b17a55c33fe8fTgclocals·e699116c0f498e16ce59c4e4fa0a75a3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�"".NewUlimitOpt��eH� %H;aw���H��(H�\$01�H9�u\H�H�$�H�\$H�\$ H�H�$H�D$�H�\$H�\$H�\$ H�$H�<$t[�H�\$ H�\$0H�H�$�H�D$H�D$H�$H�<$tH�\$0H�\$�H�\$H�\$8H��(É%�ى%� + 0runtime.morestack_noctxtR�type.map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimitd"runtime.newobject��type.map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�runtime.makemap�.runtime.writebarrierptr�"type."".UlimitOpt�"runtime.newobject�.runtime.writebarrierptr P"".autotmp_0402$type.*"".UlimitOpt"".autotmp_0401�type.*map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit "".~r1$type.*"".UlimitOpt "".ref�type.*map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.UlimitP�OP� \L1#9$/Tgclocals·31b90725c9a885e731df361f51db8f0dTgclocals·f6dcde45bff02c6c4b088b594fd52a4c�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ulimit.go�&"".(*UlimitOpt).Set��eH� %H;aw���H��@H�D$`H�D$hH�\$PH�$H�\$XH�\$�H�D$H�L$H�T$ H�T$8H��H�L$0tH�L$`H�T$hH��@�H�D$(H�H�$H�\$HH�+H�]H�\$H�D$H�|$t&H�\$(H�\$�H�D$`H�D$hH��@É%�� + + 0runtime.morestack_noctxt��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Parse��type.map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�$runtime.mapassign1P� +"".autotmp_0403/�type.*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit "".errtype.error "".~r10type.error "".valtype.string"".o$type.*"".UlimitOpt �W�S� +�*,, = +?k&Tgclocals·61fa3b017c2e156e481b3d912c20f49bTgclocals·61e2515c69061b8fed0e66ece719f936�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ulimit.go�,"".(*UlimitOpt).String� � eH� %H��$`���H;Aw���H�� HDŽ$0HDŽ$8H�D$pH�D$xHDŽ$�H��$(H�+H�MH��$�1��H�H�$H�L$H��$�H�\$�H��$�1�H9��=H��$�H�+H��$�H���IH�,$�H�\$H�\$`H�\$H�\$hH�T$pH�L$xH��$�H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H��$�H��Hk�H�H�$H�\$`H�\$H�\$hH�\$�H��$�H��$�H��$�H�T$pH�L$xH��$�H��$�H�$�H��$�1�H9������H�\$pH��$�H�\$xH��$�H��$�H��$�H�\$PH�H�CH�\$PH����H��H��H��$�H��$�H��$�H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H�L$@H�L$H�D$HH�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$0H��$8H�� É�$�������� +00runtime.morestack_noctxt�� runtime.duffzero��type.map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�&runtime.mapiterinit��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.(*Ulimit).String�type.[]string�"runtime.growslice�4runtime.writebarrierstring�&runtime.mapiternext� type.[]string� runtime.convT2E� +2runtime.writebarrieriface� +go.string."%v"� fmt.Sprintf0�"".autotmp_0419�"type.interface {}"".autotmp_0417�&type.[]interface {}"".autotmp_0412�type.[]string"".autotmp_0410type.string"".autotmp_0409�type.[]string"".autotmp_0408�(type.[1]interface {}"".autotmp_0406�type.string"".autotmp_0405��type.map.iter[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit "".out�type.[]string "".~r0type.string"".o$type.*"".UlimitOpt%�����&@=p�$� +$��W;�p:Tgclocals·893bc98fd3630511d02cf4cf8c0f1f93Tgclocals·e10b17f36388f52b2772c8b1b26a55b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ulimit.go�."".(*UlimitOpt).GetList��eH� %H�D$�H;Aw���H���HDŽ$�HDŽ$�HDŽ$�H�D$HH�D$PH�D$XH��$�H�+H�MH�|$x1��H�H�$H�L$H�\$xH�\$�H�\$x1�H9���H��$�H�+H�\$xH����H�l$@H�T$HH�L$PH�\$XH��H)�H��}FH�H�$H�T$`H�T$H�L$hH�L$H�D$pH�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$hH�D$pH�T$`H��H�$H�\$@H�\$�H�T$`H�L$hH�D$pH�T$HH�L$PH�D$XH�\$xH�$�H�\$x1�H9�����H�\$HH��$�H�\$PH��$�H�\$XH��$�H���É����� +*0runtime.morestack_noctxt�� runtime.duffzero��type.map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�&runtime.mapiterinit��type.[]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�"runtime.growslice�.runtime.writebarrierptr�&runtime.mapiternext@� "".autotmp_0430��type.[]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit"".autotmp_0427��type.map.iter[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit"".v��type.*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit"".ulimits��type.[]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit "".~r0�type.[]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit"".o$type.*"".UlimitOpt"����� RFi�/��;,RTgclocals·afd56e89fe406cd8321967b6f2c293efTgclocals·26d269e519e13fce3b5a9726f3ff5d6d�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ulimit.go�"".init��eH� %H;aw���H��p���t���uH��p�� �����������H�H�,$H��H��H�H��H�D$H�H�$H�D$�H�H�$�H�\$H���?H��H��H�\$XH�T$`H�L$hH�H�$H�H�\$�H�L$H�D$H�\$XH�$H�L$8H�L$H�D$@H�D$�H�H�,$H��H��H�H�H�\$XH�\$H�\$`H�\$H�\$hH�\$ �H�L$(H�D$0H�H�$H�L$HH�L$H�D$PH�D$�H�H�,$H��H��H�H��H�D$H�H�$H�D$�H�H�,$H��H��H�H��H�D$H�H�$H�D$��H��pÉ����N + 0runtime.morestack_noctxt:"".initdone·R"".initdone·p"runtime.throwinit�"".initdone·��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.init��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume.init��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers.init�path.init�net.init�strings.init�regexp.init�os.init�fmt.init�bufio.init�`go.string."^[[:alpha:]_][[:alpha:][:digit:]_]*$"�$regexp.MustCompile�8"".EnvironmentVariableRegexp�.runtime.writebarrierptr�(type.[1]interface {}�"runtime.newobject�type.string�("".DefaultUnixSocket�runtime.convT2E�2runtime.writebarrieriface�*go.string."unix://%s"�fmt.Sprintf�"".DefaultHost�4runtime.writebarrierstring�(go.string."[a-zA-Z]"�$regexp.MustCompile�"".alphaRegexp�.runtime.writebarrierptr�""..gostring.1�$regexp.MustCompile�"".domainRegexp�.runtime.writebarrierptr�"".initdone·� +"".autotmp_0444o"type.interface {}"".autotmp_0442/&type.[]interface {}"".autotmp_0441&type.*regexp.Regexp"".autotmp_0440&type.*regexp.Regexp"".autotmp_0439Otype.string&������w4�h  `w?4�44< O7�d�Tgclocals·0115f8d53b75c1696444f08ad03251d9Tgclocals·da64d0820e77fbc27f563f985efdc21f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ulimit.go�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/hosts_unix.go�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go�(type..hash.[0]string��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�runtime.strhash@` "".autotmp_0450type.int"".autotmp_0449type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*[0]string`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go�$type..eq.[0]string��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$`H����H��Hk�H�H�3H�KH�\$hH��tvH��Hk�H�H�H�CH9�uVH�t$HH�4$H�L$PH�L$H�T$8H�T$H�D$@H�D$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt� runtime.eqstring@�"".autotmp_0454?type.string"".autotmp_0453type.string"".autotmp_0452_type.int"".autotmp_0451Otype.int "".~r30type.bool"".s type.uintptr"".qtype.*[0]string"".ptype.*[0]string&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go�:"".(*ErrBadEnvVariable).Error��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$@H�D$HH�\$81�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�t$8H��t)H�,$H��H�H��H�L$H�D$H�L$@H�D$HH��0É�� + 0runtime.morestack_noctxt� go.string."opts"�:go.string."ErrBadEnvVariable"�"go.string."Error"�"runtime.panicwrap�4"".ErrBadEnvVariable.Error0` "".~r0type.string""..this4type.*"".ErrBadEnvVariable`�_`�� �ATgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�4type..hash.[1]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_0458type.int"".autotmp_0457type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[1]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go�0type..eq.[1]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_0462?"type.interface {}"".autotmp_0461"type.interface {}"".autotmp_0460_type.int"".autotmp_0459Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[1]interface {}"".p*type.*[1]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go�4type..hash.[2]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_0465type.int"".autotmp_0464type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[2]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go�0type..eq.[2]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_0469?"type.interface {}"".autotmp_0468"type.interface {}"".autotmp_0467_type.int"".autotmp_0466Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[2]interface {}"".p*type.*[2]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go�."".(*IpOpt).DefaultMask��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$@H�D$HH�D$PH�|$8H�7H��t2H�<$H�H�H��H�T$H�L$ H�D$(H�T$@H�L$HH�D$PH��0É�� + 0runtime.morestack_noctxt�$net.IP.DefaultMask@` "".~r1type.net.IPMask""..thistype.*"".IpOpt`p_`�� +c-Tgclocals·13d3af77a5bf02af6db4588efb2ea811Tgclocals·3280bececceccd33cb74587feedb1f9f�""".(*IpOpt).Equal��eH� %H;aw���H��8H�Y H��t H�|$@H9;uH�#H�|$@H�7H��t;H�<$H�H�H�H�\$HH�\$H�\$PH�\$ H�\$XH�\$(��\$0�\$`H��8É�� + 0runtime.morestack_noctxt�net.IP.EqualPp "".~r2@type.boolnet.x·3type.net.IP""..thistype.*"".IpOptp^op�� +fTgclocals·14c45952157723c8762210d9c661bf29Tgclocals·3280bececceccd33cb74587feedb1f9f�6"".(*IpOpt).IsGlobalUnicast��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�|$(H�7H��tH�<$H�H�H���\$�\$0H�� É�� + 0runtime.morestack_noctxt�,net.IP.IsGlobalUnicast @ "".~r1type.bool""..thistype.*"".IpOpt@@?@` +` +HTgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�J"".(*IpOpt).IsInterfaceLocalMulticast��eH� %H;aw���H��(H�Y H��t H�|$0H9;uH�#H�H�$�H�D$H�D$ H�$H�|$0H�7H����H�|$H�H�H��H�T$ H�jH��u_H� +H�BL�BH��vR����uFH��<t0H�H�BL�BH��v#H���H����uH���D$8H��(�1���� 1��� ��p��� + 0runtime.morestack_noctxtftype.net.IPx"runtime.newobject�2runtime.writebarrierslice�$runtime.panicindex�$runtime.panicindex P"".&net.ip·2type.*net.IP "".~r1type.bool""..thistype.*"".IpOptP�OP� �;0lTgclocals·2148c3737b2bb476685a1100a2e8343eTgclocals·e1ae6533a9e39048ba0735a2264ce16a�@"".(*IpOpt).IsLinkLocalMulticast��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�|$(H�7H��tH�<$H�H�H���\$�\$0H�� É�� + 0runtime.morestack_noctxt�6net.IP.IsLinkLocalMulticast @ "".~r1type.bool""..thistype.*"".IpOpt@@?@`` +HTgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�<"".(*IpOpt).IsLinkLocalUnicast��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�|$(H�7H��tH�<$H�H�H���\$�\$0H�� É�� + 0runtime.morestack_noctxt�2net.IP.IsLinkLocalUnicast @ "".~r1type.bool""..thistype.*"".IpOpt@@?@`` +HTgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�,"".(*IpOpt).IsLoopback��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�|$(H�7H��tH�<$H�H�H���\$�\$0H�� É�� + 0runtime.morestack_noctxt�"net.IP.IsLoopback @ "".~r1type.bool""..thistype.*"".IpOpt@@?@`` +HTgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�."".(*IpOpt).IsMulticast��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�|$(H�7H��tH�<$H�H�H���\$�\$0H�� É�� + 0runtime.morestack_noctxt�$net.IP.IsMulticast @ "".~r1type.bool""..thistype.*"".IpOpt@@?@`` +HTgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�2"".(*IpOpt).IsUnspecified��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�|$(H�7H��tH�<$H�H�H���\$�\$0H�� É�� + 0runtime.morestack_noctxt�(net.IP.IsUnspecified @ "".~r1type.bool""..thistype.*"".IpOpt@@?@`` +HTgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�."".(*IpOpt).MarshalText��eH� %H;aw���H��@H�Y H��t H�|$HH9;uH�#H�D$PH�D$XH�D$`H�D$hH�D$pH�|$HH�7H��tFH�<$H�H�H��H�t$H�l$ H�T$(H�L$0H�D$8H�t$PH�l$XH�T$`H�L$hH�D$pH��@É� + 0runtime.morestack_noctxt�$net.IP.MarshalText`� "".~r2@type.error "".~r1type.[]uint8""..thistype.*"".IpOpt����� +uKTgclocals·13c015770347481bee7a16dde25a3e2fTgclocals·3280bececceccd33cb74587feedb1f9f� "".(*IpOpt).Mask��eH� %H;aw���H��HH�Y H��t H�|$PH9;uH�#H�D$pH�D$xHDŽ$�H�|$PH�7H��tSH�<$H�H�H�H�\$XH�\$H�\$`H�\$ H�\$hH�\$(�H�T$0H�L$8H�D$@H�T$pH�L$xH��$�H��HÉ� + 0runtime.morestack_noctxt�net.IP.Maskp� "".~r2@type.net.IPnet.mask·3type.net.IPMask""..thistype.*"".IpOpt������ �<Tgclocals·9877a4ef732a0f966b889793f9b99b87Tgclocals·3280bececceccd33cb74587feedb1f9f� "".(*IpOpt).To16��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$@H�D$HH�D$PH�|$8H�7H��t2H�<$H�H�H��H�T$H�L$ H�D$(H�T$@H�L$HH�D$PH��0É�� + 0runtime.morestack_noctxt�net.IP.To16@` "".~r1type.net.IP""..thistype.*"".IpOpt`p_`�� +c-Tgclocals·13d3af77a5bf02af6db4588efb2ea811Tgclocals·3280bececceccd33cb74587feedb1f9f�"".(*IpOpt).To4��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$@H�D$HH�D$PH�|$8H�7H��t2H�<$H�H�H��H�T$H�L$ H�D$(H�T$@H�L$HH�D$PH��0É�� + 0runtime.morestack_noctxt�net.IP.To4@` "".~r1type.net.IP""..thistype.*"".IpOpt`p_`�� +c-Tgclocals·13d3af77a5bf02af6db4588efb2ea811Tgclocals·3280bececceccd33cb74587feedb1f9f�2"".(*IpOpt).UnmarshalText`HH�D$(H�D$0H�\$H�+H�l$�@.net.(*IP).UnmarshalText` "".~r2@type.errornet.text·3type.[]uint8""..thistype.*"".IpOpt00 0Tgclocals·9f0d5ba6770c4a1ed4fa771547e96df1Tgclocals·3280bececceccd33cb74587feedb1f9f�("".IpOpt.DefaultMask��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$@H�D$HH�D$PH�t$8H��t5H�,$H��H�H�H��H�T$H�L$ H�D$(H�T$@H�L$HH�D$PH��0É�� + 0runtime.morestack_noctxt�$net.IP.DefaultMask@` "".~r1type.net.IPMask""..thistype."".IpOpt`p_`�"� +c-Tgclocals·13d3af77a5bf02af6db4588efb2ea811Tgclocals·3280bececceccd33cb74587feedb1f9f�"".IpOpt.Equal��eH� %H;aw���H��8H�Y H��t H�|$@H9;uH�#H�t$@H��t>H�,$H��H�H�H�H�\$HH�\$H�\$PH�\$ H�\$XH�\$(��\$0�\$`H��8É� + 0runtime.morestack_noctxt�net.IP.EqualPp "".~r2@type.boolnet.x·3type.net.IP""..thistype."".IpOptp^op�$� +fTgclocals·14c45952157723c8762210d9c661bf29Tgclocals·3280bececceccd33cb74587feedb1f9f�0"".IpOpt.IsGlobalUnicast��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�t$(H��t H�,$H��H�H�H���\$�\$0H�� É�� + 0runtime.morestack_noctxt�,net.IP.IsGlobalUnicast @ "".~r1type.bool""..thistype."".IpOpt@@?@`&` +HTgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�D"".IpOpt.IsInterfaceLocalMulticast��eH� %H;aw���H��(H�Y H��t H�|$0H9;uH�#H�H�$�H�D$H�D$ H�$H�t$0H����H�l$H��H�H�H��H�T$ H�jH��u_H� +H�BL�BH��vR����uFH��<t0H�H�BL�BH��v#H���H����uH���D$8H��(�1���� 1��� ��m��� + 0runtime.morestack_noctxtftype.net.IPx"runtime.newobject�2runtime.writebarrierslice�$runtime.panicindex�$runtime.panicindex P"".&net.ip·2type.*net.IP "".~r1type.bool""..thistype."".IpOptP�OP�(�;0lTgclocals·2148c3737b2bb476685a1100a2e8343eTgclocals·e1ae6533a9e39048ba0735a2264ce16a�:"".IpOpt.IsLinkLocalMulticast��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�t$(H��t H�,$H��H�H�H���\$�\$0H�� É�� + 0runtime.morestack_noctxt�6net.IP.IsLinkLocalMulticast @ "".~r1type.bool""..thistype."".IpOpt@@?@`*` +HTgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�6"".IpOpt.IsLinkLocalUnicast��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�t$(H��t H�,$H��H�H�H���\$�\$0H�� É�� + 0runtime.morestack_noctxt�2net.IP.IsLinkLocalUnicast @ "".~r1type.bool""..thistype."".IpOpt@@?@`,` +HTgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�&"".IpOpt.IsLoopback��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�t$(H��t H�,$H��H�H�H���\$�\$0H�� É�� + 0runtime.morestack_noctxt�"net.IP.IsLoopback @ "".~r1type.bool""..thistype."".IpOpt@@?@`.` +HTgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�("".IpOpt.IsMulticast��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�t$(H��t H�,$H��H�H�H���\$�\$0H�� É�� + 0runtime.morestack_noctxt�$net.IP.IsMulticast @ "".~r1type.bool""..thistype."".IpOpt@@?@`0` +HTgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�,"".IpOpt.IsUnspecified��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�t$(H��t H�,$H��H�H�H���\$�\$0H�� É�� + 0runtime.morestack_noctxt�(net.IP.IsUnspecified @ "".~r1type.bool""..thistype."".IpOpt@@?@`2` +HTgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�("".IpOpt.MarshalText��eH� %H;aw���H��@H�Y H��t H�|$HH9;uH�#H�D$PH�D$XH�D$`H�D$hH�D$pH�t$HH��tIH�,$H��H�H�H��H�t$H�l$ H�T$(H�L$0H�D$8H�t$PH�l$XH�T$`H�L$hH�D$pH��@É� + 0runtime.morestack_noctxt�$net.IP.MarshalText`� "".~r2@type.error "".~r1type.[]uint8""..thistype."".IpOpt����4� +uKTgclocals·13c015770347481bee7a16dde25a3e2fTgclocals·3280bececceccd33cb74587feedb1f9f�"".IpOpt.Mask��eH� %H;aw���H��HH�Y H��t H�|$PH9;uH�#H�D$pH�D$xHDŽ$�H�t$PH��tVH�,$H��H�H�H�H�\$XH�\$H�\$`H�\$ H�\$hH�\$(�H�T$0H�L$8H�D$@H�T$pH�L$xH��$�H��HÉ� + 0runtime.morestack_noctxt�net.IP.Maskp� "".~r2@type.net.IPnet.mask·3type.net.IPMask""..thistype."".IpOpt�����6� �<Tgclocals·9877a4ef732a0f966b889793f9b99b87Tgclocals·3280bececceccd33cb74587feedb1f9f�"".IpOpt.To16��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$@H�D$HH�D$PH�t$8H��t5H�,$H��H�H�H��H�T$H�L$ H�D$(H�T$@H�L$HH�D$PH��0É�� + 0runtime.morestack_noctxt�net.IP.To16@` "".~r1type.net.IP""..thistype."".IpOpt`p_`�8� +c-Tgclocals·13d3af77a5bf02af6db4588efb2ea811Tgclocals·3280bececceccd33cb74587feedb1f9f�"".IpOpt.To4��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$@H�D$HH�D$PH�t$8H��t5H�,$H��H�H�H��H�T$H�L$ H�D$(H�T$@H�L$HH�D$PH��0É�� + 0runtime.morestack_noctxt�net.IP.To4@` "".~r1type.net.IP""..thistype."".IpOpt`p_`�:� +c-Tgclocals·13d3af77a5bf02af6db4588efb2ea811Tgclocals·3280bececceccd33cb74587feedb1f9f�,"".IpOpt.UnmarshalText��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�D$XH�D$`H�\$8H�$H�\$@H�\$H�\$HH�\$H�\$PH�\$�H�L$ H�D$(H�L$XH�D$`H��0� + 0runtime.morestack_noctxt�.net.(*IP).UnmarshalText`` "".~r2@type.errornet.text·3type.[]uint8""..thistype."".IpOpt`l_ +�<� +i'Tgclocals·9f0d5ba6770c4a1ed4fa771547e96df1Tgclocals·3280bececceccd33cb74587feedb1f9f�(type..hash.[8]string��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�runtime.strhash@` "".autotmp_0511type.int"".autotmp_0510type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*[8]string`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go�$type..eq.[8]string��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$`H����H��Hk�H�H�3H�KH�\$hH��tvH��Hk�H�H�H�CH9�uVH�t$HH�4$H�L$PH�L$H�T$8H�T$H�D$@H�D$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt� runtime.eqstring@�"".autotmp_0515?type.string"".autotmp_0514type.string"".autotmp_0513_type.int"".autotmp_0512Otype.int "".~r30type.bool"".s type.uintptr"".qtype.*[8]string"".ptype.*[8]string&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go�(type..hash.[3]string��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�runtime.strhash@` "".autotmp_0518type.int"".autotmp_0517type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*[3]string`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go�$type..eq.[3]string��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$`H����H��Hk�H�H�3H�KH�\$hH��tvH��Hk�H�H�H�CH9�uVH�t$HH�4$H�L$PH�L$H�T$8H�T$H�D$@H�D$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt� runtime.eqstring@�"".autotmp_0522?type.string"".autotmp_0521type.string"".autotmp_0520_type.int"".autotmp_0519Otype.int "".~r30type.bool"".s type.uintptr"".qtype.*[3]string"".ptype.*[3]string&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go�4type..hash.[3]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_0525type.int"".autotmp_0524type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[3]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go�0type..eq.[3]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_0529?"type.interface {}"".autotmp_0528"type.interface {}"".autotmp_0527_type.int"".autotmp_0526Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[3]interface {}"".p*type.*[3]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go�,4go.itab.*os.File.io.Reader�,Dgo.itab."".ErrBadEnvVariable.error�go.string."#"0$# go.string."#"�go.string."="0$= go.string."="�zgo.string."variable '%s' is not a valid environment variable"��1variable '%s' is not a valid environment variable zgo.string."variable '%s' is not a valid environment variable"�"go.string."%s=%s"0,%s=%s "go.string."%s=%s"�Tgclocals·ad819538c58aa8f64fd2a144ce4dfed6��v� ,,�������������?�?��?�?�? �?��?�?�?�?�Tgclocals·231e82aa2fc136f1b81a915e25ec3cfa��",","",�Xgo.string."poorly formatted environment: %s"pb poorly formatted environment: %s Xgo.string."poorly formatted environment: %s"�Tgclocals·403a8d79fd24b295e8557f6970497aa3((���Tgclocals·363b18caf0020ca418fd378dbb75c855((�Tgclocals·f6dcde45bff02c6c4b088b594fd52a4c((�Tgclocals·b29a376724b9675f7c9e576a6dabc1e0(( + + +�Fgo.string."%s is not an ip address"PP%s is not an ip address Fgo.string."%s is not an ip address"�Tgclocals·e11ef9888c395c84aa28a6aa44bae264((  �Tgclocals·149f5bf45741ad4d84849674a456615e(( + + + +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·9307bf1379da22b408b9b243276c0115((���Tgclocals·6d340c3bdac448a6ef1256f331f68dd3((�Tgclocals·e1ae6533a9e39048ba0735a2264ce16a �Tgclocals·bd51743682bd6c0f7b9f2e8e6dffed99  + +�go.string."%v"0&%v go.string."%v"�Tgclocals·7876b70d8da64fa07ca2fd3ecc71f905((�����Tgclocals·6d340c3bdac448a6ef1256f331f68dd3((�Tgclocals·1509598f597bd125bdfd9d44972821c7 �Tgclocals·61fa3b017c2e156e481b3d912c20f49b  + + +�Tgclocals·3548d93f69958cd35a7e42f3f32eff97PP"���Tgclocals·2018557e3ee0abccf2865b16663e690b00 + + + +�Tgclocals·895437610367c247af3cb64952bed446  + �Tgclocals·2148c3737b2bb476685a1100a2e8343e �Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·13d3af77a5bf02af6db4588efb2ea811�Tgclocals·4398bb51467914f29637b614067b995f �Tgclocals·9ff42bf311af152488d11f0f78c8d5ce  + +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·06cab038d51064a089bda21fa03e00f7�Tgclocals·729deb178891d0cb0d4c5b2058f91105 �Tgclocals·61fa3b017c2e156e481b3d912c20f49b  + + +�Tgclocals·403a8d79fd24b295e8557f6970497aa3((���Tgclocals·6d340c3bdac448a6ef1256f331f68dd3((�Tgclocals·e1ae6533a9e39048ba0735a2264ce16a �Tgclocals·bd51743682bd6c0f7b9f2e8e6dffed99  + +�"go.string."stdin"0,stdin "go.string."stdin"�$go.string."stdout"0.stdout $go.string."stdout"�$go.string."stderr"0.stderr $go.string."stderr"�lgo.string."valid streams are STDIN, STDOUT and STDERR"�v*valid streams are STDIN, STDOUT and STDERR lgo.string."valid streams are STDIN, STDOUT and STDERR"�Tgclocals·5de38da5eeb0729bf417a80c29b78c42(( � " "�Tgclocals·6d3fa487f5e45db9cb9199d2a5e0e216(( �Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·13c015770347481bee7a16dde25a3e2f �Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·13c015770347481bee7a16dde25a3e2f �Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·13c015770347481bee7a16dde25a3e2f �go.string.":"0$: go.string.":"�Lgo.string."bad format for volumes: %s"`Vbad format for volumes: %s Lgo.string."bad format for volumes: %s"�"go.string."%s:%s"0,%s:%s "go.string."%s:%s"�Rgo.string."bad mount mode specified : %s"`\bad mount mode specified : %s Rgo.string."bad mount mode specified : %s"�(go.string."%s:%s:%s"@2%s:%s:%s (go.string."%s:%s:%s"�Lgo.string."%s is not an absolute path"`V%s is not an absolute path Lgo.string."%s is not an absolute path"�Tgclocals·468082ab545c1bae5a803029688d85e2��P��  �������" ��������Tgclocals·17574d085f5f5b0763ac1aaf01ce4b67���Tgclocals·e77b4954b1067dbe825673e24b2082a5��,��� � ��Tgclocals·b343c92068d41d468064df311efb05d1HH "�Tgclocals·0d8996d96a62b65d7a617f192875fe09(( � �Tgclocals·6d3fa487f5e45db9cb9199d2a5e0e216(( �Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·13c015770347481bee7a16dde25a3e2f �go.string." "0$  go.string." "�go.string."."0$. go.string."."�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·13c015770347481bee7a16dde25a3e2f �Hgo.string."%s is not a valid domain"`R%s is not a valid domain Hgo.string."%s is not a valid domain"�Tgclocals·e11ef9888c395c84aa28a6aa44bae264((  �Tgclocals·6d3fa487f5e45db9cb9199d2a5e0e216(( �Ngo.string."bad format for add-host: %q"`Xbad format for add-host: %q Ngo.string."bad format for add-host: %q"�\go.string."invalid IP address in add-host: %q"pf"invalid IP address in add-host: %q \go.string."invalid IP address in add-host: %q"�Tgclocals·0a70b462877543c2d66d492afda34c99@@������Tgclocals·4a0bb136639836c86d1f426111a5a477@@ �Hgo.string."bad attribute format: %s"`Rbad attribute format: %s Hgo.string."bad attribute format: %s"�Tgclocals·1eb9d8ec9969f1d922533aa863dff6f6(( / �Tgclocals·6d3fa487f5e45db9cb9199d2a5e0e216(( �Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·13c015770347481bee7a16dde25a3e2f �Tgclocals·e699116c0f498e16ce59c4e4fa0a75a3 �Tgclocals·d7e8a62d22b1cde6d92b17a55c33fe8f �Tgclocals·f6dcde45bff02c6c4b088b594fd52a4c((�Tgclocals·31b90725c9a885e731df361f51db8f0d((�Tgclocals·61e2515c69061b8fed0e66ece719f936 �Tgclocals·61fa3b017c2e156e481b3d912c20f49b  + + +�Tgclocals·e10b17f36388f52b2772c8b1b26a55b3pp8 ��U��U��U�����Tgclocals·893bc98fd3630511d02cf4cf8c0f1f93@@�Tgclocals·26d269e519e13fce3b5a9726f3ff5d6dPP"��V��V��V�Tgclocals·afd56e89fe406cd8321967b6f2c293ef00�go.string." \t"0&  go.string." \t"�@go.string."/var/run/docker.sock"PJ/var/run/docker.sock @go.string."/var/run/docker.sock"�*go.string."127.0.0.1"@4 127.0.0.1 *go.string."127.0.0.1"�`go.string."^[[:alpha:]_][[:alpha:][:digit:]_]*$"pj$^[[:alpha:]_][[:alpha:][:digit:]_]*$ `go.string."^[[:alpha:]_][[:alpha:][:digit:]_]*$"�*go.string."unix://%s"@4 unix://%s *go.string."unix://%s"�(go.string."[a-zA-Z]"@2[a-zA-Z] (go.string."[a-zA-Z]"�""..gostring.1���^(:?(:?[a-zA-Z0-9]|(:?[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9]))(:?\.(:?[a-zA-Z0-9]|(:?[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9])))*)\.?\s*$ ""..gostring.1�Tgclocals·da64d0820e77fbc27f563f985efdc21f �Tgclocals·0115f8d53b75c1696444f08ad03251d9�*8"".EnvironmentVariableRegexp&type.*regexp.Regexp�*"".whiteSpaces type.string  go.string." \t"�*"".DefaultHost type.string�*"".alphaRegexp&type.*regexp.Regexp�*"".domainRegexp&type.*regexp.Regexp�*$"".DefaultHTTPHost type.string  *go.string."127.0.0.1"�,$"".DefaultHTTPPorttype.intG �*("".DefaultUnixSocket type.string  @go.string."/var/run/docker.sock"�""".statictmp_0194`type.[3]string` "go.string."stdin"  $go.string."stdout"@ $go.string."stderr"�,"".initdone·type.uint8�$"".ParseEnvFile·f"".ParseEnvFile�os.Open·fos.Open�(runtime.newobject·f"runtime.newobject�,runtime.deferreturn·f&runtime.deferreturn�&os.(*File).Close·f os.(*File).Close�(runtime.deferproc·f"runtime.deferproc�&runtime.typ2Itab·f runtime.typ2Itab�(runtime.makeslice·f"runtime.makeslice�8runtime.writebarrieriface·f2runtime.writebarrieriface�$bufio.ScanLines·fbufio.ScanLines�4runtime.writebarrierptr·f.runtime.writebarrierptr�8runtime.writebarrierslice·f2runtime.writebarrierslice�0bufio.(*Scanner).Scan·f*bufio.(*Scanner).Scan�8runtime.slicebytetostring·f2runtime.slicebytetostring�*runtime.panicslice·f$runtime.panicslice�&runtime.eqstring·f runtime.eqstring�"strings.SplitN·fstrings.SplitN�*runtime.panicindex·f$runtime.panicindex�&strings.TrimLeft·f strings.TrimLeft�>regexp.(*Regexp).MatchString·f8regexp.(*Regexp).MatchString�$runtime.convT2E·fruntime.convT2E�fmt.Sprintf·ffmt.Sprintf�$runtime.convT2I·fruntime.convT2I�(runtime.growslice·f"runtime.growslice�:runtime.writebarrierstring·f4runtime.writebarrierstring�(strings.TrimSpace·f"strings.TrimSpace�os.Getenv·fos.Getenv�$runtime.ifaceeq·fruntime.ifaceeq�,runtime.throwreturn·f&runtime.throwreturn�:"".ErrBadEnvVariable.Error·f4"".ErrBadEnvVariable.Error�"".NewIpOpt·f"".NewIpOpt�$"".(*IpOpt).Set·f"".(*IpOpt).Set�net.ParseIP·fnet.ParseIP�fmt.Errorf·ffmt.Errorf�*"".(*IpOpt).String·f$"".(*IpOpt).String� net.IP.String·fnet.IP.String�""".NewListOpts·f"".NewListOpts�("".NewListOptsRef·f""".NewListOptsRef�0"".(*ListOpts).String·f*"".(*ListOpts).String�*"".(*ListOpts).Set·f$"".(*ListOpts).Set�0"".(*ListOpts).Delete·f*"".(*ListOpts).Delete�$runtime.memmove·fruntime.memmove�0"".(*ListOpts).GetMap·f*"".(*ListOpts).GetMap�$runtime.makemap·fruntime.makemap�*runtime.mapassign1·f$runtime.mapassign1�0"".(*ListOpts).GetAll·f*"".(*ListOpts).GetAll�*"".(*ListOpts).Get·f$"".(*ListOpts).Get�*"".(*ListOpts).Len·f$"".(*ListOpts).Len�("".(*MapOpts).Set·f""".(*MapOpts).Set�."".(*MapOpts).String·f("".(*MapOpts).String� "".NewMapOpts·f"".NewMapOpts�("".ValidateAttach·f""".ValidateAttach�$strings.ToLower·fstrings.ToLower�$"".ValidateLink·f"".ValidateLink��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers.ParseLink·f�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers.ParseLink�("".ValidateDevice·f""".ValidateDevice�$"".validatePath·f"".validatePath�$"".ValidatePath·f"".ValidatePath� strings.Count·fstrings.Count�path.Clean·fpath.Clean�:runtime.mapaccess1_faststr·f4runtime.mapaccess1_faststr�""".ValidateEnv·f"".ValidateEnv� strings.Split·fstrings.Split�$"".doesEnvExist·f"".doesEnvExist�."".ValidateIPAddress·f("".ValidateIPAddress�0"".ValidateMACAddress·f*"".ValidateMACAddress�net.ParseMAC·fnet.ParseMAC�."".ValidateDNSSearch·f("".ValidateDNSSearch�strings.Trim·fstrings.Trim�("".validateDomain·f""".validateDomain�go.typelink.[0]string/[0]stringtype.[0]string�&go.string."[]uint8"00[]uint8 &go.string."[]uint8"�type.[]uint8���~.8 � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P&go.string."[]uint8"p*go.weak.type.*[]uint8�"runtime.zerovalue�type.uint8�6go.typelink.[]uint8/[]uint8type.[]uint8�bruntime.gcbits.0x88000000000000000000000000000000 ��Fgo.string."*opts.ErrBadEnvVariable"PP*opts.ErrBadEnvVariable Fgo.string."*opts.ErrBadEnvVariable"� go.string."opts"0*opts go.string."opts"�:go.string."ErrBadEnvVariable"PDErrBadEnvVariable :go.string."ErrBadEnvVariable"�"go.string."Error"0,Error "go.string."Error"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�`go.string."func(*opts.ErrBadEnvVariable) string"pj$func(*opts.ErrBadEnvVariable) string `go.string."func(*opts.ErrBadEnvVariable) string"�Ntype.func(*"".ErrBadEnvVariable) string�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P`go.string."func(*opts.ErrBadEnvVariable) string"p`go.weak.type.*func(*"".ErrBadEnvVariable) string�"runtime.zerovalue��Ntype.func(*"".ErrBadEnvVariable) string��Ntype.func(*"".ErrBadEnvVariable) string�4type.*"".ErrBadEnvVariable�type.string�2go.string."func() string"@< func() string 2go.string."func() string"�$type.func() string���m�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P2go.string."func() string"p6go.weak.type.*func() string�"runtime.zerovalue��$type.func() string��$type.func() string�type.string�4type.*"".ErrBadEnvVariable���kv�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PFgo.string."*opts.ErrBadEnvVariable"pFgo.weak.type.**"".ErrBadEnvVariable�"runtime.zerovalue�2type."".ErrBadEnvVariable`�4type.*"".ErrBadEnvVariable��4type.*"".ErrBadEnvVariable�"go.string."Error"�$type.func() string�Ntype.func(*"".ErrBadEnvVariable) string�:"".(*ErrBadEnvVariable).Error�:"".(*ErrBadEnvVariable).Error�bruntime.gcbits.0x48000000000000000000000000000000 H�Dgo.string."opts.ErrBadEnvVariable"PNopts.ErrBadEnvVariable Dgo.string."opts.ErrBadEnvVariable"�go.string."msg"0(msg go.string."msg"�^go.string."func(opts.ErrBadEnvVariable) string"ph#func(opts.ErrBadEnvVariable) string ^go.string."func(opts.ErrBadEnvVariable) string"�Ltype.func("".ErrBadEnvVariable) string��㾀�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P^go.string."func(opts.ErrBadEnvVariable) string"p^go.weak.type.*func("".ErrBadEnvVariable) string�"runtime.zerovalue��Ltype.func("".ErrBadEnvVariable) string��Ltype.func("".ErrBadEnvVariable) string�2type."".ErrBadEnvVariable�type.string�2type."".ErrBadEnvVariable��T �U$ � runtime.algarray0bruntime.gcbits.0x48000000000000000000000000000000PDgo.string."opts.ErrBadEnvVariable"p4type.*"".ErrBadEnvVariable�"runtime.zerovalue��2type."".ErrBadEnvVariable�go.string."msg"�"go.importpath."".�type.string`�2type."".ErrBadEnvVariable�:go.string."ErrBadEnvVariable"�"go.importpath."".��2type."".ErrBadEnvVariable�"go.string."Error"�$type.func() string�Ltype.func("".ErrBadEnvVariable) string�:"".(*ErrBadEnvVariable).Error�4"".ErrBadEnvVariable.Error�bruntime.gcbits.0xcc000000000000000000000000000000 ��0go.string."interface {}"@: interface {} 0go.string."interface {}"�"type.interface {}���W� � runtime.algarray0bruntime.gcbits.0xcc000000000000000000000000000000P0go.string."interface {}"p4go.weak.type.*interface {}�"runtime.zerovalue��"type.interface {}�4go.string."[]interface {}"@>[]interface {} 4go.string."[]interface {}"�&type.[]interface {}��p��/ � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P4go.string."[]interface {}"p8go.weak.type.*[]interface {}�"runtime.zerovalue�"type.interface {}�Rgo.typelink.[]interface {}/[]interface {}&type.[]interface {}�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�type.func(*"".IpOpt) net.IPMask��xH��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."func(*opts.IpOpt) net.IPMask"pPgo.weak.type.*func(*"".IpOpt) net.IPMask�"runtime.zerovalue��>type.func(*"".IpOpt) net.IPMask��>type.func(*"".IpOpt) net.IPMask�type.*"".IpOpt�type.net.IPMask�Tgo.string."func(*opts.IpOpt, net.IP) bool"`^func(*opts.IpOpt, net.IP) bool Tgo.string."func(*opts.IpOpt, net.IP) bool"�Btype.func(*"".IpOpt, net.IP) bool��&�*3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PTgo.string."func(*opts.IpOpt, net.IP) bool"pTgo.weak.type.*func(*"".IpOpt, net.IP) bool�"runtime.zerovalue��Btype.func(*"".IpOpt, net.IP) bool��Btype.func(*"".IpOpt, net.IP) bool�type.*"".IpOpt�type.net.IP�type.bool�Dgo.string."func(*opts.IpOpt) bool"PNfunc(*opts.IpOpt) bool Dgo.string."func(*opts.IpOpt) bool"�2type.func(*"".IpOpt) bool�����^3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PDgo.string."func(*opts.IpOpt) bool"pDgo.weak.type.*func(*"".IpOpt) bool�"runtime.zerovalue��2type.func(*"".IpOpt) bool��2type.func(*"".IpOpt) bool�type.*"".IpOpt�type.bool�\go.string."func(*opts.IpOpt) ([]uint8, error)"pf"func(*opts.IpOpt) ([]uint8, error) \go.string."func(*opts.IpOpt) ([]uint8, error)"�Jtype.func(*"".IpOpt) ([]uint8, error)��Bu3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P\go.string."func(*opts.IpOpt) ([]uint8, error)"p\go.weak.type.*func(*"".IpOpt) ([]uint8, error)�"runtime.zerovalue��Jtype.func(*"".IpOpt) ([]uint8, error)��Jtype.func(*"".IpOpt) ([]uint8, error)�type.*"".IpOpt�type.[]uint8�type.error�`go.string."func(*opts.IpOpt, net.IPMask) net.IP"pj$func(*opts.IpOpt, net.IPMask) net.IP `go.string."func(*opts.IpOpt, net.IPMask) net.IP"�Ntype.func(*"".IpOpt, net.IPMask) net.IP��� i�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P`go.string."func(*opts.IpOpt, net.IPMask) net.IP"p`go.weak.type.*func(*"".IpOpt, net.IPMask) net.IP�"runtime.zerovalue��Ntype.func(*"".IpOpt, net.IPMask) net.IP��Ntype.func(*"".IpOpt, net.IPMask) net.IP�type.*"".IpOpt�type.net.IPMask�type.net.IP�Vgo.string."func(*opts.IpOpt, string) error"``func(*opts.IpOpt, string) error Vgo.string."func(*opts.IpOpt, string) error"�Dtype.func(*"".IpOpt, string) error�� c� 3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PVgo.string."func(*opts.IpOpt, string) error"pVgo.weak.type.*func(*"".IpOpt, string) error�"runtime.zerovalue��Dtype.func(*"".IpOpt, string) error��Dtype.func(*"".IpOpt, string) error�type.*"".IpOpt�type.string�type.error�Hgo.string."func(*opts.IpOpt) string"`Rfunc(*opts.IpOpt) string Hgo.string."func(*opts.IpOpt) string"�6type.func(*"".IpOpt) string��uX�!3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PHgo.string."func(*opts.IpOpt) string"pHgo.weak.type.*func(*"".IpOpt) string�"runtime.zerovalue��6type.func(*"".IpOpt) string��6type.func(*"".IpOpt) string�type.*"".IpOpt�type.string�Hgo.string."func(*opts.IpOpt) net.IP"`Rfunc(*opts.IpOpt) net.IP Hgo.string."func(*opts.IpOpt) net.IP"�6type.func(*"".IpOpt) net.IP�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PHgo.string."func(*opts.IpOpt) net.IP"pHgo.weak.type.*func(*"".IpOpt) net.IP�"runtime.zerovalue��6type.func(*"".IpOpt) net.IP��6type.func(*"".IpOpt) net.IP�type.*"".IpOpt�type.net.IP�Xgo.string."func(*opts.IpOpt, []uint8) error"pb func(*opts.IpOpt, []uint8) error Xgo.string."func(*opts.IpOpt, []uint8) error"�Ftype.func(*"".IpOpt, []uint8) error��O�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PXgo.string."func(*opts.IpOpt, []uint8) error"pXgo.weak.type.*func(*"".IpOpt, []uint8) error�"runtime.zerovalue��Ftype.func(*"".IpOpt, []uint8) error��Ftype.func(*"".IpOpt, []uint8) error�type.*"".IpOpt�type.[]uint8�type.error�.go.string."DefaultMask"@8 DefaultMask .go.string."DefaultMask"�:go.string."func() net.IPMask"PDfunc() net.IPMask :go.string."func() net.IPMask"�,type.func() net.IPMask��'a�'3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."func() net.IPMask"p>go.weak.type.*func() net.IPMask�"runtime.zerovalue��,type.func() net.IPMask��,type.func() net.IPMask�type.net.IPMask�"go.string."Equal"0,Equal "go.string."Equal"�:go.string."func(net.IP) bool"PDfunc(net.IP) bool :go.string."func(net.IP) bool"�,type.func(net.IP) bool���� �3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."func(net.IP) bool"p>go.weak.type.*func(net.IP) bool�"runtime.zerovalue��,type.func(net.IP) bool��,type.func(net.IP) bool�type.net.IP�type.bool�6go.string."IsGlobalUnicast"@@IsGlobalUnicast 6go.string."IsGlobalUnicast"�.go.string."func() bool"@8 func() bool .go.string."func() bool"� type.func() bool��T�x3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P.go.string."func() bool"p2go.weak.type.*func() bool�"runtime.zerovalue�� type.func() bool�� type.func() bool�type.bool�Jgo.string."IsInterfaceLocalMulticast"`TIsInterfaceLocalMulticast Jgo.string."IsInterfaceLocalMulticast"�@go.string."IsLinkLocalMulticast"PJIsLinkLocalMulticast @go.string."IsLinkLocalMulticast"�go.string."func([]uint8) error"PHfunc([]uint8) error >go.string."func([]uint8) error"�0type.func([]uint8) error��_�[:3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."func([]uint8) error"pBgo.weak.type.*func([]uint8) error�"runtime.zerovalue��0type.func([]uint8) error��0type.func([]uint8) error�type.[]uint8�type.error�type.*"".IpOpt� � ;�g6� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P.go.string."*opts.IpOpt"p.go.weak.type.**"".IpOpt�"runtime.zerovalue�type."".IpOpt`�type.*"".IpOpt��type.*"".IpOpt�.go.string."DefaultMask"�,type.func() net.IPMask�>type.func(*"".IpOpt) net.IPMask�."".(*IpOpt).DefaultMask�."".(*IpOpt).DefaultMask�"go.string."Equal"�,type.func(net.IP) bool�Btype.func(*"".IpOpt, net.IP) bool�""".(*IpOpt).Equal�""".(*IpOpt).Equal�6go.string."IsGlobalUnicast"� type.func() bool�2type.func(*"".IpOpt) bool�6"".(*IpOpt).IsGlobalUnicast�6"".(*IpOpt).IsGlobalUnicast�Jgo.string."IsInterfaceLocalMulticast"� type.func() bool�2type.func(*"".IpOpt) bool�J"".(*IpOpt).IsInterfaceLocalMulticast�J"".(*IpOpt).IsInterfaceLocalMulticast�@go.string."IsLinkLocalMulticast"� type.func() bool�2type.func(*"".IpOpt) bool�@"".(*IpOpt).IsLinkLocalMulticast�@"".(*IpOpt).IsLinkLocalMulticast�*opts.ListOpts 4go.string."*opts.ListOpts"�Pgo.string."func(*opts.ListOpts, string)"`Zfunc(*opts.ListOpts, string) Pgo.string."func(*opts.ListOpts, string)"�>type.func(*"".ListOpts, string)�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."func(*opts.ListOpts, string)"pPgo.weak.type.*func(*"".ListOpts, string)�"runtime.zerovalue��>type.func(*"".ListOpts, string)��>type.func(*"".ListOpts, string)�"type.*"".ListOpts�type.string�Zgo.string."func(*opts.ListOpts, string) bool"pd!func(*opts.ListOpts, string) bool Zgo.string."func(*opts.ListOpts, string) bool"�Htype.func(*"".ListOpts, string) bool��~�ɘ3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PZgo.string."func(*opts.ListOpts, string) bool"pZgo.weak.type.*func(*"".ListOpts, string) bool�"runtime.zerovalue��Htype.func(*"".ListOpts, string) bool��Htype.func(*"".ListOpts, string) bool�"type.*"".ListOpts�type.string�type.bool�Rgo.string."func(*opts.ListOpts) []string"`\func(*opts.ListOpts) []string Rgo.string."func(*opts.ListOpts) []string"�@type.func(*"".ListOpts) []string���>c�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PRgo.string."func(*opts.ListOpts) []string"pRgo.weak.type.*func(*"".ListOpts) []string�"runtime.zerovalue��@type.func(*"".ListOpts) []string��@type.func(*"".ListOpts) []string�"type.*"".ListOpts�type.[]string�*go.string."struct {}"@4 struct {} *go.string."struct {}"�type.struct {}����'�  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P*go.string."struct {}"p.go.weak.type.*struct {}�"runtime.zerovalue��type.struct {}�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�0type..hashfunc.[8]string(type..hash.[8]string�,type..eqfunc.[8]string$type..eq.[8]string�&type..alg.[8]string 0type..hashfunc.[8]string,type..eqfunc.[8]string�bruntime.gcbits.0x48484848484848480000000000000000 HHHHHHHH�*go.string."[8]string"@4 [8]string *go.string."[8]string"�type.[8]string���US�> &type..alg.[8]string0bruntime.gcbits.0x48484848484848480000000000000000P*go.string."[8]string"p.go.weak.type.*[8]string�"runtime.zerovalue�type.string�type.[]string�>go.typelink.[8]string/[8]stringtype.[8]string�.go.string."[]struct {}"@8 []struct {} .go.string."[]struct {}"� type.[]struct {}���̥� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P.go.string."[]struct {}"p2go.weak.type.*[]struct {}�"runtime.zerovalue�type.struct {}�Fgo.typelink.[]struct {}/[]struct {} type.[]struct {}�0go.string."[8]struct {}"@: [8]struct {} 0go.string."[8]struct {}"�"type.[8]struct {}��>�y �  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P0go.string."[8]struct {}"p4go.weak.type.*[8]struct {}�"runtime.zerovalue�type.struct {}� type.[]struct {}�Jgo.typelink.[8]struct {}/[8]struct {}"type.[8]struct {}�Pgo.string."*map.bucket[string]struct {}"`Z*map.bucket[string]struct {} Pgo.string."*map.bucket[string]struct {}"�Btype.*map.bucket[string]struct {}����6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."*map.bucket[string]struct {}"pTgo.weak.type.**map.bucket[string]struct {}�"runtime.zerovalue�@type.map.bucket[string]struct {}�bruntime.gcbits.0x84848484848484848400000000000000 ����������Ngo.string."map.bucket[string]struct {}"`Xmap.bucket[string]struct {} Ngo.string."map.bucket[string]struct {}"� go.string."keys"0*keys go.string."keys"�$go.string."values"0.values $go.string."values"�(go.string."overflow"@2overflow (go.string."overflow"�@type.map.bucket[string]struct {}���@���� � runtime.algarray0bruntime.gcbits.0x84848484848484848400000000000000PNgo.string."map.bucket[string]struct {}"pRgo.weak.type.*map.bucket[string]struct {}�"runtime.zerovalue��@type.map.bucket[string]struct {}� go.string."keys"�type.[8]string�$go.string."values"�"type.[8]struct {}�(go.string."overflow"�Btype.*map.bucket[string]struct {}�bruntime.gcbits.0x44844800000000000000000000000000 D�H�Hgo.string."map.hdr[string]struct {}"`Rmap.hdr[string]struct {} Hgo.string."map.hdr[string]struct {}"�&go.string."buckets"00buckets &go.string."buckets"�,go.string."oldbuckets"@6 +oldbuckets ,go.string."oldbuckets"�:type.map.hdr[string]struct {}��0v��  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000PHgo.string."map.hdr[string]struct {}"pLgo.weak.type.*map.hdr[string]struct {}�"runtime.zerovalue��:type.map.hdr[string]struct {}�&go.string."buckets"�Btype.*map.bucket[string]struct {}�,go.string."oldbuckets"�Btype.*map.bucket[string]struct {}�@go.string."map[string]struct {}"PJmap[string]struct {} @go.string."map[string]struct {}"�2type.map[string]struct {}���QR�5� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P@go.string."map[string]struct {}"pDgo.weak.type.*map[string]struct {}�"runtime.zerovalue�type.string�type.struct {}�@type.map.bucket[string]struct {}�:type.map.hdr[string]struct {}�jgo.typelink.map[string]struct {}/map[string]struct {}2type.map[string]struct {}�jgo.string."func(*opts.ListOpts) map[string]struct {}"�t)func(*opts.ListOpts) map[string]struct {} jgo.string."func(*opts.ListOpts) map[string]struct {}"�Xtype.func(*"".ListOpts) map[string]struct {}�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pjgo.string."func(*opts.ListOpts) map[string]struct {}"pjgo.weak.type.*func(*"".ListOpts) map[string]struct {}�"runtime.zerovalue��Xtype.func(*"".ListOpts) map[string]struct {}��Xtype.func(*"".ListOpts) map[string]struct {}�"type.*"".ListOpts�2type.map[string]struct {}�Hgo.string."func(*opts.ListOpts) int"`Rfunc(*opts.ListOpts) int Hgo.string."func(*opts.ListOpts) int"�6type.func(*"".ListOpts) int���< �3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PHgo.string."func(*opts.ListOpts) int"pHgo.weak.type.*func(*"".ListOpts) int�"runtime.zerovalue��6type.func(*"".ListOpts) int��6type.func(*"".ListOpts) int�"type.*"".ListOpts�type.int�\go.string."func(*opts.ListOpts, string) error"pf"func(*opts.ListOpts, string) error \go.string."func(*opts.ListOpts, string) error"�Jtype.func(*"".ListOpts, string) error���E+�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P\go.string."func(*opts.ListOpts, string) error"p\go.weak.type.*func(*"".ListOpts, string) error�"runtime.zerovalue��Jtype.func(*"".ListOpts, string) error��Jtype.func(*"".ListOpts, string) error�"type.*"".ListOpts�type.string�type.error�Ngo.string."func(*opts.ListOpts) string"`Xfunc(*opts.ListOpts) string Ngo.string."func(*opts.ListOpts) string"�go.weak.type.*func(string) bool�"runtime.zerovalue��,type.func(string) bool��,type.func(string) bool�type.string�type.bool�$go.string."GetAll"0.GetAll $go.string."GetAll"�6go.string."func() []string"@@func() []string 6go.string."func() []string"�(type.func() []string������3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."func() []string"p:go.weak.type.*func() []string�"runtime.zerovalue��(type.func() []string��(type.func() []string�type.[]string�$go.string."GetMap"0.GetMap $go.string."GetMap"�Ngo.string."func() map[string]struct {}"`Xfunc() map[string]struct {} Ngo.string."func() map[string]struct {}"�@type.func() map[string]struct {}����%�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PNgo.string."func() map[string]struct {}"pRgo.weak.type.*func() map[string]struct {}�"runtime.zerovalue��@type.func() map[string]struct {}��@type.func() map[string]struct {}�2type.map[string]struct {}�go.string."Len"0(Len go.string."Len"�,go.string."func() int"@6 +func() int ,go.string."func() int"�type.func() int���9�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."func() int"p0go.weak.type.*func() int�"runtime.zerovalue��type.func() int��type.func() int�type.int�"type.*"".ListOpts��Ü�6V � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."*opts.ListOpts"p4go.weak.type.**"".ListOpts�"runtime.zerovalue� type."".ListOpts`�"type.*"".ListOpts��"type.*"".ListOpts�$go.string."Delete"�"type.func(string)�>type.func(*"".ListOpts, string)�*"".(*ListOpts).Delete�*"".(*ListOpts).Delete�go.string."Get"�,type.func(string) bool�Htype.func(*"".ListOpts, string) bool�$"".(*ListOpts).Get�$"".(*ListOpts).Get�$go.string."GetAll"�(type.func() []string�@type.func(*"".ListOpts) []string�*"".(*ListOpts).GetAll�*"".(*ListOpts).GetAll�$go.string."GetMap"�@type.func() map[string]struct {}�Xtype.func(*"".ListOpts) map[string]struct {}�*"".(*ListOpts).GetMap�*"".(*ListOpts).GetMap�go.string."Len"�type.func() int�6type.func(*"".ListOpts) int�$"".(*ListOpts).Len�$"".(*ListOpts).Len�go.string."Set"�.type.func(string) error�Jtype.func(*"".ListOpts, string) error�$"".(*ListOpts).Set�$"".(*ListOpts).Set�$go.string."String"�$type.func() string�Y� � runtime.algarray0Btype..gc.map.bucket[string]string@Jtype..gcprog.map.bucket[string]stringPHgo.string."map.bucket[string]string"pLgo.weak.type.*map.bucket[string]string�"runtime.zerovalue��:type.map.bucket[string]string� go.string."keys"�type.[8]string�$go.string."values"�type.[8]string�(go.string."overflow"�go.weak.type.*map[string]string�"runtime.zerovalue�type.string�type.string�:type.map.bucket[string]string�4type.map.hdr[string]string�^go.typelink.map[string]string/map[string]string,type.map[string]string�0go.string."opts.MapOpts"@: opts.MapOpts 0go.string."opts.MapOpts"�&go.string."MapOpts"00MapOpts &go.string."MapOpts"�type."".MapOpts��#�B� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P0go.string."opts.MapOpts"p type.*"".MapOpts�"runtime.zerovalue��type."".MapOpts�$go.string."values"�"go.importpath."".�,type.map[string]string�*go.string."validator"�"go.importpath."".�0type."".ValidatorFctType`�type."".MapOpts�&go.string."MapOpts"�"go.importpath."".��type."".MapOpts�2go.string."*opts.MapOpts"@< *opts.MapOpts 2go.string."*opts.MapOpts"�Zgo.string."func(*opts.MapOpts, string) error"pd!func(*opts.MapOpts, string) error Zgo.string."func(*opts.MapOpts, string) error"�Htype.func(*"".MapOpts, string) error��go 3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PZgo.string."func(*opts.MapOpts, string) error"pZgo.weak.type.*func(*"".MapOpts, string) error�"runtime.zerovalue��Htype.func(*"".MapOpts, string) error��Htype.func(*"".MapOpts, string) error� type.*"".MapOpts�type.string�type.error�Lgo.string."func(*opts.MapOpts) string"`Vfunc(*opts.MapOpts) string Lgo.string."func(*opts.MapOpts) string"�:type.func(*"".MapOpts) string��*�x3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PLgo.string."func(*opts.MapOpts) string"pLgo.weak.type.*func(*"".MapOpts) string�"runtime.zerovalue��:type.func(*"".MapOpts) string��:type.func(*"".MapOpts) string� type.*"".MapOpts�type.string� type.*"".MapOpts��7��6$ � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P2go.string."*opts.MapOpts"p2go.weak.type.**"".MapOpts�"runtime.zerovalue�type."".MapOpts`� type.*"".MapOpts�� type.*"".MapOpts�go.string."Set"�.type.func(string) error�Htype.func(*"".MapOpts, string) error�""".(*MapOpts).Set�""".(*MapOpts).Set�$go.string."String"�$type.func() string�:type.func(*"".MapOpts) string�("".(*MapOpts).String�("".(*MapOpts).String�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�0type..hashfunc.[3]string(type..hash.[3]string�,type..eqfunc.[3]string$type..eq.[3]string�&type..alg.[3]string 0type..hashfunc.[3]string,type..eqfunc.[3]string�bruntime.gcbits.0x48484800000000000000000000000000 HHH�*go.string."[3]string"@4 [3]string *go.string."[3]string"�type.[3]string��0C�iB &type..alg.[3]string0bruntime.gcbits.0x48484800000000000000000000000000P*go.string."[3]string"p.go.weak.type.*[3]string�"runtime.zerovalue�type.string�type.[]string�>go.typelink.[3]string/[3]stringtype.[3]string�,go.string."*[3]string"@6 +*[3]string ,go.string."*[3]string"�type.*[3]string�� ++� 6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."*[3]string"p0go.weak.type.**[3]string�"runtime.zerovalue�type.[3]string�$go.string."[]bool"0.[]bool $go.string."[]bool"�type.[]bool����� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P$go.string."[]bool"p(go.weak.type.*[]bool�"runtime.zerovalue�type.bool�2go.typelink.[]bool/[]booltype.[]bool�&go.string."[8]bool"00[8]bool &go.string."[8]bool"�type.[8]bool��s�5� � runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P&go.string."[8]bool"p*go.weak.type.*[8]bool�"runtime.zerovalue�type.bool�type.[]bool�6go.typelink.[8]bool/[8]booltype.[8]bool�Fgo.string."*map.bucket[string]bool"PP*map.bucket[string]bool Fgo.string."*map.bucket[string]bool"�8type.*map.bucket[string]bool���[�E6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PFgo.string."*map.bucket[string]bool"pJgo.weak.type.**map.bucket[string]bool�"runtime.zerovalue�6type.map.bucket[string]bool�,>type..gc.map.bucket[string]bool(�Ftype..gcprog.map.bucket[string]bool����%�Dgo.string."map.bucket[string]bool"PNmap.bucket[string]bool Dgo.string."map.bucket[string]bool"�6type.map.bucket[string]bool���2aB�Y�� � runtime.algarray0>type..gc.map.bucket[string]bool@Ftype..gcprog.map.bucket[string]boolPDgo.string."map.bucket[string]bool"pHgo.weak.type.*map.bucket[string]bool�"runtime.zerovalue��6type.map.bucket[string]bool� go.string."keys"�type.[8]string�$go.string."values"�type.[8]bool�(go.string."overflow"�8type.*map.bucket[string]bool�>go.string."map.hdr[string]bool"PHmap.hdr[string]bool >go.string."map.hdr[string]bool"�0type.map.hdr[string]bool��03�(  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000P>go.string."map.hdr[string]bool"pBgo.weak.type.*map.hdr[string]bool�"runtime.zerovalue��0type.map.hdr[string]bool�&go.string."buckets"�8type.*map.bucket[string]bool�,go.string."oldbuckets"�8type.*map.bucket[string]bool�6go.string."map[string]bool"@@map[string]bool 6go.string."map[string]bool"�(type.map[string]bool����5� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."map[string]bool"p:go.weak.type.*map[string]bool�"runtime.zerovalue�type.string�type.bool�6type.map.bucket[string]bool�0type.map.hdr[string]bool�Vgo.typelink.map[string]bool/map[string]bool(type.map[string]bool�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�go.typelink.[][]uint8/[][]uint8type.[][]uint8�8go.string."[]*ulimit.Ulimit"PB[]*ulimit.Ulimit 8go.string."[]*ulimit.Ulimit"��type.[]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��ȯ � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P8go.string."[]*ulimit.Ulimit"p�go.weak.type.*[]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�"runtime.zerovalue��type.*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��go.typelink.[]*ulimit.Ulimit/[]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�type.[]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�bruntime.gcbits.0x88888888000000000000000000000000 �����:go.string."[8]*ulimit.Ulimit"PD[8]*ulimit.Ulimit :go.string."[8]*ulimit.Ulimit"��type.[8]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��@��@  runtime.algarray0bruntime.gcbits.0x88888888000000000000000000000000P:go.string."[8]*ulimit.Ulimit"p�go.weak.type.*[8]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�"runtime.zerovalue��type.*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��type.[]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��go.typelink.[8]*ulimit.Ulimit/[8]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�type.[8]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�Zgo.string."*map.bucket[string]*ulimit.Ulimit"pd!*map.bucket[string]*ulimit.Ulimit Zgo.string."*map.bucket[string]*ulimit.Ulimit"��type.*map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�����6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PZgo.string."*map.bucket[string]*ulimit.Ulimit"p�go.weak.type.**map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�"runtime.zerovalue��type.map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�bruntime.gcbits.0x84848484848484848488888888000000 ��������������Xgo.string."map.bucket[string]*ulimit.Ulimit"pb map.bucket[string]*ulimit.Ulimit Xgo.string."map.bucket[string]*ulimit.Ulimit"��type.map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit���*���� � runtime.algarray0bruntime.gcbits.0x84848484848484848488888888000000PXgo.string."map.bucket[string]*ulimit.Ulimit"p�go.weak.type.*map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�"runtime.zerovalue���type.map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit� go.string."keys"�type.[8]string�$go.string."values"��type.[8]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�(go.string."overflow"��type.*map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�Rgo.string."map.hdr[string]*ulimit.Ulimit"`\map.hdr[string]*ulimit.Ulimit Rgo.string."map.hdr[string]*ulimit.Ulimit"��type.map.hdr[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��0�I:"  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000PRgo.string."map.hdr[string]*ulimit.Ulimit"p�go.weak.type.*map.hdr[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�"runtime.zerovalue���type.map.hdr[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�&go.string."buckets"��type.*map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�,go.string."oldbuckets"��type.*map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�Jgo.string."map[string]*ulimit.Ulimit"`Tmap[string]*ulimit.Ulimit Jgo.string."map[string]*ulimit.Ulimit"��type.map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit����pt5� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PJgo.string."map[string]*ulimit.Ulimit"p�go.weak.type.*map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�"runtime.zerovalue�type.string��type.*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��type.map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��type.map.hdr[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��go.typelink.map[string]*ulimit.Ulimit/map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�type.map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�Lgo.string."*map[string]*ulimit.Ulimit"`V*map[string]*ulimit.Ulimit Lgo.string."*map[string]*ulimit.Ulimit"��type.*map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�����6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PLgo.string."*map[string]*ulimit.Ulimit"p�go.weak.type.**map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�"runtime.zerovalue��type.map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�6go.string."*opts.UlimitOpt"@@*opts.UlimitOpt 6go.string."*opts.UlimitOpt"�dgo.string."func(*opts.UlimitOpt) []*ulimit.Ulimit"pn&func(*opts.UlimitOpt) []*ulimit.Ulimit dgo.string."func(*opts.UlimitOpt) []*ulimit.Ulimit"��type.func(*"".UlimitOpt) []*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��昩3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pdgo.string."func(*opts.UlimitOpt) []*ulimit.Ulimit"p�go.weak.type.*func(*"".UlimitOpt) []*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�"runtime.zerovalue���type.func(*"".UlimitOpt) []*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit���type.func(*"".UlimitOpt) []*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�$type.*"".UlimitOpt��type.[]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�^go.string."func(*opts.UlimitOpt, string) error"ph#func(*opts.UlimitOpt, string) error ^go.string."func(*opts.UlimitOpt, string) error"�Ltype.func(*"".UlimitOpt, string) error��U�p3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P^go.string."func(*opts.UlimitOpt, string) error"p^go.weak.type.*func(*"".UlimitOpt, string) error�"runtime.zerovalue��Ltype.func(*"".UlimitOpt, string) error��Ltype.func(*"".UlimitOpt, string) error�$type.*"".UlimitOpt�type.string�type.error�Pgo.string."func(*opts.UlimitOpt) string"`Zfunc(*opts.UlimitOpt) string Pgo.string."func(*opts.UlimitOpt) string"�>type.func(*"".UlimitOpt) string��_��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."func(*opts.UlimitOpt) string"pPgo.weak.type.*func(*"".UlimitOpt) string�"runtime.zerovalue��>type.func(*"".UlimitOpt) string��>type.func(*"".UlimitOpt) string�$type.*"".UlimitOpt�type.string�&go.string."GetList"00GetList &go.string."GetList"�Fgo.string."func() []*ulimit.Ulimit"PPfunc() []*ulimit.Ulimit Fgo.string."func() []*ulimit.Ulimit"��type.func() []*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��:�x3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PFgo.string."func() []*ulimit.Ulimit"p�go.weak.type.*func() []*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�"runtime.zerovalue���type.func() []*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit���type.func() []*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��type.[]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�$type.*"".UlimitOpt��m�Q�6. � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."*opts.UlimitOpt"p6go.weak.type.**"".UlimitOpt�"runtime.zerovalue�"type."".UlimitOpt`�$type.*"".UlimitOpt��$type.*"".UlimitOpt�&go.string."GetList"��type.func() []*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��type.func(*"".UlimitOpt) []*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�."".(*UlimitOpt).GetList�."".(*UlimitOpt).GetList�go.string."Set"�.type.func(string) error�Ltype.func(*"".UlimitOpt, string) error�&"".(*UlimitOpt).Set�&"".(*UlimitOpt).Set�$go.string."String"�$type.func() string�>type.func(*"".UlimitOpt) string�,"".(*UlimitOpt).String�,"".(*UlimitOpt).String�4go.string."opts.UlimitOpt"@>opts.UlimitOpt 4go.string."opts.UlimitOpt"�*go.string."UlimitOpt"@4 UlimitOpt *go.string."UlimitOpt"�"type."".UlimitOpt��3��e9 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."opts.UlimitOpt"p$type.*"".UlimitOpt�"runtime.zerovalue��"type."".UlimitOpt�$go.string."values"�"go.importpath."".��type.*map[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit`�"type."".UlimitOpt�*go.string."UlimitOpt"�"go.importpath."".��"type."".UlimitOpt�6go.string."**ulimit.Ulimit"@@**ulimit.Ulimit 6go.string."**ulimit.Ulimit"��type.**github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��3�*�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."**ulimit.Ulimit"p�go.weak.type.***github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�"runtime.zerovalue��type.*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�Tgo.string."*map.hdr[string]*ulimit.Ulimit"`^*map.hdr[string]*ulimit.Ulimit Tgo.string."*map.hdr[string]*ulimit.Ulimit"��type.*map.hdr[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit����W�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PTgo.string."*map.hdr[string]*ulimit.Ulimit"p�go.weak.type.**map.hdr[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�"runtime.zerovalue��type.map.hdr[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�*go.string."[]uintptr"@4 []uintptr *go.string."[]uintptr"�type.[]uintptr���3�] � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P*go.string."[]uintptr"p.go.weak.type.*[]uintptr�"runtime.zerovalue�type.uintptr�>go.typelink.[]uintptr/[]uintptrtype.[]uintptr�,go.string."[4]uintptr"@6 +[4]uintptr ,go.string."[4]uintptr"�type.[4]uintptr�� l<�  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P,go.string."[4]uintptr"p0go.weak.type.*[4]uintptr�"runtime.zerovalue�type.uintptr�type.[]uintptr�Bgo.typelink.[4]uintptr/[4]uintptrtype.[4]uintptr�bruntime.gcbits.0x88888844440000000000000000000000 ���DD�Tgo.string."map.iter[string]*ulimit.Ulimit"`^map.iter[string]*ulimit.Ulimit Tgo.string."map.iter[string]*ulimit.Ulimit"�go.string."key"0(key go.string."key"�go.string."val"0(val go.string."val"�go.string."t"0$t go.string."t"�go.string."h"0$h go.string."h"� go.string."bptr"0*bptr go.string."bptr"�"go.string."other"0,other "go.string."other"��type.map.iter[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit��P� (0( � runtime.algarray0bruntime.gcbits.0x88888844440000000000000000000000PTgo.string."map.iter[string]*ulimit.Ulimit"p�go.weak.type.*map.iter[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�"runtime.zerovalue���type.map.iter[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�go.string."key"�type.*string�go.string."val"��type.**github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�go.string."t"�type.*uint8�go.string."h"��type.*map.hdr[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�&go.string."buckets"��type.*map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit� go.string."bptr"��type.*map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit.Ulimit�"go.string."other"�type.[4]uintptr�Lgo.string."*opts.ValidatorFctListType"`V*opts.ValidatorFctListType Lgo.string."*opts.ValidatorFctListType"�:type.*"".ValidatorFctListType��D?+�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PLgo.string."*opts.ValidatorFctListType"pLgo.weak.type.**"".ValidatorFctListType�"runtime.zerovalue�8type."".ValidatorFctListType�Jgo.string."opts.ValidatorFctListType"`Topts.ValidatorFctListType Jgo.string."opts.ValidatorFctListType"�@go.string."ValidatorFctListType"PJValidatorFctListType @go.string."ValidatorFctListType"�8type."".ValidatorFctListType��-W3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PJgo.string."opts.ValidatorFctListType"p:type.*"".ValidatorFctListType�"runtime.zerovalue��8type."".ValidatorFctListType��8type."".ValidatorFctListType�type.string�type.[]string�type.error`�8type."".ValidatorFctListType�@go.string."ValidatorFctListType"�"go.importpath."".��8type."".ValidatorFctListType�,go.string."*[8]string"@6 +*[8]string ,go.string."*[8]string"�type.*[8]string����o6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."*[8]string"p0go.weak.type.**[8]string�"runtime.zerovalue�type.[8]string�go.string."net"0(net go.string."net"�$go.importpath.net.  go.string."net"�&go.string."runtime"00runtime &go.string."runtime"�,go.importpath.runtime.  &go.string."runtime"�"go.string."bufio"0,bufio "go.string."bufio"�(go.importpath.bufio.  "go.string."bufio"��go.string."github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume"��Jjackfan.us.kg/fsouza/go-dockerclient/external/github.com/docker/docker/volume �go.string."github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume"��go.importpath.github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume. J �go.string."github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume"��go.string."github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit"��Njackfan.us.kg/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit �go.string."github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit"��go.importpath.github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit. N �go.string."github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit"�go.string."os"0&os go.string."os"�"go.importpath.os.  go.string."os"�&go.string."strings"00strings &go.string."strings"�,go.importpath.strings.  &go.string."strings"�go.string."fmt"0(fmt go.string."fmt"�$go.importpath.fmt.  go.string."fmt"�$go.string."regexp"0.regexp $go.string."regexp"�*go.importpath.regexp.  $go.string."regexp"��go.string."github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers"��Ojackfan.us.kg/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers �go.string."github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers"��go.importpath.github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers. O �go.string."github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers"� go.string."path"0*path go.string."path"�&go.importpath.path.  go.string."path"�.type..hash.[0]string·f(type..hash.[0]string�$runtime.strhash·fruntime.strhash�*type..eq.[0]string·f$type..eq.[0]string�@"".(*ErrBadEnvVariable).Error·f:"".(*ErrBadEnvVariable).Error�(runtime.panicwrap·f"runtime.panicwrap�:type..hash.[1]interface {}·f4type..hash.[1]interface {}�.runtime.nilinterhash·f(runtime.nilinterhash�6type..eq.[1]interface {}·f0type..eq.[1]interface {}�$runtime.efaceeq·fruntime.efaceeq�:type..hash.[2]interface {}·f4type..hash.[2]interface {}�6type..eq.[2]interface {}·f0type..eq.[2]interface {}�4"".(*IpOpt).DefaultMask·f."".(*IpOpt).DefaultMask�*net.IP.DefaultMask·f$net.IP.DefaultMask�("".(*IpOpt).Equal·f""".(*IpOpt).Equal�net.IP.Equal·fnet.IP.Equal�<"".(*IpOpt).IsGlobalUnicast·f6"".(*IpOpt).IsGlobalUnicast�2net.IP.IsGlobalUnicast·f,net.IP.IsGlobalUnicast�P"".(*IpOpt).IsInterfaceLocalMulticast·fJ"".(*IpOpt).IsInterfaceLocalMulticast�F"".(*IpOpt).IsLinkLocalMulticast·f@"".(*IpOpt).IsLinkLocalMulticast� +__.PKGDEF 0 0 0 644 49694 ` +go object darwin amd64 go1.4.2 X:precisestack + +$$ +package context + import sync "sync" + import runtime "runtime" + import time "time" + import http "net/http" + import url "net/url" // indirect + type @"net/url".Userinfo struct { @"net/url".username string; @"net/url".password string; @"net/url".passwordSet bool } + func (@"net/url".u·3 *@"net/url".Userinfo "esc:0x1") Password () (? string, ? bool) { if @"net/url".u·3.@"net/url".passwordSet { return @"net/url".u·3.@"net/url".password, true }; return "", false } + func (@"net/url".u·2 *@"net/url".Userinfo "esc:0x1") String () (? string) + func (@"net/url".u·2 *@"net/url".Userinfo "esc:0x1") Username () (? string) { return @"net/url".u·2.@"net/url".username } + type @"net/url".Values map[string][]string + func (@"net/url".v·1 @"net/url".Values "esc:0x0") Add (@"net/url".key·2 string, @"net/url".value·3 string) { @"net/url".v·1[@"net/url".key·2] = append(@"net/url".v·1[@"net/url".key·2], @"net/url".value·3) } + func (@"net/url".v·1 @"net/url".Values "esc:0x0") Del (@"net/url".key·2 string "esc:0x0") { delete(@"net/url".v·1, @"net/url".key·2) } + func (@"net/url".v·2 @"net/url".Values "esc:0x0") Encode () (? string) + func (@"net/url".v·2 @"net/url".Values "esc:0x0") Get (@"net/url".key·3 string "esc:0x0") (? string) { if @"net/url".v·2 == nil { return "" }; var @"net/url".vs·4 []string; ; var @"net/url".ok·5 bool; ; @"net/url".vs·4, @"net/url".ok·5 = @"net/url".v·2[@"net/url".key·3]; if !@"net/url".ok·5 || len(@"net/url".vs·4) == 0x0 { return "" }; return @"net/url".vs·4[0x0] } + func (@"net/url".v·1 @"net/url".Values "esc:0x0") Set (@"net/url".key·2 string, @"net/url".value·3 string) { @"net/url".v·1[@"net/url".key·2] = ([]string{ 0x0:@"net/url".value·3 }) } + type @"net/url".URL struct { Scheme string; Opaque string; User *@"net/url".Userinfo; Host string; Path string; RawQuery string; Fragment string } + func (@"net/url".u·2 *@"net/url".URL "esc:0x0") IsAbs () (? bool) { return @"net/url".u·2.Scheme != "" } + func (@"net/url".u·3 *@"net/url".URL "esc:0x2") Parse (@"net/url".ref·4 string) (? *@"net/url".URL, ? error) + func (@"net/url".u·2 *@"net/url".URL) Query () (? @"net/url".Values) + func (@"net/url".u·2 *@"net/url".URL "esc:0x1") RequestURI () (? string) + func (@"net/url".u·2 *@"net/url".URL "esc:0x2") ResolveReference (@"net/url".ref·3 *@"net/url".URL "esc:0x2") (? *@"net/url".URL) + func (@"net/url".u·2 *@"net/url".URL "esc:0x0") String () (? string) + import io "io" // indirect + type @"io".Writer interface { Write(@"io".p []byte) (@"io".n int, @"io".err error) } + type @"net/http".keyValues struct { @"net/http".key string; @"net/http".values []string } + type @"net/http".headerSorter struct { @"net/http".kvs []@"net/http".keyValues } + func (@"net/http".s·2 *@"net/http".headerSorter "esc:0x0") Len () (? int) { return len(@"net/http".s·2.@"net/http".kvs) } + func (@"net/http".s·2 *@"net/http".headerSorter "esc:0x0") Less (@"net/http".i·3 int, @"net/http".j·4 int) (? bool) { return @"net/http".s·2.@"net/http".kvs[@"net/http".i·3].@"net/http".key < @"net/http".s·2.@"net/http".kvs[@"net/http".j·4].@"net/http".key } + func (@"net/http".s·1 *@"net/http".headerSorter "esc:0x0") Swap (@"net/http".i·2 int, @"net/http".j·3 int) { @"net/http".s·1.@"net/http".kvs[@"net/http".i·2], @"net/http".s·1.@"net/http".kvs[@"net/http".j·3] = @"net/http".s·1.@"net/http".kvs[@"net/http".j·3], @"net/http".s·1.@"net/http".kvs[@"net/http".i·2] } + type @"net/http".Header map[string][]string + func (@"net/http".h·1 @"net/http".Header "esc:0x0") Add (@"net/http".key·2 string, @"net/http".value·3 string) + func (@"net/http".h·1 @"net/http".Header "esc:0x0") Del (@"net/http".key·2 string "esc:0x0") + func (@"net/http".h·2 @"net/http".Header "esc:0x0") Get (@"net/http".key·3 string "esc:0x0") (? string) + func (@"net/http".h·1 @"net/http".Header "esc:0x0") Set (@"net/http".key·2 string, @"net/http".value·3 string) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") Write (@"net/http".w·3 @"io".Writer) (? error) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") WriteSubset (@"net/http".w·3 @"io".Writer, @"net/http".exclude·4 map[string]bool "esc:0x0") (? error) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") @"net/http".clone () (? @"net/http".Header) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") @"net/http".get (@"net/http".key·3 string "esc:0x0") (? string) { var @"net/http".v·4 []string; ; @"net/http".v·4 = @"net/http".h·2[@"net/http".key·3]; if len(@"net/http".v·4) > 0x0 { return @"net/http".v·4[0x0] }; return "" } + func (@"net/http".h·3 @"net/http".Header "esc:0x0") @"net/http".sortedKeyValues (@"net/http".exclude·4 map[string]bool "esc:0x0") (@"net/http".kvs·1 []@"net/http".keyValues, @"net/http".hs·2 *@"net/http".headerSorter) + type @"io".ReadCloser interface { Close() (? error); Read(@"io".p []byte) (@"io".n int, @"io".err error) } + import multipart "mime/multipart" // indirect + import textproto "net/textproto" // indirect + type @"net/textproto".MIMEHeader map[string][]string + func (@"net/textproto".h·1 @"net/textproto".MIMEHeader "esc:0x0") Add (@"net/textproto".key·2 string, @"net/textproto".value·3 string) + func (@"net/textproto".h·1 @"net/textproto".MIMEHeader "esc:0x0") Del (@"net/textproto".key·2 string "esc:0x0") + func (@"net/textproto".h·2 @"net/textproto".MIMEHeader "esc:0x0") Get (@"net/textproto".key·3 string "esc:0x0") (? string) + func (@"net/textproto".h·1 @"net/textproto".MIMEHeader "esc:0x0") Set (@"net/textproto".key·2 string, @"net/textproto".value·3 string) + type @"mime/multipart".File interface { Close() (? error); Read(@"io".p []byte) (@"io".n int, @"io".err error); ReadAt(@"io".p []byte, @"io".off int64) (@"io".n int, @"io".err error); Seek(@"io".offset int64, @"io".whence int) (? int64, ? error) } + type @"mime/multipart".FileHeader struct { Filename string; Header @"net/textproto".MIMEHeader; @"mime/multipart".content []byte; @"mime/multipart".tmpfile string } + func (@"mime/multipart".fh·3 *@"mime/multipart".FileHeader) Open () (? @"mime/multipart".File, ? error) + type @"mime/multipart".Form struct { Value map[string][]string; File map[string][]*@"mime/multipart".FileHeader } + func (@"mime/multipart".f·2 *@"mime/multipart".Form "esc:0x0") RemoveAll () (? error) + import tls "crypto/tls" // indirect + import x509 "crypto/x509" // indirect + type @"crypto/x509".SignatureAlgorithm int + type @"crypto/x509".PublicKeyAlgorithm int + import big "math/big" // indirect + type @"math/big".Word uintptr + type @"math/big".divisor struct { @"math/big".bbb @"math/big".nat; @"math/big".nbits int; @"math/big".ndigits int } + import rand "math/rand" // indirect + type @"math/rand".Source interface { Int63() (? int64); Seed(@"math/rand".seed int64) } + type @"math/rand".Rand struct { @"math/rand".src @"math/rand".Source } + func (@"math/rand".r·2 *@"math/rand".Rand) ExpFloat64 () (? float64) + func (@"math/rand".r·2 *@"math/rand".Rand) Float32 () (? float32) + func (@"math/rand".r·2 *@"math/rand".Rand) Float64 () (? float64) + func (@"math/rand".r·2 *@"math/rand".Rand) Int () (? int) + func (@"math/rand".r·2 *@"math/rand".Rand) Int31 () (? int32) + func (@"math/rand".r·2 *@"math/rand".Rand) Int31n (@"math/rand".n·3 int32) (? int32) + func (@"math/rand".r·2 *@"math/rand".Rand) Int63 () (? int64) + func (@"math/rand".r·2 *@"math/rand".Rand) Int63n (@"math/rand".n·3 int64) (? int64) + func (@"math/rand".r·2 *@"math/rand".Rand) Intn (@"math/rand".n·3 int) (? int) + func (@"math/rand".r·2 *@"math/rand".Rand) NormFloat64 () (? float64) + func (@"math/rand".r·2 *@"math/rand".Rand) Perm (@"math/rand".n·3 int) (? []int) + func (@"math/rand".r·1 *@"math/rand".Rand) Seed (@"math/rand".seed·2 int64) + func (@"math/rand".r·2 *@"math/rand".Rand) Uint32 () (? uint32) + type @"io".RuneScanner interface { ReadRune() (@"io".r rune, @"io".size int, @"io".err error); UnreadRune() (? error) } + type @"math/big".nat []@"math/big".Word + func (@"math/big".z·2 @"math/big".nat) @"math/big".add (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".and (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".andNot (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x0") @"math/big".bit (@"math/big".i·3 uint) (? uint) { var @"math/big".j·4 int; ; @"math/big".j·4 = int(@"math/big".i·3 / 0x40); if @"math/big".j·4 >= len(@"math/big".z·2) { return 0x0 }; return uint(@"math/big".z·2[@"math/big".j·4] >> (@"math/big".i·3 % 0x40) & @"math/big".Word(0x1)) } + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".bitLen () (? int) + func (@"math/big".z·2 @"math/big".nat "esc:0x0") @"math/big".bytes (@"math/big".buf·3 []byte "esc:0x0") (@"math/big".i·1 int) + func (@"math/big".z·1 @"math/big".nat "esc:0x0") @"math/big".clear () + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".cmp (@"math/big".y·3 @"math/big".nat "esc:0x0") (@"math/big".r·1 int) + func (@"math/big".q·1 @"math/big".nat) @"math/big".convertWords (@"math/big".s·2 []byte "esc:0x0", @"math/big".charset·3 string "esc:0x0", @"math/big".b·4 @"math/big".Word, @"math/big".ndigits·5 int, @"math/big".bb·6 @"math/big".Word, @"math/big".table·7 []@"math/big".divisor "esc:0x0") + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".decimalString () (? string) + func (@"math/big".z·3 @"math/big".nat) @"math/big".div (@"math/big".z2·4 @"math/big".nat, @"math/big".u·5 @"math/big".nat, @"math/big".v·6 @"math/big".nat) (@"math/big".q·1 @"math/big".nat, @"math/big".r·2 @"math/big".nat) + func (@"math/big".z·3 @"math/big".nat "esc:0x2") @"math/big".divLarge (@"math/big".u·4 @"math/big".nat, @"math/big".uIn·5 @"math/big".nat, @"math/big".v·6 @"math/big".nat) (@"math/big".q·1 @"math/big".nat, @"math/big".r·2 @"math/big".nat) + func (@"math/big".z·3 @"math/big".nat) @"math/big".divW (@"math/big".x·4 @"math/big".nat, @"math/big".y·5 @"math/big".Word) (@"math/big".q·1 @"math/big".nat, @"math/big".r·2 @"math/big".Word) + func (@"math/big".z·2 @"math/big".nat) @"math/big".expNN (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat "esc:0x0", @"math/big".m·5 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".expNNWindowed (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat "esc:0x0", @"math/big".m·5 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".expWW (@"math/big".x·3 @"math/big".Word, @"math/big".y·4 @"math/big".Word) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".make (@"math/big".n·3 int) (? @"math/big".nat) + func (@"math/big".x·2 @"math/big".nat) @"math/big".modW (@"math/big".d·3 @"math/big".Word) (@"math/big".r·1 @"math/big".Word) + func (@"math/big".z·2 @"math/big".nat) @"math/big".mul (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".mulAddWW (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".Word, @"math/big".r·5 @"math/big".Word) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".mulRange (@"math/big".a·3 uint64, @"math/big".b·4 uint64) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".norm () (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".or (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".n·2 @"math/big".nat) @"math/big".probablyPrime (@"math/big".reps·3 int) (? bool) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".random (@"math/big".rand·3 *@"math/rand".Rand, @"math/big".limit·4 @"math/big".nat "esc:0x0", @"math/big".n·5 int) (? @"math/big".nat) + func (@"math/big".z·4 @"math/big".nat) @"math/big".scan (@"math/big".r·5 @"io".RuneScanner, @"math/big".base·6 int) (? @"math/big".nat, ? int, ? error) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".set (@"math/big".x·3 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setBit (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".i·4 uint, @"math/big".b·5 uint) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setBytes (@"math/big".buf·3 []byte "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setUint64 (@"math/big".x·3 uint64) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setWord (@"math/big".x·3 @"math/big".Word) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".shl (@"math/big".x·3 @"math/big".nat, @"math/big".s·4 uint) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".shr (@"math/big".x·3 @"math/big".nat, @"math/big".s·4 uint) (? @"math/big".nat) + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".string (@"math/big".charset·3 string "esc:0x0") (? string) + func (@"math/big".z·2 @"math/big".nat) @"math/big".sub (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".trailingZeroBits () (? uint) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".xor (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + import fmt "fmt" // indirect + type @"fmt".State interface { Flag(@"fmt".c int) (? bool); Precision() (@"fmt".prec int, @"fmt".ok bool); Width() (@"fmt".wid int, @"fmt".ok bool); Write(@"fmt".b []byte) (@"fmt".ret int, @"fmt".err error) } + type @"fmt".ScanState interface { Read(@"fmt".buf []byte) (@"fmt".n int, @"fmt".err error); ReadRune() (@"fmt".r rune, @"fmt".size int, @"fmt".err error); SkipSpace(); Token(@"fmt".skipSpace bool, @"fmt".f func(? rune) (? bool)) (@"fmt".token []byte, @"fmt".err error); UnreadRune() (? error); Width() (@"fmt".wid int, @"fmt".ok bool) } + type @"math/big".Int struct { @"math/big".neg bool; @"math/big".abs @"math/big".nat } + func (@"math/big".z·2 *@"math/big".Int) Abs (@"math/big".x·3 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Add (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) And (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) AndNot (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Binomial (@"math/big".n·3 int64, @"math/big".k·4 int64) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int) Bit (@"math/big".i·3 int) (? uint) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") BitLen () (? int) + func (@"math/big".x·2 *@"math/big".Int "esc:0x1") Bits () (? []@"math/big".Word) { return @"math/big".x·2.@"math/big".abs } + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Bytes () (? []byte) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Cmp (@"math/big".y·3 *@"math/big".Int "esc:0x0") (@"math/big".r·1 int) + func (@"math/big".z·2 *@"math/big".Int) Div (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int) DivMod (@"math/big".x·4 *@"math/big".Int, @"math/big".y·5 *@"math/big".Int, @"math/big".m·6 *@"math/big".Int) (? *@"math/big".Int, ? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Exp (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int "esc:0x0", @"math/big".m·5 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".x·1 *@"math/big".Int "esc:0x0") Format (@"math/big".s·2 @"fmt".State, @"math/big".ch·3 rune) + func (@"math/big".z·2 *@"math/big".Int) GCD (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int, @"math/big".a·5 *@"math/big".Int, @"math/big".b·6 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) GobDecode (@"math/big".buf·3 []byte "esc:0x0") (? error) + func (@"math/big".x·3 *@"math/big".Int "esc:0x0") GobEncode () (? []byte, ? error) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Int64 () (? int64) + func (@"math/big".z·2 *@"math/big".Int) Lsh (@"math/big".x·3 *@"math/big".Int, @"math/big".n·4 uint) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int "esc:0x0") MarshalJSON () (? []byte, ? error) + func (@"math/big".z·3 *@"math/big".Int "esc:0x0") MarshalText () (@"math/big".text·1 []byte, @"math/big".err·2 error) + func (@"math/big".z·2 *@"math/big".Int) Mod (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) ModInverse (@"math/big".g·3 *@"math/big".Int, @"math/big".n·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Mul (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) MulRange (@"math/big".a·3 int64, @"math/big".b·4 int64) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Neg (@"math/big".x·3 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Not (@"math/big".x·3 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Or (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int) ProbablyPrime (@"math/big".n·3 int) (? bool) + func (@"math/big".z·2 *@"math/big".Int) Quo (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int) QuoRem (@"math/big".x·4 *@"math/big".Int, @"math/big".y·5 *@"math/big".Int, @"math/big".r·6 *@"math/big".Int) (? *@"math/big".Int, ? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Rand (@"math/big".rnd·3 *@"math/rand".Rand, @"math/big".n·4 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Rem (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Rsh (@"math/big".x·3 *@"math/big".Int, @"math/big".n·4 uint) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Scan (@"math/big".s·3 @"fmt".ScanState, @"math/big".ch·4 rune) (? error) + func (@"math/big".z·2 *@"math/big".Int) Set (@"math/big".x·3 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) SetBit (@"math/big".x·3 *@"math/big".Int, @"math/big".i·4 int, @"math/big".b·5 uint) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int "esc:0x2") SetBits (@"math/big".abs·3 []@"math/big".Word) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) SetBytes (@"math/big".buf·3 []byte "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) SetInt64 (@"math/big".x·3 int64) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int) SetString (@"math/big".s·4 string, @"math/big".base·5 int) (? *@"math/big".Int, ? bool) + func (@"math/big".z·2 *@"math/big".Int) SetUint64 (@"math/big".x·3 uint64) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Sign () (? int) { if len(@"math/big".x·2.@"math/big".abs) == 0x0 { return 0x0 }; if @"math/big".x·2.@"math/big".neg { return -0x1 }; return 0x1 } + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") String () (? string) + func (@"math/big".z·2 *@"math/big".Int) Sub (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Uint64 () (? uint64) + func (@"math/big".z·2 *@"math/big".Int) UnmarshalJSON (@"math/big".text·3 []byte) (? error) + func (@"math/big".z·2 *@"math/big".Int) UnmarshalText (@"math/big".text·3 []byte) (? error) + func (@"math/big".z·2 *@"math/big".Int) Xor (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) @"math/big".binaryGCD (@"math/big".a·3 *@"math/big".Int, @"math/big".b·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·4 *@"math/big".Int) @"math/big".scan (@"math/big".r·5 @"io".RuneScanner, @"math/big".base·6 int) (? *@"math/big".Int, ? int, ? error) + import pkix "crypto/x509/pkix" // indirect + import asn1 "encoding/asn1" // indirect + type @"encoding/asn1".ObjectIdentifier []int + func (@"encoding/asn1".oi·2 @"encoding/asn1".ObjectIdentifier "esc:0x0") Equal (@"encoding/asn1".other·3 @"encoding/asn1".ObjectIdentifier "esc:0x0") (? bool) + func (@"encoding/asn1".oi·2 @"encoding/asn1".ObjectIdentifier "esc:0x0") String () (? string) + type @"crypto/x509/pkix".AttributeTypeAndValue struct { Type @"encoding/asn1".ObjectIdentifier; Value interface {} } + type @"crypto/x509/pkix".RelativeDistinguishedNameSET []@"crypto/x509/pkix".AttributeTypeAndValue + type @"crypto/x509/pkix".RDNSequence []@"crypto/x509/pkix".RelativeDistinguishedNameSET + type @"crypto/x509/pkix".Name struct { Country []string; Organization []string; OrganizationalUnit []string; Locality []string; Province []string; StreetAddress []string; PostalCode []string; SerialNumber string; CommonName string; Names []@"crypto/x509/pkix".AttributeTypeAndValue } + func (@"crypto/x509/pkix".n·1 *@"crypto/x509/pkix".Name) FillFromRDNSequence (@"crypto/x509/pkix".rdns·2 *@"crypto/x509/pkix".RDNSequence "esc:0x0") + func (@"crypto/x509/pkix".n·2 @"crypto/x509/pkix".Name) ToRDNSequence () (@"crypto/x509/pkix".ret·1 @"crypto/x509/pkix".RDNSequence) + type @"time".zone struct { @"time".name string; @"time".offset int; @"time".isDST bool } + type @"time".zoneTrans struct { @"time".when int64; @"time".index uint8; @"time".isstd bool; @"time".isutc bool } + type @"time".Location struct { @"time".name string; @"time".zone []@"time".zone; @"time".tx []@"time".zoneTrans; @"time".cacheStart int64; @"time".cacheEnd int64; @"time".cacheZone *@"time".zone } + func (@"time".l·2 *@"time".Location "esc:0x0") String () (? string) + func (@"time".l·2 *@"time".Location "esc:0x0") @"time".firstZoneUsed () (? bool) + func (@"time".l·2 *@"time".Location "esc:0x2") @"time".get () (? *@"time".Location) + func (@"time".l·6 *@"time".Location "esc:0x1") @"time".lookup (@"time".sec·7 int64) (@"time".name·1 string, @"time".offset·2 int, @"time".isDST·3 bool, @"time".start·4 int64, @"time".end·5 int64) + func (@"time".l·2 *@"time".Location "esc:0x0") @"time".lookupFirstZone () (? int) + func (@"time".l·4 *@"time".Location "esc:0x0") @"time".lookupName (@"time".name·5 string "esc:0x0", @"time".unix·6 int64) (@"time".offset·1 int, @"time".isDST·2 bool, @"time".ok·3 bool) + type @"time".Duration int64 + func (@"time".d·2 @"time".Duration) Hours () (? float64) { var @"time".hour·3 @"time".Duration; ; @"time".hour·3 = @"time".d·2 / @"time".Duration(0x34630B8A000); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0x34630B8A000); return float64(@"time".hour·3) + float64(@"time".nsec·4) * 0x9C5FFF26ED75Fp-93 } + func (@"time".d·2 @"time".Duration) Minutes () (? float64) { var @"time".min·3 @"time".Duration; ; @"time".min·3 = @"time".d·2 / @"time".Duration(0xDF8475800); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0xDF8475800); return float64(@"time".min·3) + float64(@"time".nsec·4) * 0x9299FF347E9E9p-87 } + func (@"time".d·2 @"time".Duration) Nanoseconds () (? int64) { return int64(@"time".d·2) } + func (@"time".d·2 @"time".Duration) Seconds () (? float64) { var @"time".sec·3 @"time".Duration; ; @"time".sec·3 = @"time".d·2 / @"time".Duration(0x3B9ACA00); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0x3B9ACA00); return float64(@"time".sec·3) + float64(@"time".nsec·4) * 0x112E0BE826D695p-82 } + func (@"time".d·2 @"time".Duration) String () (? string) + type @"time".Month int + func (@"time".m·2 @"time".Month) String () (? string) { return @"time".months[@"time".m·2 - @"time".Month(0x1)] } + type @"time".Weekday int + func (@"time".d·2 @"time".Weekday) String () (? string) { return @"time".days[@"time".d·2] } + type @"time".Time struct { @"time".sec int64; @"time".nsec int32; @"time".loc *@"time".Location } + func (@"time".t·2 @"time".Time "esc:0x2") Add (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x2") AddDate (@"time".years·3 int, @"time".months·4 int, @"time".days·5 int) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") After (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec > @"time".u·3.@"time".sec || @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec > @"time".u·3.@"time".nsec } + func (@"time".t·2 @"time".Time "esc:0x0") Before (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec < @"time".u·3.@"time".sec || @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec < @"time".u·3.@"time".nsec } + func (@"time".t·4 @"time".Time "esc:0x0") Clock () (@"time".hour·1 int, @"time".min·2 int, @"time".sec·3 int) + func (@"time".t·4 @"time".Time "esc:0x0") Date () (@"time".year·1 int, @"time".month·2 @"time".Month, @"time".day·3 int) + func (@"time".t·2 @"time".Time "esc:0x0") Day () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") Equal (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec == @"time".u·3.@"time".nsec } + func (@"time".t·2 @"time".Time "esc:0x0") Format (@"time".layout·3 string "esc:0x0") (? string) + func (@"time".t·2 *@"time".Time "esc:0x0") GobDecode (@"time".data·3 []byte "esc:0x0") (? error) + func (@"time".t·3 @"time".Time "esc:0x0") GobEncode () (? []byte, ? error) + func (@"time".t·2 @"time".Time "esc:0x0") Hour () (? int) + func (@"time".t·3 @"time".Time "esc:0x0") ISOWeek () (@"time".year·1 int, @"time".week·2 int) + func (@"time".t·2 @"time".Time "esc:0x2") In (@"time".loc·3 *@"time".Location "esc:0x2") (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") IsZero () (? bool) { return @"time".t·2.@"time".sec == 0x0 && @"time".t·2.@"time".nsec == 0x0 } + func (@"time".t·2 @"time".Time "esc:0x2") Local () (? @"time".Time) { @"time".t·2.@"time".loc = @"time".Local; return @"time".t·2 } + func (@"time".t·2 @"time".Time "esc:0x2") Location () (? *@"time".Location) { var @"time".l·3 *@"time".Location; ; @"time".l·3 = @"time".t·2.@"time".loc; if @"time".l·3 == nil { @"time".l·3 = @"time".UTC }; return @"time".l·3 } + func (@"time".t·3 @"time".Time "esc:0x0") MarshalBinary () (? []byte, ? error) + func (@"time".t·3 @"time".Time "esc:0x0") MarshalJSON () (? []byte, ? error) + func (@"time".t·3 @"time".Time "esc:0x0") MarshalText () (? []byte, ? error) + func (@"time".t·2 @"time".Time "esc:0x0") Minute () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") Month () (? @"time".Month) + func (@"time".t·2 @"time".Time "esc:0x0") Nanosecond () (? int) { return int(@"time".t·2.@"time".nsec) } + func (@"time".t·2 @"time".Time "esc:0x2") Round (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") Second () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") String () (? string) + func (@"time".t·2 @"time".Time "esc:0x0") Sub (@"time".u·3 @"time".Time "esc:0x0") (? @"time".Duration) + func (@"time".t·2 @"time".Time "esc:0x2") Truncate (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x2") UTC () (? @"time".Time) { @"time".t·2.@"time".loc = @"time".UTC; return @"time".t·2 } + func (@"time".t·2 @"time".Time "esc:0x0") Unix () (? int64) { return @"time".t·2.@"time".sec + -0xE7791F700 } + func (@"time".t·2 @"time".Time "esc:0x0") UnixNano () (? int64) { return (@"time".t·2.@"time".sec + -0xE7791F700) * 0x3B9ACA00 + int64(@"time".t·2.@"time".nsec) } + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalBinary (@"time".data·3 []byte "esc:0x0") (? error) + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalJSON (@"time".data·3 []byte "esc:0x0") (@"time".err·1 error) + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalText (@"time".data·3 []byte "esc:0x0") (@"time".err·1 error) + func (@"time".t·2 @"time".Time "esc:0x0") Weekday () (? @"time".Weekday) + func (@"time".t·2 @"time".Time "esc:0x0") Year () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") YearDay () (? int) + func (@"time".t·3 @"time".Time "esc:0x0") Zone () (@"time".name·1 string, @"time".offset·2 int) + func (@"time".t·2 @"time".Time "esc:0x0") @"time".abs () (? uint64) + func (@"time".t·5 @"time".Time "esc:0x0") @"time".date (@"time".full·6 bool) (@"time".year·1 int, @"time".month·2 @"time".Month, @"time".day·3 int, @"time".yday·4 int) + func (@"time".t·4 @"time".Time "esc:0x1") @"time".locabs () (@"time".name·1 string, @"time".offset·2 int, @"time".abs·3 uint64) + type @"crypto/x509".KeyUsage int + type @"crypto/x509/pkix".Extension struct { Id @"encoding/asn1".ObjectIdentifier; Critical bool "asn1:\"optional\""; Value []byte } + type @"crypto/x509".ExtKeyUsage int + import net "net" // indirect + type @"net".IPMask []byte + func (@"net".m·3 @"net".IPMask "esc:0x0") Size () (@"net".ones·1 int, @"net".bits·2 int) + func (@"net".m·2 @"net".IPMask "esc:0x0") String () (? string) + type @"net".IP []byte + func (@"net".ip·2 @"net".IP "esc:0x0") DefaultMask () (? @"net".IPMask) + func (@"net".ip·2 @"net".IP "esc:0x0") Equal (@"net".x·3 @"net".IP "esc:0x0") (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsGlobalUnicast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsInterfaceLocalMulticast () (? bool) { return len(@"net".ip·2) == 0x10 && @"net".ip·2[0x0] == byte(0xFF) && @"net".ip·2[0x1] & byte(0xF) == byte(0x1) } + func (@"net".ip·2 @"net".IP "esc:0x0") IsLinkLocalMulticast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsLinkLocalUnicast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsLoopback () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsMulticast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsUnspecified () (? bool) + func (@"net".ip·3 @"net".IP "esc:0x0") MarshalText () (? []byte, ? error) + func (@"net".ip·2 @"net".IP "esc:0x0") Mask (@"net".mask·3 @"net".IPMask "esc:0x0") (? @"net".IP) + func (@"net".ip·2 @"net".IP "esc:0x0") String () (? string) + func (@"net".ip·2 @"net".IP "esc:0x2") To16 () (? @"net".IP) + func (@"net".ip·2 @"net".IP "esc:0x2") To4 () (? @"net".IP) + func (@"net".ip·2 *@"net".IP "esc:0x0") UnmarshalText (@"net".text·3 []byte "esc:0x0") (? error) + type @"encoding/asn1".RawContent []byte + type @"encoding/asn1".RawValue struct { Class int; Tag int; IsCompound bool; Bytes []byte; FullBytes []byte } + type @"crypto/x509/pkix".AlgorithmIdentifier struct { Algorithm @"encoding/asn1".ObjectIdentifier; Parameters @"encoding/asn1".RawValue "asn1:\"optional\"" } + type @"crypto/x509/pkix".RevokedCertificate struct { SerialNumber *@"math/big".Int; RevocationTime @"time".Time; Extensions []@"crypto/x509/pkix".Extension "asn1:\"optional\"" } + type @"crypto/x509/pkix".TBSCertificateList struct { Raw @"encoding/asn1".RawContent; Version int "asn1:\"optional,default:2\""; Signature @"crypto/x509/pkix".AlgorithmIdentifier; Issuer @"crypto/x509/pkix".RDNSequence; ThisUpdate @"time".Time; NextUpdate @"time".Time "asn1:\"optional\""; RevokedCertificates []@"crypto/x509/pkix".RevokedCertificate "asn1:\"optional\""; Extensions []@"crypto/x509/pkix".Extension "asn1:\"tag:0,optional,explicit\"" } + type @"encoding/asn1".BitString struct { Bytes []byte; BitLength int } + func (@"encoding/asn1".b·2 @"encoding/asn1".BitString "esc:0x0") At (@"encoding/asn1".i·3 int) (? int) { if @"encoding/asn1".i·3 < 0x0 || @"encoding/asn1".i·3 >= @"encoding/asn1".b·2.BitLength { return 0x0 }; var @"encoding/asn1".x·4 int; ; @"encoding/asn1".x·4 = @"encoding/asn1".i·3 / 0x8; var @"encoding/asn1".y·5 uint; ; @"encoding/asn1".y·5 = 0x7 - uint(@"encoding/asn1".i·3 % 0x8); return int(@"encoding/asn1".b·2.Bytes[@"encoding/asn1".x·4] >> @"encoding/asn1".y·5) & 0x1 } + func (@"encoding/asn1".b·2 @"encoding/asn1".BitString "esc:0x2") RightAlign () (? []byte) + type @"crypto/x509/pkix".CertificateList struct { TBSCertList @"crypto/x509/pkix".TBSCertificateList; SignatureAlgorithm @"crypto/x509/pkix".AlgorithmIdentifier; SignatureValue @"encoding/asn1".BitString } + func (@"crypto/x509/pkix".certList·2 *@"crypto/x509/pkix".CertificateList "esc:0x0") HasExpired (@"crypto/x509/pkix".now·3 @"time".Time "esc:0x0") (? bool) + type @"io".Reader interface { Read(@"io".p []byte) (@"io".n int, @"io".err error) } + type @"crypto/x509".CertPool struct { @"crypto/x509".bySubjectKeyId map[string][]int; @"crypto/x509".byName map[string][]int; @"crypto/x509".certs []*@"crypto/x509".Certificate } + func (@"crypto/x509".s·1 *@"crypto/x509".CertPool) AddCert (@"crypto/x509".cert·2 *@"crypto/x509".Certificate) + func (@"crypto/x509".s·2 *@"crypto/x509".CertPool) AppendCertsFromPEM (@"crypto/x509".pemCerts·3 []byte) (@"crypto/x509".ok·1 bool) + func (@"crypto/x509".s·2 *@"crypto/x509".CertPool "esc:0x0") Subjects () (@"crypto/x509".res·1 [][]byte) + func (@"crypto/x509".s·4 *@"crypto/x509".CertPool "esc:0x0") @"crypto/x509".findVerifiedParents (@"crypto/x509".cert·5 *@"crypto/x509".Certificate) (@"crypto/x509".parents·1 []int, @"crypto/x509".errCert·2 *@"crypto/x509".Certificate, @"crypto/x509".err·3 error) + type @"crypto/x509".VerifyOptions struct { DNSName string; Intermediates *@"crypto/x509".CertPool; Roots *@"crypto/x509".CertPool; CurrentTime @"time".Time; KeyUsages []@"crypto/x509".ExtKeyUsage } + type @"crypto/x509".Certificate struct { Raw []byte; RawTBSCertificate []byte; RawSubjectPublicKeyInfo []byte; RawSubject []byte; RawIssuer []byte; Signature []byte; SignatureAlgorithm @"crypto/x509".SignatureAlgorithm; PublicKeyAlgorithm @"crypto/x509".PublicKeyAlgorithm; PublicKey interface {}; Version int; SerialNumber *@"math/big".Int; Issuer @"crypto/x509/pkix".Name; Subject @"crypto/x509/pkix".Name; NotBefore @"time".Time; NotAfter @"time".Time; KeyUsage @"crypto/x509".KeyUsage; Extensions []@"crypto/x509/pkix".Extension; ExtraExtensions []@"crypto/x509/pkix".Extension; ExtKeyUsage []@"crypto/x509".ExtKeyUsage; UnknownExtKeyUsage []@"encoding/asn1".ObjectIdentifier; BasicConstraintsValid bool; IsCA bool; MaxPathLen int; MaxPathLenZero bool; SubjectKeyId []byte; AuthorityKeyId []byte; OCSPServer []string; IssuingCertificateURL []string; DNSNames []string; EmailAddresses []string; IPAddresses []@"net".IP; PermittedDNSDomainsCritical bool; PermittedDNSDomains []string; CRLDistributionPoints []string; PolicyIdentifiers []@"encoding/asn1".ObjectIdentifier } + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate) CheckCRLSignature (@"crypto/x509".crl·3 *@"crypto/x509/pkix".CertificateList) (@"crypto/x509".err·1 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate) CheckSignature (@"crypto/x509".algo·3 @"crypto/x509".SignatureAlgorithm, @"crypto/x509".signed·4 []byte, @"crypto/x509".signature·5 []byte) (@"crypto/x509".err·1 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate) CheckSignatureFrom (@"crypto/x509".parent·3 *@"crypto/x509".Certificate) (@"crypto/x509".err·1 error) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate) CreateCRL (@"crypto/x509".rand·4 @"io".Reader, @"crypto/x509".priv·5 interface {}, @"crypto/x509".revokedCerts·6 []@"crypto/x509/pkix".RevokedCertificate, @"crypto/x509".now·7 @"time".Time, @"crypto/x509".expiry·8 @"time".Time) (@"crypto/x509".crlBytes·1 []byte, @"crypto/x509".err·2 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate "esc:0x0") Equal (@"crypto/x509".other·3 *@"crypto/x509".Certificate "esc:0x0") (? bool) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate) Verify (@"crypto/x509".opts·4 @"crypto/x509".VerifyOptions "esc:0x4") (@"crypto/x509".chains·1 [][]*@"crypto/x509".Certificate, @"crypto/x509".err·2 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate "esc:0x2") VerifyHostname (@"crypto/x509".h·3 string "esc:0x2") (? error) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate) @"crypto/x509".buildChains (@"crypto/x509".cache·4 map[int][][]*@"crypto/x509".Certificate "esc:0x0", @"crypto/x509".currentChain·5 []*@"crypto/x509".Certificate "esc:0x0", @"crypto/x509".opts·6 *@"crypto/x509".VerifyOptions "esc:0x0") (@"crypto/x509".chains·1 [][]*@"crypto/x509".Certificate, @"crypto/x509".err·2 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate "esc:0x2") @"crypto/x509".isValid (@"crypto/x509".certType·3 int, @"crypto/x509".currentChain·4 []*@"crypto/x509".Certificate "esc:0x0", @"crypto/x509".opts·5 *@"crypto/x509".VerifyOptions "esc:0x0") (? error) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate "esc:0x0") @"crypto/x509".systemVerify (@"crypto/x509".opts·4 *@"crypto/x509".VerifyOptions "esc:0x0") (@"crypto/x509".chains·1 [][]*@"crypto/x509".Certificate, @"crypto/x509".err·2 error) { return nil, nil } + type @"crypto/tls".ConnectionState struct { Version uint16; HandshakeComplete bool; DidResume bool; CipherSuite uint16; NegotiatedProtocol string; NegotiatedProtocolIsMutual bool; ServerName string; PeerCertificates []*@"crypto/x509".Certificate; VerifiedChains [][]*@"crypto/x509".Certificate; TLSUnique []byte } + type @"net/http".Cookie struct { Name string; Value string; Path string; Domain string; Expires @"time".Time; RawExpires string; MaxAge int; Secure bool; HttpOnly bool; Raw string; Unparsed []string } + func (@"net/http".c·2 *@"net/http".Cookie) String () (? string) + import bufio "bufio" // indirect + type @"bufio".Reader struct { @"bufio".buf []byte; @"bufio".rd @"io".Reader; @"bufio".r int; @"bufio".w int; @"bufio".err error; @"bufio".lastByte int; @"bufio".lastRuneSize int } + func (@"bufio".b·2 *@"bufio".Reader "esc:0x0") Buffered () (? int) { return @"bufio".b·2.@"bufio".w - @"bufio".b·2.@"bufio".r } + func (@"bufio".b·3 *@"bufio".Reader) Peek (@"bufio".n·4 int) (? []byte, ? error) + func (@"bufio".b·3 *@"bufio".Reader) Read (@"bufio".p·4 []byte) (@"bufio".n·1 int, @"bufio".err·2 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadByte () (@"bufio".c·1 byte, @"bufio".err·2 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadBytes (@"bufio".delim·4 byte) (@"bufio".line·1 []byte, @"bufio".err·2 error) + func (@"bufio".b·4 *@"bufio".Reader) ReadLine () (@"bufio".line·1 []byte, @"bufio".isPrefix·2 bool, @"bufio".err·3 error) + func (@"bufio".b·4 *@"bufio".Reader) ReadRune () (@"bufio".r·1 rune, @"bufio".size·2 int, @"bufio".err·3 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadSlice (@"bufio".delim·4 byte) (@"bufio".line·1 []byte, @"bufio".err·2 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadString (@"bufio".delim·4 byte) (@"bufio".line·1 string, @"bufio".err·2 error) + func (@"bufio".b·1 *@"bufio".Reader) Reset (@"bufio".r·2 @"io".Reader) + func (@"bufio".b·2 *@"bufio".Reader "esc:0x0") UnreadByte () (? error) + func (@"bufio".b·2 *@"bufio".Reader "esc:0x0") UnreadRune () (? error) { if @"bufio".b·2.@"bufio".lastRuneSize < 0x0 || @"bufio".b·2.@"bufio".r < @"bufio".b·2.@"bufio".lastRuneSize { return @"bufio".ErrInvalidUnreadRune }; @"bufio".b·2.@"bufio".r -= @"bufio".b·2.@"bufio".lastRuneSize; @"bufio".b·2.@"bufio".lastByte = -0x1; @"bufio".b·2.@"bufio".lastRuneSize = -0x1; return nil } + func (@"bufio".b·3 *@"bufio".Reader) WriteTo (@"bufio".w·4 @"io".Writer) (@"bufio".n·1 int64, @"bufio".err·2 error) + func (@"bufio".b·1 *@"bufio".Reader) @"bufio".fill () + func (@"bufio".b·2 *@"bufio".Reader "esc:0x1") @"bufio".readErr () (? error) { var @"bufio".err·3 error; ; @"bufio".err·3 = @"bufio".b·2.@"bufio".err; @"bufio".b·2.@"bufio".err = nil; return @"bufio".err·3 } + func (@"bufio".b·1 *@"bufio".Reader "esc:0x0") @"bufio".reset (@"bufio".buf·2 []byte, @"bufio".r·3 @"io".Reader) { *@"bufio".b·1 = (@"bufio".Reader{ @"bufio".buf:@"bufio".buf·2, @"bufio".rd:@"bufio".r·3, @"bufio".lastByte:-0x1, @"bufio".lastRuneSize:-0x1 }) } + func (@"bufio".b·3 *@"bufio".Reader) @"bufio".writeBuf (@"bufio".w·4 @"io".Writer) (? int64, ? error) + import bytes "bytes" // indirect + type @"bytes".readOp int + type @"bytes".Buffer struct { @"bytes".buf []byte; @"bytes".off int; @"bytes".runeBytes [4]byte; @"bytes".bootstrap [64]byte; @"bytes".lastRead @"bytes".readOp } + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x1") Bytes () (? []byte) { return @"bytes".b·2.@"bytes".buf[@"bytes".b·2.@"bytes".off:] } + func (@"bytes".b·1 *@"bytes".Buffer) Grow (@"bytes".n·2 int) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") Len () (? int) { return len(@"bytes".b·2.@"bytes".buf) - @"bytes".b·2.@"bytes".off } + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x1") Next (@"bytes".n·3 int) (? []byte) + func (@"bytes".b·3 *@"bytes".Buffer) Read (@"bytes".p·4 []byte "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) ReadByte () (@"bytes".c·1 byte, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x0") ReadBytes (@"bytes".delim·4 byte) (@"bytes".line·1 []byte, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) ReadFrom (@"bytes".r·4 @"io".Reader) (@"bytes".n·1 int64, @"bytes".err·2 error) + func (@"bytes".b·4 *@"bytes".Buffer) ReadRune () (@"bytes".r·1 rune, @"bytes".size·2 int, @"bytes".err·3 error) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x0") ReadString (@"bytes".delim·4 byte) (@"bytes".line·1 string, @"bytes".err·2 error) + func (@"bytes".b·1 *@"bytes".Buffer) Reset () + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") String () (? string) { if @"bytes".b·2 == nil { return "" }; return string(@"bytes".b·2.@"bytes".buf[@"bytes".b·2.@"bytes".off:]) } + func (@"bytes".b·1 *@"bytes".Buffer) Truncate (@"bytes".n·2 int) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") UnreadByte () (? error) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") UnreadRune () (? error) + func (@"bytes".b·3 *@"bytes".Buffer) Write (@"bytes".p·4 []byte "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·2 *@"bytes".Buffer) WriteByte (@"bytes".c·3 byte) (? error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteRune (@"bytes".r·4 rune) (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteString (@"bytes".s·4 string "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteTo (@"bytes".w·4 @"io".Writer) (@"bytes".n·1 int64, @"bytes".err·2 error) + func (@"bytes".b·2 *@"bytes".Buffer) @"bytes".grow (@"bytes".n·3 int) (? int) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x1") @"bytes".readSlice (@"bytes".delim·4 byte) (@"bytes".line·1 []byte, @"bytes".err·2 error) + type @"mime/multipart".Part struct { Header @"net/textproto".MIMEHeader; @"mime/multipart".buffer *@"bytes".Buffer; @"mime/multipart".mr *@"mime/multipart".Reader; @"mime/multipart".bytesRead int; @"mime/multipart".disposition string; @"mime/multipart".dispositionParams map[string]string; @"mime/multipart".r @"io".Reader } + func (@"mime/multipart".p·2 *@"mime/multipart".Part) Close () (? error) + func (@"mime/multipart".p·2 *@"mime/multipart".Part "esc:0x0") FileName () (? string) + func (@"mime/multipart".p·2 *@"mime/multipart".Part "esc:0x0") FormName () (? string) + func (@"mime/multipart".p·3 *@"mime/multipart".Part) Read (@"mime/multipart".d·4 []byte) (@"mime/multipart".n·1 int, @"mime/multipart".err·2 error) + func (@"mime/multipart".p·1 *@"mime/multipart".Part "esc:0x0") @"mime/multipart".parseContentDisposition () + func (@"mime/multipart".bp·2 *@"mime/multipart".Part) @"mime/multipart".populateHeaders () (? error) + type @"mime/multipart".Reader struct { @"mime/multipart".bufReader *@"bufio".Reader; @"mime/multipart".currentPart *@"mime/multipart".Part; @"mime/multipart".partsRead int; @"mime/multipart".nl []byte; @"mime/multipart".nlDashBoundary []byte; @"mime/multipart".dashBoundaryDash []byte; @"mime/multipart".dashBoundary []byte } + func (@"mime/multipart".r·3 *@"mime/multipart".Reader) NextPart () (? *@"mime/multipart".Part, ? error) + func (@"mime/multipart".r·3 *@"mime/multipart".Reader) ReadForm (@"mime/multipart".maxMemory·4 int64) (@"mime/multipart".f·1 *@"mime/multipart".Form, @"mime/multipart".err·2 error) + func (@"mime/multipart".mr·2 *@"mime/multipart".Reader) @"mime/multipart".isBoundaryDelimiterLine (@"mime/multipart".line·3 []byte "esc:0x0") (@"mime/multipart".ret·1 bool) + func (@"mime/multipart".mr·2 *@"mime/multipart".Reader "esc:0x0") @"mime/multipart".isFinalBoundary (@"mime/multipart".line·3 []byte "esc:0x0") (? bool) + func (@"mime/multipart".mr·2 *@"mime/multipart".Reader "esc:0x0") @"mime/multipart".peekBufferIsEmptyPart (@"mime/multipart".peek·3 []byte "esc:0x0") (? bool) + type @"net/http".Request struct { Method string; URL *@"net/url".URL; Proto string; ProtoMajor int; ProtoMinor int; Header @"net/http".Header; Body @"io".ReadCloser; ContentLength int64; TransferEncoding []string; Close bool; Host string; Form @"net/url".Values; PostForm @"net/url".Values; MultipartForm *@"mime/multipart".Form; Trailer @"net/http".Header; RemoteAddr string; RequestURI string; TLS *@"crypto/tls".ConnectionState } + func (@"net/http".r·1 *@"net/http".Request "esc:0x0") AddCookie (@"net/http".c·2 *@"net/http".Cookie) + func (@"net/http".r·4 *@"net/http".Request "esc:0x0") BasicAuth () (@"net/http".username·1 string, @"net/http".password·2 string, @"net/http".ok·3 bool) + func (@"net/http".r·3 *@"net/http".Request "esc:0x0") Cookie (@"net/http".name·4 string "esc:0x0") (? *@"net/http".Cookie, ? error) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") Cookies () (? []*@"net/http".Cookie) + func (@"net/http".r·4 *@"net/http".Request) FormFile (@"net/http".key·5 string "esc:0x0") (? @"mime/multipart".File, ? *@"mime/multipart".FileHeader, ? error) + func (@"net/http".r·2 *@"net/http".Request) FormValue (@"net/http".key·3 string "esc:0x0") (? string) + func (@"net/http".r·3 *@"net/http".Request) MultipartReader () (? *@"mime/multipart".Reader, ? error) + func (@"net/http".r·2 *@"net/http".Request) ParseForm () (? error) + func (@"net/http".r·2 *@"net/http".Request) ParseMultipartForm (@"net/http".maxMemory·3 int64) (? error) + func (@"net/http".r·2 *@"net/http".Request) PostFormValue (@"net/http".key·3 string "esc:0x0") (? string) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") ProtoAtLeast (@"net/http".major·3 int, @"net/http".minor·4 int) (? bool) { return @"net/http".r·2.ProtoMajor > @"net/http".major·3 || @"net/http".r·2.ProtoMajor == @"net/http".major·3 && @"net/http".r·2.ProtoMinor >= @"net/http".minor·4 } + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") Referer () (? string) + func (@"net/http".r·1 *@"net/http".Request "esc:0x0") SetBasicAuth (@"net/http".username·2 string "esc:0x0", @"net/http".password·3 string "esc:0x0") + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") UserAgent () (? string) + func (@"net/http".r·2 *@"net/http".Request) Write (@"net/http".w·3 @"io".Writer) (? error) + func (@"net/http".r·2 *@"net/http".Request) WriteProxy (@"net/http".w·3 @"io".Writer) (? error) + func (@"net/http".r·1 *@"net/http".Request) @"net/http".closeBody () + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") @"net/http".expectsContinue () (? bool) + func (@"net/http".r·3 *@"net/http".Request) @"net/http".multipartReader () (? *@"mime/multipart".Reader, ? error) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") @"net/http".wantsClose () (? bool) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") @"net/http".wantsHttp10KeepAlive () (? bool) + func (@"net/http".req·2 *@"net/http".Request) @"net/http".write (@"net/http".w·3 @"io".Writer, @"net/http".usingProxy·4 bool, @"net/http".extraHeaders·5 @"net/http".Header "esc:0x0") (? error) + func @"".Set (@"".r·1 *@"net/http".Request, @"".key·2 interface {}, @"".val·3 interface {}) + func @"".Get (@"".r·2 *@"net/http".Request "esc:0x0", @"".key·3 interface {} "esc:0x0") (? interface {}) + func @"".GetOk (@"".r·3 *@"net/http".Request "esc:0x0", @"".key·4 interface {} "esc:0x0") (? interface {}, ? bool) + func @"".GetAll (@"".r·2 *@"net/http".Request "esc:0x0") (? map[interface {}]interface {}) + func @"".GetAllOk (@"".r·3 *@"net/http".Request "esc:0x0") (? map[interface {}]interface {}, ? bool) + func @"".Delete (@"".r·1 *@"net/http".Request "esc:0x0", @"".key·2 interface {} "esc:0x0") + func @"".Clear (@"".r·1 *@"net/http".Request "esc:0x0") + func @"".Purge (@"".maxAge·2 int) (? int) + type @"net/http".ResponseWriter interface { Header() (? @"net/http".Header); Write(? []byte) (? int, ? error); WriteHeader(? int) } + type @"net/http".Handler interface { ServeHTTP(? @"net/http".ResponseWriter, ? *@"net/http".Request) } + func @"".ClearHandler (@"".h·2 @"net/http".Handler) (? @"net/http".Handler) + func @"".init () + var @"time".months [12]string + var @"time".days [7]string + var @"time".Local *@"time".Location + var @"time".UTC *@"time".Location + var @"bufio".ErrInvalidUnreadRune error + +$$ +_go_.6 0 0 0 644 49111 ` +go object darwin amd64 go1.4.2 X:precisestack + +! +go13ldnet/http.a sync.a time.a� "".Set��eH� %H;aw���H��pH�H�$�H�D$xH�H�$H�H�\$H�D$�H�\$H�1�H9��iH�\$xH�\$0H�H�$H�D$�H�\$H�\$(H�H�$H�H�\$H�\$0H�\$H�\$(H�\$�H�\$xH�\$0�H�$�L$H�D$�L$`H�D$hH�\$XH� n�����H�H�\$ H�H�$H�H�\$H�\$0H�\$H�\$ H�\$�H�D$xH�H�$H�H�\$H�D$�H�\$H�+H��$�H�\$HH��$�H�\$PH��$�H�\$8H��$�H�\$@H�H�$H�l$H�\$HH�\$H�\$8H�\$�H�H�$�H��p��[���. + 0runtime.morestack_noctxt:"".mutexL(sync.(*RWMutex).Lockdptype.map[*net/http.Request]map[interface {}]interface {}z"".data�2runtime.mapaccess1_fast64�Dtype.map[interface {}]interface {}�runtime.makemap�ptype.map[*net/http.Request]map[interface {}]interface {}�"".data�$runtime.mapassign1�time.Now�@type.map[*net/http.Request]int64�"".datat�$runtime.mapassign1�ptype.map[*net/http.Request]map[interface {}]interface {}�"".data�2runtime.mapaccess1_fast64�Dtype.map[interface {}]interface {}�$runtime.mapassign1�"".mutex�,sync.(*RWMutex).UnlockP�"".autotmp_0011o"type.interface {}"".autotmp_0010O"type.interface {}"".autotmp_0009Dtype.map[interface {}]interface {}"".autotmp_0008,type.*net/http.Request"".autotmp_0007�type.int64"".autotmp_0005,type.*net/http.Request"".autotmp_0004�Dtype.map[interface {}]interface {}"".autotmp_0003,type.*net/http.Requesttime.t·2/type.time.Time "".val0"type.interface {} "".key"type.interface {}"".r,type.*net/http.Request�����(*9]l�$%\:�eTgclocals·1245bf52b89d91ace0efa42912a5da62Tgclocals·341ab1f3c66e663a93f22f10d9eab46d�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go� "".Get��eH� %H;aw���H��@H�D$`H�D$hH�H�$�H�D$HH�H�$H�H�\$H�D$�H�\$H�1�H9�t|H�\$PH�\$0H�\$XH�\$8H�H�$H�D$H�\$0H�\$�H�\$H��t:H� H�kH�L$ H�l$(H�H�$�H�\$ H�\$`H�\$(H�\$hH��@É��H�H�$�H�D$`H�D$hH��@� + 0runtime.morestack_noctxt^"".mutexp*sync.(*RWMutex).RLock�ptype.map[*net/http.Request]map[interface {}]interface {}�"".data�2runtime.mapaccess1_fast64�Dtype.map[interface {}]interface {}�$runtime.mapaccess1�"".mutex�.sync.(*RWMutex).RUnlock�"".mutex�.sync.(*RWMutex).RUnlockP� +"".autotmp_0015"type.interface {}"".value?"type.interface {} "".~r20"type.interface {} "".key"type.interface {}"".r,type.*net/http.Request���* �&@,5O#7h,-(Tgclocals·473d4314ba155bc5d9af9ad66f1c242aTgclocals·4ac5edf299a8cadc46c808258e6fc6d0�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go�"".GetOk��eH� %H;aw���H��PH�D$pH�D$xH�H�$�H�D$XH�H�$H�H�\$H�D$�H�L$�\$ H��������H�D$XH�H�$H�H�\$H�D$�H�\$H�+H�\$`H�\$@H�\$hH�\$HH�H�$H�l$H�\$@H�\$�H�D$�\$ �\$/H��tFH�(H�l$0H�hH�l$8H�H�$�H�\$0H�\$pH�\$8H�\$x�\$/��$�H��PÉ�H�H�$�H�D$pH�D$xƄ$�H��PÉ���� + 0runtime.morestack_noctxt^"".mutexp*sync.(*RWMutex).RLock�ptype.map[*net/http.Request]map[interface {}]interface {}�"".data�2runtime.mapaccess2_fast64�ptype.map[*net/http.Request]map[interface {}]interface {}�"".data�2runtime.mapaccess1_fast64�Dtype.map[interface {}]interface {}�$runtime.mapaccess2�"".mutex�.sync.(*RWMutex).RUnlock�"".mutex�.sync.(*RWMutex).RUnlock`�"".autotmp_0020"type.interface {}"".autotmp_0018,type.*net/http.Request +"".okAtype.bool"".value?"type.interface {} "".~r3Ptype.bool "".~r20"type.interface {} "".key"type.interface {}"".r,type.*net/http.Request&����2���,X,C�% 7�597Tgclocals·a5bd3918675a24472245dfa9083d6deeTgclocals·4ac5edf299a8cadc46c808258e6fc6d0�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go�"".GetAll��eH� %H�D$�H;Aw���H���H�H�$�H��$�H�H�$H�H�\$H�D$�H�L$�\$ H�)���hH��H�l$0H��tH�H��H�H�$H�D$�H�\$H�\$(H�L$0H��$�1��H�H�$H�L$H��$�H�\$�H��$�1�H9���H��$�H����H� H�CH��$�H����H�3H�kH�L$XH�D$`H�t$HH�t$xH�l$PH��$�H�L$8H�L$hH�D$@H�D$pH�H�$H�\$(H�\$H�\$xH�\$H�\$hH�\$�H��$�H�$�H��$�1�H9��G���H�H�$�H�\$(H��$�H���É�F�����&���H�H�$�HDŽ$�H����& +*0runtime.morestack_noctxtJ"".mutex\*sync.(*RWMutex).RLockzptype.map[*net/http.Request]map[interface {}]interface {}�"".data�2runtime.mapaccess2_fast64�Dtype.map[interface {}]interface {}�runtime.makemap�� runtime.duffzero�Dtype.map[interface {}]interface {}�&runtime.mapiterinit�Dtype.map[interface {}]interface {}�$runtime.mapassign1�&runtime.mapiternext�"".mutex�.sync.(*RWMutex).RUnlock�"".mutex�.sync.(*RWMutex).RUnlock �"".autotmp_0031�"type.interface {}"".autotmp_0029�"type.interface {}"".autotmp_0028�"type.interface {}"".autotmp_0027�Ntype.map.iter[interface {}]interface {}"".autotmp_0026Dtype.map[interface {}]interface {}"".v�"type.interface {}"".k�"type.interface {}"".result�Dtype.map[interface {}]interface {}"".context�Dtype.map[interface {}]interface {} "".~r1Dtype.map[interface {}]interface {}"".r,type.*net/http.Request "����1��4p"?3�Y$ $-h@�#3Tgclocals·7ba969af8c72fca351526f5bd553df36Tgclocals·af2d3162727927f9557359011162d41c�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go�"".GetAllOk��eH� %H�D$�H;Aw���H���H�H�$�H��$�H�H�$H�H�\$H�D$�H�D$�\$ �\$/H�H�\$8H��tH�H��H�H�$H�D$�H�\$H�\$0H�L$8H��$�1��H�H�$H�L$H��$�H�\$�H��$�1�H9���H��$�H����H� H�CH��$�H����H�3H�kH�L$`H�D$hH�t$PH��$�H�l$XH��$�H�L$@H�L$pH�D$HH�D$xH�H�$H�\$0H�\$H��$�H�\$H�\$pH�\$�H��$�H�$�H��$�1�H9��A���H�H�$�H�\$0H��$��\$/��$�H���É�4��������" +*0runtime.morestack_noctxtJ"".mutex\*sync.(*RWMutex).RLockzptype.map[*net/http.Request]map[interface {}]interface {}�"".data�2runtime.mapaccess2_fast64�Dtype.map[interface {}]interface {}�runtime.makemap�� runtime.duffzero�Dtype.map[interface {}]interface {}�&runtime.mapiterinit�Dtype.map[interface {}]interface {}�$runtime.mapassign1�&runtime.mapiternext�"".mutex�.sync.(*RWMutex).RUnlock0�"".autotmp_0040�"type.interface {}"".autotmp_0038�"type.interface {}"".autotmp_0037�"type.interface {}"".autotmp_0036�Ntype.map.iter[interface {}]interface {}"".autotmp_0035Dtype.map[interface {}]interface {}"".v�"type.interface {}"".k�"type.interface {}"".result�Dtype.map[interface {}]interface {} +"".ok�type.bool"".context�Dtype.map[interface {}]interface {} "".~r2 type.bool "".~r1Dtype.map[interface {}]interface {}"".r,type.*net/http.Request"�����.�":0�_$! -`@�#ATgclocals·b46c7a32cd3cbdb99d262657bbb5cb46Tgclocals·af2d3162727927f9557359011162d41c�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go�"".Delete��eH� %H;aw���H��0H�H�$�H�D$8H�H�$H�H�\$H�D$�H�\$H�1�H9�tvH�D$8H�H�$H�H�\$H�D$�H�\$H�+H�\$@H�\$ H�\$HH�\$(H�H�$H�l$H�\$ H�\$�H�H�$�H��0��� + 0runtime.morestack_noctxt:"".mutexL(sync.(*RWMutex).Lockdptype.map[*net/http.Request]map[interface {}]interface {}z"".data�2runtime.mapaccess1_fast64�ptype.map[*net/http.Request]map[interface {}]interface {}�"".data�2runtime.mapaccess1_fast64�Dtype.map[interface {}]interface {}�"runtime.mapdelete�"".mutex�,sync.(*RWMutex).Unlock0`"".autotmp_0045"type.interface {} "".key"type.interface {}"".r,type.*net/http.Request`�_` � �5a %�Tgclocals·e6f65f4d24a282ff6c0e03aa9f867bfeTgclocals·d9578cf05e73f94c5bc1acfa30cff71f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go�"".Clear��eH� %H;aw���H��(H�H�$�H�D$0H�D$H�D$ H�H�$H�H�\$H�\$ H�\$�H�\$H�\$ H�H�$H�H�\$H�\$ H�\$�H�H�$�H��(� + 0runtime.morestack_noctxt:"".mutexL(sync.(*RWMutex).Lockxptype.map[*net/http.Request]map[interface {}]interface {}�"".data�"runtime.mapdelete�@type.map[*net/http.Request]int64�"".datat�"runtime.mapdelete�"".mutex�,sync.(*RWMutex).UnlockP"".autotmp_0047,type.*net/http.Request"".autotmp_0046,type.*net/http.Request"".r,type.*net/http.Request"".r,type.*net/http.RequestP�O ��e%50Tgclocals·15395a9df917b4c9aa74d5c6c7e1ebf4Tgclocals·ab0f5354c6e12d990f77504eee3efe59�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go�"".clear��eH� %H;aw���H�� H�\$(H�\$H�H�$H�H�\$H�\$H�\$�H�\$(H�\$H�H�$H�H�\$H�\$H�\$�H�� � + 0runtime.morestack_noctxtNptype.map[*net/http.Request]map[interface {}]interface {}d"".data�"runtime.mapdelete�@type.map[*net/http.Request]int64�"".datat�"runtime.mapdelete@"".autotmp_0049,type.*net/http.Request"".autotmp_0048,type.*net/http.Request"".r,type.*net/http.Request@d?��00 +E;Tgclocals·ac5bea9c8a91f5fb1d31bdacc5067b57Tgclocals·e1ae6533a9e39048ba0735a2264ce16a�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go�"".Purge� � eH� %H�D$�H;Aw���H��H�H�$�H�D$(H��$�H����H�H��tH�H�\$(H�H�$H�D$�H�D$H�H�$H�D$�H�H�$H�D$�H�D$H�H�$H�D$�H�H�$�H�\$(H��$�H�İ��H�$�L$H�D$�L$PH�D$XH�\$HH� n�����H�H��$�H)�H�\$ H� H�|$`1��H�H�$H�L$H�\$`H�\$�H�\$`1�H9���H�\$`H�+H�l$0H�H�$H�H�\$H�l$�H�\$H�H�l$ H9���H�D$0H�D$8H�D$@H�H�$H�H�\$H�\$@H�\$�H�\$8H�\$@H�H�$H�H�\$H�\$@H�\$�H�\$(H��H�\$(H�\$`H�$�H�\$`1�H9��,���������< +*0runtime.morestack_noctxtJ"".mutex\(sync.(*RWMutex).Lock�"".data�ptype.map[*net/http.Request]map[interface {}]interface {}�runtime.makemap�"".data�.runtime.writebarrierptr�@type.map[*net/http.Request]int64�runtime.makemap�"".datat�.runtime.writebarrierptr�"".mutex�,sync.(*RWMutex).Unlock�time.Now�"".data�� runtime.duffzero�ptype.map[*net/http.Request]map[interface {}]interface {}�&runtime.mapiterinit�@type.map[*net/http.Request]int64�"".datat�2runtime.mapaccess1_fast64�ptype.map[*net/http.Request]map[interface {}]interface {}�"".data�"runtime.mapdelete�@type.map[*net/http.Request]int64�"".datat�"runtime.mapdelete�&runtime.mapiternext �"".autotmp_0060type.int"".autotmp_0059,type.*net/http.Request"".autotmp_0058�,type.*net/http.Request"".autotmp_0057type.int64"".autotmp_0055�ztype.map.iter[*net/http.Request]map[interface {}]interface {}"".autotmp_0054ptype.map[*net/http.Request]map[interface {}]interface {}"".r�,type.*net/http.Requesttime.t·2�type.time.Time"".r�,type.*net/http.Request "".min�type.int64"".count�type.int "".~r1type.int"".maxAgetype.int"������H�" 33=J<e   -�>K0%Tgclocals·844ab127e744730859f0523d83a2c61dTgclocals·d7104e385b089a9888fc486f3f965182�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go�"".ClearHandler��eH� %H;aw���H��8H�H�$�H�|$H�|$0H�l$@H��H�H�H�D$PH�D$XH�H�$�H�D$H�-H�(H�D$ H�$H�<$toH�$H�\$0H�\$�H�\$ H�\$(H�1�H9�tH�\$(H�\$XH�D$PH��8�H�H�$H�H�\$H�H�\$�H�D$뽉%� + 0runtime.morestack_noctxt:*type.net/http.HandlerL"runtime.newobject�^type.struct { F uintptr; A0 *net/http.Handler }�"runtime.newobject�"".func·001�.runtime.writebarrierptr�Zgo.itab.net/http.HandlerFunc.net/http.Handler�2type.net/http.HandlerFunc�*type.net/http.Handler�Zgo.itab.net/http.HandlerFunc.net/http.Handler� runtime.typ2Itab@p"".autotmp_0063/`type.*struct { F uintptr; A0 *net/http.Handler }"".autotmp_00622type.net/http.HandlerFunc +"".&h,type.*net/http.Handler "".~r1 *type.net/http.Handlerp�op?��R�%83TTgclocals·2532067b4c791f17d2eaa4ddadeea3c0Tgclocals·f71d58207dae87d05175ac11727cdd3b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go�"".func·001��eH� %H;aw���H��8H�ZH�\$ H�\$PH�$H� Qj�YYH��uIH�\$ H�H�kH�\$@H�\$H�\$HH�\$H�\$PH�\$H�l$0H�,$H�T$(H�Z �Ӑ�H��8Ð�H��8� + "runtime.morestack^"".Clear·fn"runtime.deferproc� +�&runtime.deferreturn�&runtime.deferreturn0p +"".&h/,type.*net/http.Handler"".r ,type.*net/http.Request"".w8type.net/http.ResponseWriter&pMop +o ��#> 6H"Tgclocals·268041cca0e36eeedf29dd117f06a485Tgclocals·61e2515c69061b8fed0e66ece719f936�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go�"".init��eH� %H;aw���H�����t���uH���� ����H�H�$H�D$�H�D$H�H�$H�D$�H�H�$H�D$�H�D$H�H�$H�D$��H���$ + 0runtime.morestack_noctxt:"".initdone·R"".initdone·p"runtime.throwinit�"".initdone·�time.init�sync.init�net/http.init�ptype.map[*net/http.Request]map[interface {}]interface {}�runtime.makemap�"".data�.runtime.writebarrierptr�@type.map[*net/http.Request]int64�runtime.makemap�"".datat�.runtime.writebarrierptr�"".initdone·00/0�/ Tf�T�33� 7�Tgclocals·3280bececceccd33cb74587feedb1f9fTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/doc.go�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go�4type..hash.[8]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_0071type.int"".autotmp_0070type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[8]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go�0type..eq.[8]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_0075?"type.interface {}"".autotmp_0074"type.interface {}"".autotmp_0073_type.int"".autotmp_0072Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[8]interface {}"".p*type.*[8]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go�Tgclocals·341ab1f3c66e663a93f22f10d9eab46d00 +��Tgclocals·1245bf52b89d91ace0efa42912a5da6200 +�����Tgclocals·4ac5edf299a8cadc46c808258e6fc6d0((��Tgclocals·473d4314ba155bc5d9af9ad66f1c242a(( +>>>�Tgclocals·4ac5edf299a8cadc46c808258e6fc6d0((��Tgclocals·a5bd3918675a24472245dfa9083d6dee(( >>>�Tgclocals·af2d3162727927f9557359011162d41c``,�Z��Z�Tgclocals·7ba969af8c72fca351526f5bd553df3688�Tgclocals·af2d3162727927f9557359011162d41c``,�Z��Z�Tgclocals·b46c7a32cd3cbdb99d262657bbb5cb4688�Tgclocals·d9578cf05e73f94c5bc1acfa30cff71f �Tgclocals·e6f65f4d24a282ff6c0e03aa9f867bfe >>�Tgclocals·ab0f5354c6e12d990f77504eee3efe59(( +�Tgclocals·15395a9df917b4c9aa74d5c6c7e1ebf4((�Tgclocals·e1ae6533a9e39048ba0735a2264ce16a �Tgclocals·ac5bea9c8a91f5fb1d31bdacc5067b57 �Tgclocals·d7104e385b089a9888fc486f3f96518288 ��U��U(��U ��U�Tgclocals·844ab127e744730859f0523d83a2c61d88�,Zgo.itab.net/http.HandlerFunc.net/http.Handler�Tgclocals·f71d58207dae87d05175ac11727cdd3b00 �Tgclocals·2532067b4c791f17d2eaa4ddadeea3c000 �Tgclocals·61e2515c69061b8fed0e66ece719f936 �Tgclocals·268041cca0e36eeedf29dd117f06a485 ++�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·3280bececceccd33cb74587feedb1f9f�,"".mutex0"type.sync.RWMutex�*"".dataptype.map[*net/http.Request]map[interface {}]interface {}�*"".datat@type.map[*net/http.Request]int64�,"".initdone·type.uint8�"".Set·f "".Set�.sync.(*RWMutex).Lock·f(sync.(*RWMutex).Lock�8runtime.mapaccess1_fast64·f2runtime.mapaccess1_fast64�$runtime.makemap·fruntime.makemap�*runtime.mapassign1·f$runtime.mapassign1�time.Now·ftime.Now�2sync.(*RWMutex).Unlock·f,sync.(*RWMutex).Unlock�"".Get·f "".Get�0sync.(*RWMutex).RLock·f*sync.(*RWMutex).RLock�*runtime.mapaccess1·f$runtime.mapaccess1�4sync.(*RWMutex).RUnlock·f.sync.(*RWMutex).RUnlock�,runtime.throwreturn·f&runtime.throwreturn�"".GetOk·f"".GetOk�8runtime.mapaccess2_fast64·f2runtime.mapaccess2_fast64�*runtime.mapaccess2·f$runtime.mapaccess2�"".GetAll·f"".GetAll�,runtime.mapiterinit·f&runtime.mapiterinit�,runtime.mapiternext·f&runtime.mapiternext�"".GetAllOk·f"".GetAllOk�"".Delete·f"".Delete�(runtime.mapdelete·f"runtime.mapdelete�"".Clear·f"".Clear�"".clear·f"".clear�"".Purge·f"".Purge�4runtime.writebarrierptr·f.runtime.writebarrierptr�$"".ClearHandler·f"".ClearHandler�(runtime.newobject·f"runtime.newobject�"".func·001·f"".func·001�&runtime.typ2Itab·f runtime.typ2Itab�(runtime.deferproc·f"runtime.deferproc�,runtime.deferreturn·f&runtime.deferreturn�"".init·f"".init�(runtime.throwinit·f"runtime.throwinit�time.init·ftime.init�sync.init·fsync.init� net/http.init·fnet/http.init�bruntime.gcbits.0xcc000000000000000000000000000000 ��0go.string."interface {}"@: interface {} 0go.string."interface {}"�"type.interface {}���W� � runtime.algarray0bruntime.gcbits.0xcc000000000000000000000000000000P0go.string."interface {}"p4go.weak.type.*interface {}�"runtime.zerovalue��"type.interface {}�bruntime.gcbits.0x48844400000000000000000000000000 H�D�4go.string."[]interface {}"@>[]interface {} 4go.string."[]interface {}"�&type.[]interface {}��p��/ � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P4go.string."[]interface {}"p8go.weak.type.*[]interface {}�"runtime.zerovalue�"type.interface {}�Rgo.typelink.[]interface {}/[]interface {}&type.[]interface {}�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�go.typelink.[]uintptr/[]uintptrtype.[]uintptr�,go.string."[4]uintptr"@6 +[4]uintptr ,go.string."[4]uintptr"�type.[4]uintptr�� l<�  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P,go.string."[4]uintptr"p0go.weak.type.*[4]uintptr�"runtime.zerovalue�type.uintptr�type.[]uintptr�Bgo.typelink.[4]uintptr/[4]uintptrtype.[4]uintptr�bruntime.gcbits.0x88888844440000000000000000000000 ���DD�\go.string."map.iter[interface {}]interface {}"pf"map.iter[interface {}]interface {} \go.string."map.iter[interface {}]interface {}"�go.string."key"0(key go.string."key"�go.string."val"0(val go.string."val"�go.string."t"0$t go.string."t"�go.string."h"0$h go.string."h"� go.string."bptr"0*bptr go.string."bptr"�"go.string."other"0,other "go.string."other"�Ntype.map.iter[interface {}]interface {}��Pe@� (0( � runtime.algarray0bruntime.gcbits.0x88888844440000000000000000000000P\go.string."map.iter[interface {}]interface {}"p`go.weak.type.*map.iter[interface {}]interface {}�"runtime.zerovalue��Ntype.map.iter[interface {}]interface {}�go.string."key"�$type.*interface {}�go.string."val"�$type.*interface {}�go.string."t"�type.*uint8�go.string."h"�Ntype.*map.hdr[interface {}]interface {}�&go.string."buckets"�Ttype.*map.bucket[interface {}]interface {}� go.string."bptr"�Ttype.*map.bucket[interface {}]interface {}�"go.string."other"�type.[4]uintptr�4go.string."**http.Request"@>**http.Request 4go.string."**http.Request"�.type.**net/http.Request��"g�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."**http.Request"p@go.weak.type.***net/http.Request�"runtime.zerovalue�,type.*net/http.Request��go.string."*map.hdr[*http.Request]map[interface {}]interface {}"��4*map.hdr[*http.Request]map[interface {}]interface {} �go.string."*map.hdr[*http.Request]map[interface {}]interface {}"�ztype.*map.hdr[*net/http.Request]map[interface {}]interface {}���G��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."*map.hdr[*http.Request]map[interface {}]interface {}"p�go.weak.type.**map.hdr[*net/http.Request]map[interface {}]interface {}�"runtime.zerovalue�xtype.map.hdr[*net/http.Request]map[interface {}]interface {}��go.string."map.iter[*http.Request]map[interface {}]interface {}"��4map.iter[*http.Request]map[interface {}]interface {} �go.string."map.iter[*http.Request]map[interface {}]interface {}"�ztype.map.iter[*net/http.Request]map[interface {}]interface {}��P��t (0( � runtime.algarray0bruntime.gcbits.0x88888844440000000000000000000000P�go.string."map.iter[*http.Request]map[interface {}]interface {}"p�go.weak.type.*map.iter[*net/http.Request]map[interface {}]interface {}�"runtime.zerovalue��ztype.map.iter[*net/http.Request]map[interface {}]interface {}�go.string."key"�.type.**net/http.Request�go.string."val"�Ftype.*map[interface {}]interface {}�go.string."t"�type.*uint8�go.string."h"�ztype.*map.hdr[*net/http.Request]map[interface {}]interface {}�&go.string."buckets"��type.*map.bucket[*net/http.Request]map[interface {}]interface {}� go.string."bptr"��type.*map.bucket[*net/http.Request]map[interface {}]interface {}�"go.string."other"�type.[4]uintptr�bruntime.gcbits.0x84000000000000000000000000000000 ��dgo.string."struct { F uintptr; A0 *http.Handler }"pn&struct { F uintptr; A0 *http.Handler } dgo.string."struct { F uintptr; A0 *http.Handler }"�go.string."F"0$F go.string."F"�go.string."A0"0&A0 go.string."A0"�^type.struct { F uintptr; A0 *net/http.Handler }����K � runtime.algarray0bruntime.gcbits.0x84000000000000000000000000000000Pdgo.string."struct { F uintptr; A0 *http.Handler }"ppgo.weak.type.*struct { F uintptr; A0 *net/http.Handler }�"runtime.zerovalue��^type.struct { F uintptr; A0 *net/http.Handler }�go.string."F"�type.uintptr�go.string."A0"�,type.*net/http.Handler�fgo.string."*struct { F uintptr; A0 *http.Handler }"pp'*struct { F uintptr; A0 *http.Handler } fgo.string."*struct { F uintptr; A0 *http.Handler }"�`type.*struct { F uintptr; A0 *net/http.Handler }���8��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pfgo.string."*struct { F uintptr; A0 *http.Handler }"prgo.weak.type.**struct { F uintptr; A0 *net/http.Handler }�"runtime.zerovalue�^type.struct { F uintptr; A0 *net/http.Handler }�8go.string."*[8]interface {}"PB*[8]interface {} 8go.string."*[8]interface {}"�*type.*[8]interface {}���aK6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P8go.string."*[8]interface {}"p +__.PKGDEF 0 0 0 644 69882 ` +go object darwin amd64 go1.4.2 X:precisestack + +$$ +package mux + import runtime "runtime" + import url "net/url" + import errors "errors" + import http "net/http" + import strconv "strconv" + import strings "strings" + import fmt "fmt" + import context "github.com/fsouza/go-dockerclient/external/github.com/gorilla/context" + import regexp "regexp" + import bytes "bytes" + import path "path" + import io "io" // indirect + type @"io".Writer interface { Write(@"io".p []byte) (@"io".n int, @"io".err error) } + type @"net/http".keyValues struct { @"net/http".key string; @"net/http".values []string } + type @"net/http".headerSorter struct { @"net/http".kvs []@"net/http".keyValues } + func (@"net/http".s·2 *@"net/http".headerSorter "esc:0x0") Len () (? int) { return len(@"net/http".s·2.@"net/http".kvs) } + func (@"net/http".s·2 *@"net/http".headerSorter "esc:0x0") Less (@"net/http".i·3 int, @"net/http".j·4 int) (? bool) { return @"net/http".s·2.@"net/http".kvs[@"net/http".i·3].@"net/http".key < @"net/http".s·2.@"net/http".kvs[@"net/http".j·4].@"net/http".key } + func (@"net/http".s·1 *@"net/http".headerSorter "esc:0x0") Swap (@"net/http".i·2 int, @"net/http".j·3 int) { @"net/http".s·1.@"net/http".kvs[@"net/http".i·2], @"net/http".s·1.@"net/http".kvs[@"net/http".j·3] = @"net/http".s·1.@"net/http".kvs[@"net/http".j·3], @"net/http".s·1.@"net/http".kvs[@"net/http".i·2] } + type @"net/http".Header map[string][]string + func (@"net/http".h·1 @"net/http".Header "esc:0x0") Add (@"net/http".key·2 string, @"net/http".value·3 string) + func (@"net/http".h·1 @"net/http".Header "esc:0x0") Del (@"net/http".key·2 string "esc:0x0") + func (@"net/http".h·2 @"net/http".Header "esc:0x0") Get (@"net/http".key·3 string "esc:0x0") (? string) + func (@"net/http".h·1 @"net/http".Header "esc:0x0") Set (@"net/http".key·2 string, @"net/http".value·3 string) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") Write (@"net/http".w·3 @"io".Writer) (? error) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") WriteSubset (@"net/http".w·3 @"io".Writer, @"net/http".exclude·4 map[string]bool "esc:0x0") (? error) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") @"net/http".clone () (? @"net/http".Header) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") @"net/http".get (@"net/http".key·3 string "esc:0x0") (? string) { var @"net/http".v·4 []string; ; @"net/http".v·4 = @"net/http".h·2[@"net/http".key·3]; if len(@"net/http".v·4) > 0x0 { return @"net/http".v·4[0x0] }; return "" } + func (@"net/http".h·3 @"net/http".Header "esc:0x0") @"net/http".sortedKeyValues (@"net/http".exclude·4 map[string]bool "esc:0x0") (@"net/http".kvs·1 []@"net/http".keyValues, @"net/http".hs·2 *@"net/http".headerSorter) + type @"net/http".ResponseWriter interface { Header() (? @"net/http".Header); Write(? []byte) (? int, ? error); WriteHeader(? int) } + type @"net/url".Userinfo struct { @"net/url".username string; @"net/url".password string; @"net/url".passwordSet bool } + func (@"net/url".u·3 *@"net/url".Userinfo "esc:0x1") Password () (? string, ? bool) { if @"net/url".u·3.@"net/url".passwordSet { return @"net/url".u·3.@"net/url".password, true }; return "", false } + func (@"net/url".u·2 *@"net/url".Userinfo "esc:0x1") String () (? string) + func (@"net/url".u·2 *@"net/url".Userinfo "esc:0x1") Username () (? string) { return @"net/url".u·2.@"net/url".username } + type @"net/url".Values map[string][]string + func (@"net/url".v·1 @"net/url".Values "esc:0x0") Add (@"net/url".key·2 string, @"net/url".value·3 string) { @"net/url".v·1[@"net/url".key·2] = append(@"net/url".v·1[@"net/url".key·2], @"net/url".value·3) } + func (@"net/url".v·1 @"net/url".Values "esc:0x0") Del (@"net/url".key·2 string "esc:0x0") { delete(@"net/url".v·1, @"net/url".key·2) } + func (@"net/url".v·2 @"net/url".Values "esc:0x0") Encode () (? string) + func (@"net/url".v·2 @"net/url".Values "esc:0x0") Get (@"net/url".key·3 string "esc:0x0") (? string) { if @"net/url".v·2 == nil { return "" }; var @"net/url".vs·4 []string; ; var @"net/url".ok·5 bool; ; @"net/url".vs·4, @"net/url".ok·5 = @"net/url".v·2[@"net/url".key·3]; if !@"net/url".ok·5 || len(@"net/url".vs·4) == 0x0 { return "" }; return @"net/url".vs·4[0x0] } + func (@"net/url".v·1 @"net/url".Values "esc:0x0") Set (@"net/url".key·2 string, @"net/url".value·3 string) { @"net/url".v·1[@"net/url".key·2] = ([]string{ 0x0:@"net/url".value·3 }) } + type @"net/url".URL struct { Scheme string; Opaque string; User *@"net/url".Userinfo; Host string; Path string; RawQuery string; Fragment string } + func (@"net/url".u·2 *@"net/url".URL "esc:0x0") IsAbs () (? bool) { return @"net/url".u·2.Scheme != "" } + func (@"net/url".u·3 *@"net/url".URL "esc:0x2") Parse (@"net/url".ref·4 string) (? *@"net/url".URL, ? error) + func (@"net/url".u·2 *@"net/url".URL) Query () (? @"net/url".Values) + func (@"net/url".u·2 *@"net/url".URL "esc:0x1") RequestURI () (? string) + func (@"net/url".u·2 *@"net/url".URL "esc:0x2") ResolveReference (@"net/url".ref·3 *@"net/url".URL "esc:0x2") (? *@"net/url".URL) + func (@"net/url".u·2 *@"net/url".URL "esc:0x0") String () (? string) + type @"io".ReadCloser interface { Close() (? error); Read(@"io".p []byte) (@"io".n int, @"io".err error) } + import multipart "mime/multipart" // indirect + import textproto "net/textproto" // indirect + type @"net/textproto".MIMEHeader map[string][]string + func (@"net/textproto".h·1 @"net/textproto".MIMEHeader "esc:0x0") Add (@"net/textproto".key·2 string, @"net/textproto".value·3 string) + func (@"net/textproto".h·1 @"net/textproto".MIMEHeader "esc:0x0") Del (@"net/textproto".key·2 string "esc:0x0") + func (@"net/textproto".h·2 @"net/textproto".MIMEHeader "esc:0x0") Get (@"net/textproto".key·3 string "esc:0x0") (? string) + func (@"net/textproto".h·1 @"net/textproto".MIMEHeader "esc:0x0") Set (@"net/textproto".key·2 string, @"net/textproto".value·3 string) + type @"mime/multipart".File interface { Close() (? error); Read(@"io".p []byte) (@"io".n int, @"io".err error); ReadAt(@"io".p []byte, @"io".off int64) (@"io".n int, @"io".err error); Seek(@"io".offset int64, @"io".whence int) (? int64, ? error) } + type @"mime/multipart".FileHeader struct { Filename string; Header @"net/textproto".MIMEHeader; @"mime/multipart".content []byte; @"mime/multipart".tmpfile string } + func (@"mime/multipart".fh·3 *@"mime/multipart".FileHeader) Open () (? @"mime/multipart".File, ? error) + type @"mime/multipart".Form struct { Value map[string][]string; File map[string][]*@"mime/multipart".FileHeader } + func (@"mime/multipart".f·2 *@"mime/multipart".Form "esc:0x0") RemoveAll () (? error) + import tls "crypto/tls" // indirect + import x509 "crypto/x509" // indirect + type @"crypto/x509".SignatureAlgorithm int + type @"crypto/x509".PublicKeyAlgorithm int + import big "math/big" // indirect + type @"math/big".Word uintptr + type @"math/big".divisor struct { @"math/big".bbb @"math/big".nat; @"math/big".nbits int; @"math/big".ndigits int } + import rand "math/rand" // indirect + type @"math/rand".Source interface { Int63() (? int64); Seed(@"math/rand".seed int64) } + type @"math/rand".Rand struct { @"math/rand".src @"math/rand".Source } + func (@"math/rand".r·2 *@"math/rand".Rand) ExpFloat64 () (? float64) + func (@"math/rand".r·2 *@"math/rand".Rand) Float32 () (? float32) + func (@"math/rand".r·2 *@"math/rand".Rand) Float64 () (? float64) + func (@"math/rand".r·2 *@"math/rand".Rand) Int () (? int) + func (@"math/rand".r·2 *@"math/rand".Rand) Int31 () (? int32) + func (@"math/rand".r·2 *@"math/rand".Rand) Int31n (@"math/rand".n·3 int32) (? int32) + func (@"math/rand".r·2 *@"math/rand".Rand) Int63 () (? int64) + func (@"math/rand".r·2 *@"math/rand".Rand) Int63n (@"math/rand".n·3 int64) (? int64) + func (@"math/rand".r·2 *@"math/rand".Rand) Intn (@"math/rand".n·3 int) (? int) + func (@"math/rand".r·2 *@"math/rand".Rand) NormFloat64 () (? float64) + func (@"math/rand".r·2 *@"math/rand".Rand) Perm (@"math/rand".n·3 int) (? []int) + func (@"math/rand".r·1 *@"math/rand".Rand) Seed (@"math/rand".seed·2 int64) + func (@"math/rand".r·2 *@"math/rand".Rand) Uint32 () (? uint32) + type @"io".RuneScanner interface { ReadRune() (@"io".r rune, @"io".size int, @"io".err error); UnreadRune() (? error) } + type @"math/big".nat []@"math/big".Word + func (@"math/big".z·2 @"math/big".nat) @"math/big".add (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".and (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".andNot (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x0") @"math/big".bit (@"math/big".i·3 uint) (? uint) { var @"math/big".j·4 int; ; @"math/big".j·4 = int(@"math/big".i·3 / 0x40); if @"math/big".j·4 >= len(@"math/big".z·2) { return 0x0 }; return uint(@"math/big".z·2[@"math/big".j·4] >> (@"math/big".i·3 % 0x40) & @"math/big".Word(0x1)) } + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".bitLen () (? int) + func (@"math/big".z·2 @"math/big".nat "esc:0x0") @"math/big".bytes (@"math/big".buf·3 []byte "esc:0x0") (@"math/big".i·1 int) + func (@"math/big".z·1 @"math/big".nat "esc:0x0") @"math/big".clear () + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".cmp (@"math/big".y·3 @"math/big".nat "esc:0x0") (@"math/big".r·1 int) + func (@"math/big".q·1 @"math/big".nat) @"math/big".convertWords (@"math/big".s·2 []byte "esc:0x0", @"math/big".charset·3 string "esc:0x0", @"math/big".b·4 @"math/big".Word, @"math/big".ndigits·5 int, @"math/big".bb·6 @"math/big".Word, @"math/big".table·7 []@"math/big".divisor "esc:0x0") + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".decimalString () (? string) + func (@"math/big".z·3 @"math/big".nat) @"math/big".div (@"math/big".z2·4 @"math/big".nat, @"math/big".u·5 @"math/big".nat, @"math/big".v·6 @"math/big".nat) (@"math/big".q·1 @"math/big".nat, @"math/big".r·2 @"math/big".nat) + func (@"math/big".z·3 @"math/big".nat "esc:0x2") @"math/big".divLarge (@"math/big".u·4 @"math/big".nat, @"math/big".uIn·5 @"math/big".nat, @"math/big".v·6 @"math/big".nat) (@"math/big".q·1 @"math/big".nat, @"math/big".r·2 @"math/big".nat) + func (@"math/big".z·3 @"math/big".nat) @"math/big".divW (@"math/big".x·4 @"math/big".nat, @"math/big".y·5 @"math/big".Word) (@"math/big".q·1 @"math/big".nat, @"math/big".r·2 @"math/big".Word) + func (@"math/big".z·2 @"math/big".nat) @"math/big".expNN (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat "esc:0x0", @"math/big".m·5 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".expNNWindowed (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat "esc:0x0", @"math/big".m·5 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".expWW (@"math/big".x·3 @"math/big".Word, @"math/big".y·4 @"math/big".Word) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".make (@"math/big".n·3 int) (? @"math/big".nat) + func (@"math/big".x·2 @"math/big".nat) @"math/big".modW (@"math/big".d·3 @"math/big".Word) (@"math/big".r·1 @"math/big".Word) + func (@"math/big".z·2 @"math/big".nat) @"math/big".mul (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".mulAddWW (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".Word, @"math/big".r·5 @"math/big".Word) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".mulRange (@"math/big".a·3 uint64, @"math/big".b·4 uint64) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".norm () (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".or (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".n·2 @"math/big".nat) @"math/big".probablyPrime (@"math/big".reps·3 int) (? bool) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".random (@"math/big".rand·3 *@"math/rand".Rand, @"math/big".limit·4 @"math/big".nat "esc:0x0", @"math/big".n·5 int) (? @"math/big".nat) + func (@"math/big".z·4 @"math/big".nat) @"math/big".scan (@"math/big".r·5 @"io".RuneScanner, @"math/big".base·6 int) (? @"math/big".nat, ? int, ? error) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".set (@"math/big".x·3 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setBit (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".i·4 uint, @"math/big".b·5 uint) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setBytes (@"math/big".buf·3 []byte "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setUint64 (@"math/big".x·3 uint64) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setWord (@"math/big".x·3 @"math/big".Word) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".shl (@"math/big".x·3 @"math/big".nat, @"math/big".s·4 uint) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".shr (@"math/big".x·3 @"math/big".nat, @"math/big".s·4 uint) (? @"math/big".nat) + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".string (@"math/big".charset·3 string "esc:0x0") (? string) + func (@"math/big".z·2 @"math/big".nat) @"math/big".sub (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".trailingZeroBits () (? uint) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".xor (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + type @"fmt".State interface { Flag(@"fmt".c int) (? bool); Precision() (@"fmt".prec int, @"fmt".ok bool); Width() (@"fmt".wid int, @"fmt".ok bool); Write(@"fmt".b []byte) (@"fmt".ret int, @"fmt".err error) } + type @"fmt".ScanState interface { Read(@"fmt".buf []byte) (@"fmt".n int, @"fmt".err error); ReadRune() (@"fmt".r rune, @"fmt".size int, @"fmt".err error); SkipSpace(); Token(@"fmt".skipSpace bool, @"fmt".f func(? rune) (? bool)) (@"fmt".token []byte, @"fmt".err error); UnreadRune() (? error); Width() (@"fmt".wid int, @"fmt".ok bool) } + type @"math/big".Int struct { @"math/big".neg bool; @"math/big".abs @"math/big".nat } + func (@"math/big".z·2 *@"math/big".Int) Abs (@"math/big".x·3 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Add (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) And (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) AndNot (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Binomial (@"math/big".n·3 int64, @"math/big".k·4 int64) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int) Bit (@"math/big".i·3 int) (? uint) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") BitLen () (? int) + func (@"math/big".x·2 *@"math/big".Int "esc:0x1") Bits () (? []@"math/big".Word) { return @"math/big".x·2.@"math/big".abs } + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Bytes () (? []byte) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Cmp (@"math/big".y·3 *@"math/big".Int "esc:0x0") (@"math/big".r·1 int) + func (@"math/big".z·2 *@"math/big".Int) Div (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int) DivMod (@"math/big".x·4 *@"math/big".Int, @"math/big".y·5 *@"math/big".Int, @"math/big".m·6 *@"math/big".Int) (? *@"math/big".Int, ? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Exp (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int "esc:0x0", @"math/big".m·5 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".x·1 *@"math/big".Int "esc:0x0") Format (@"math/big".s·2 @"fmt".State, @"math/big".ch·3 rune) + func (@"math/big".z·2 *@"math/big".Int) GCD (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int, @"math/big".a·5 *@"math/big".Int, @"math/big".b·6 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) GobDecode (@"math/big".buf·3 []byte "esc:0x0") (? error) + func (@"math/big".x·3 *@"math/big".Int "esc:0x0") GobEncode () (? []byte, ? error) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Int64 () (? int64) + func (@"math/big".z·2 *@"math/big".Int) Lsh (@"math/big".x·3 *@"math/big".Int, @"math/big".n·4 uint) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int "esc:0x0") MarshalJSON () (? []byte, ? error) + func (@"math/big".z·3 *@"math/big".Int "esc:0x0") MarshalText () (@"math/big".text·1 []byte, @"math/big".err·2 error) + func (@"math/big".z·2 *@"math/big".Int) Mod (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) ModInverse (@"math/big".g·3 *@"math/big".Int, @"math/big".n·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Mul (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) MulRange (@"math/big".a·3 int64, @"math/big".b·4 int64) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Neg (@"math/big".x·3 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Not (@"math/big".x·3 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Or (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int) ProbablyPrime (@"math/big".n·3 int) (? bool) + func (@"math/big".z·2 *@"math/big".Int) Quo (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int) QuoRem (@"math/big".x·4 *@"math/big".Int, @"math/big".y·5 *@"math/big".Int, @"math/big".r·6 *@"math/big".Int) (? *@"math/big".Int, ? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Rand (@"math/big".rnd·3 *@"math/rand".Rand, @"math/big".n·4 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Rem (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Rsh (@"math/big".x·3 *@"math/big".Int, @"math/big".n·4 uint) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Scan (@"math/big".s·3 @"fmt".ScanState, @"math/big".ch·4 rune) (? error) + func (@"math/big".z·2 *@"math/big".Int) Set (@"math/big".x·3 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) SetBit (@"math/big".x·3 *@"math/big".Int, @"math/big".i·4 int, @"math/big".b·5 uint) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int "esc:0x2") SetBits (@"math/big".abs·3 []@"math/big".Word) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) SetBytes (@"math/big".buf·3 []byte "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) SetInt64 (@"math/big".x·3 int64) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int) SetString (@"math/big".s·4 string, @"math/big".base·5 int) (? *@"math/big".Int, ? bool) + func (@"math/big".z·2 *@"math/big".Int) SetUint64 (@"math/big".x·3 uint64) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Sign () (? int) { if len(@"math/big".x·2.@"math/big".abs) == 0x0 { return 0x0 }; if @"math/big".x·2.@"math/big".neg { return -0x1 }; return 0x1 } + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") String () (? string) + func (@"math/big".z·2 *@"math/big".Int) Sub (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Uint64 () (? uint64) + func (@"math/big".z·2 *@"math/big".Int) UnmarshalJSON (@"math/big".text·3 []byte) (? error) + func (@"math/big".z·2 *@"math/big".Int) UnmarshalText (@"math/big".text·3 []byte) (? error) + func (@"math/big".z·2 *@"math/big".Int) Xor (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) @"math/big".binaryGCD (@"math/big".a·3 *@"math/big".Int, @"math/big".b·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·4 *@"math/big".Int) @"math/big".scan (@"math/big".r·5 @"io".RuneScanner, @"math/big".base·6 int) (? *@"math/big".Int, ? int, ? error) + import pkix "crypto/x509/pkix" // indirect + import asn1 "encoding/asn1" // indirect + type @"encoding/asn1".ObjectIdentifier []int + func (@"encoding/asn1".oi·2 @"encoding/asn1".ObjectIdentifier "esc:0x0") Equal (@"encoding/asn1".other·3 @"encoding/asn1".ObjectIdentifier "esc:0x0") (? bool) + func (@"encoding/asn1".oi·2 @"encoding/asn1".ObjectIdentifier "esc:0x0") String () (? string) + type @"crypto/x509/pkix".AttributeTypeAndValue struct { Type @"encoding/asn1".ObjectIdentifier; Value interface {} } + type @"crypto/x509/pkix".RelativeDistinguishedNameSET []@"crypto/x509/pkix".AttributeTypeAndValue + type @"crypto/x509/pkix".RDNSequence []@"crypto/x509/pkix".RelativeDistinguishedNameSET + type @"crypto/x509/pkix".Name struct { Country []string; Organization []string; OrganizationalUnit []string; Locality []string; Province []string; StreetAddress []string; PostalCode []string; SerialNumber string; CommonName string; Names []@"crypto/x509/pkix".AttributeTypeAndValue } + func (@"crypto/x509/pkix".n·1 *@"crypto/x509/pkix".Name) FillFromRDNSequence (@"crypto/x509/pkix".rdns·2 *@"crypto/x509/pkix".RDNSequence "esc:0x0") + func (@"crypto/x509/pkix".n·2 @"crypto/x509/pkix".Name) ToRDNSequence () (@"crypto/x509/pkix".ret·1 @"crypto/x509/pkix".RDNSequence) + import time "time" // indirect + type @"time".zone struct { @"time".name string; @"time".offset int; @"time".isDST bool } + type @"time".zoneTrans struct { @"time".when int64; @"time".index uint8; @"time".isstd bool; @"time".isutc bool } + type @"time".Location struct { @"time".name string; @"time".zone []@"time".zone; @"time".tx []@"time".zoneTrans; @"time".cacheStart int64; @"time".cacheEnd int64; @"time".cacheZone *@"time".zone } + func (@"time".l·2 *@"time".Location "esc:0x0") String () (? string) + func (@"time".l·2 *@"time".Location "esc:0x0") @"time".firstZoneUsed () (? bool) + func (@"time".l·2 *@"time".Location "esc:0x2") @"time".get () (? *@"time".Location) + func (@"time".l·6 *@"time".Location "esc:0x1") @"time".lookup (@"time".sec·7 int64) (@"time".name·1 string, @"time".offset·2 int, @"time".isDST·3 bool, @"time".start·4 int64, @"time".end·5 int64) + func (@"time".l·2 *@"time".Location "esc:0x0") @"time".lookupFirstZone () (? int) + func (@"time".l·4 *@"time".Location "esc:0x0") @"time".lookupName (@"time".name·5 string "esc:0x0", @"time".unix·6 int64) (@"time".offset·1 int, @"time".isDST·2 bool, @"time".ok·3 bool) + type @"time".Duration int64 + func (@"time".d·2 @"time".Duration) Hours () (? float64) { var @"time".hour·3 @"time".Duration; ; @"time".hour·3 = @"time".d·2 / @"time".Duration(0x34630B8A000); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0x34630B8A000); return float64(@"time".hour·3) + float64(@"time".nsec·4) * 0x9C5FFF26ED75Fp-93 } + func (@"time".d·2 @"time".Duration) Minutes () (? float64) { var @"time".min·3 @"time".Duration; ; @"time".min·3 = @"time".d·2 / @"time".Duration(0xDF8475800); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0xDF8475800); return float64(@"time".min·3) + float64(@"time".nsec·4) * 0x9299FF347E9E9p-87 } + func (@"time".d·2 @"time".Duration) Nanoseconds () (? int64) { return int64(@"time".d·2) } + func (@"time".d·2 @"time".Duration) Seconds () (? float64) { var @"time".sec·3 @"time".Duration; ; @"time".sec·3 = @"time".d·2 / @"time".Duration(0x3B9ACA00); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0x3B9ACA00); return float64(@"time".sec·3) + float64(@"time".nsec·4) * 0x112E0BE826D695p-82 } + func (@"time".d·2 @"time".Duration) String () (? string) + type @"time".Month int + func (@"time".m·2 @"time".Month) String () (? string) { return @"time".months[@"time".m·2 - @"time".Month(0x1)] } + type @"time".Weekday int + func (@"time".d·2 @"time".Weekday) String () (? string) { return @"time".days[@"time".d·2] } + type @"time".Time struct { @"time".sec int64; @"time".nsec int32; @"time".loc *@"time".Location } + func (@"time".t·2 @"time".Time "esc:0x2") Add (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x2") AddDate (@"time".years·3 int, @"time".months·4 int, @"time".days·5 int) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") After (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec > @"time".u·3.@"time".sec || @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec > @"time".u·3.@"time".nsec } + func (@"time".t·2 @"time".Time "esc:0x0") Before (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec < @"time".u·3.@"time".sec || @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec < @"time".u·3.@"time".nsec } + func (@"time".t·4 @"time".Time "esc:0x0") Clock () (@"time".hour·1 int, @"time".min·2 int, @"time".sec·3 int) + func (@"time".t·4 @"time".Time "esc:0x0") Date () (@"time".year·1 int, @"time".month·2 @"time".Month, @"time".day·3 int) + func (@"time".t·2 @"time".Time "esc:0x0") Day () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") Equal (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec == @"time".u·3.@"time".nsec } + func (@"time".t·2 @"time".Time "esc:0x0") Format (@"time".layout·3 string "esc:0x0") (? string) + func (@"time".t·2 *@"time".Time "esc:0x0") GobDecode (@"time".data·3 []byte "esc:0x0") (? error) + func (@"time".t·3 @"time".Time "esc:0x0") GobEncode () (? []byte, ? error) + func (@"time".t·2 @"time".Time "esc:0x0") Hour () (? int) + func (@"time".t·3 @"time".Time "esc:0x0") ISOWeek () (@"time".year·1 int, @"time".week·2 int) + func (@"time".t·2 @"time".Time "esc:0x2") In (@"time".loc·3 *@"time".Location "esc:0x2") (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") IsZero () (? bool) { return @"time".t·2.@"time".sec == 0x0 && @"time".t·2.@"time".nsec == 0x0 } + func (@"time".t·2 @"time".Time "esc:0x2") Local () (? @"time".Time) { @"time".t·2.@"time".loc = @"time".Local; return @"time".t·2 } + func (@"time".t·2 @"time".Time "esc:0x2") Location () (? *@"time".Location) { var @"time".l·3 *@"time".Location; ; @"time".l·3 = @"time".t·2.@"time".loc; if @"time".l·3 == nil { @"time".l·3 = @"time".UTC }; return @"time".l·3 } + func (@"time".t·3 @"time".Time "esc:0x0") MarshalBinary () (? []byte, ? error) + func (@"time".t·3 @"time".Time "esc:0x0") MarshalJSON () (? []byte, ? error) + func (@"time".t·3 @"time".Time "esc:0x0") MarshalText () (? []byte, ? error) + func (@"time".t·2 @"time".Time "esc:0x0") Minute () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") Month () (? @"time".Month) + func (@"time".t·2 @"time".Time "esc:0x0") Nanosecond () (? int) { return int(@"time".t·2.@"time".nsec) } + func (@"time".t·2 @"time".Time "esc:0x2") Round (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") Second () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") String () (? string) + func (@"time".t·2 @"time".Time "esc:0x0") Sub (@"time".u·3 @"time".Time "esc:0x0") (? @"time".Duration) + func (@"time".t·2 @"time".Time "esc:0x2") Truncate (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x2") UTC () (? @"time".Time) { @"time".t·2.@"time".loc = @"time".UTC; return @"time".t·2 } + func (@"time".t·2 @"time".Time "esc:0x0") Unix () (? int64) { return @"time".t·2.@"time".sec + -0xE7791F700 } + func (@"time".t·2 @"time".Time "esc:0x0") UnixNano () (? int64) { return (@"time".t·2.@"time".sec + -0xE7791F700) * 0x3B9ACA00 + int64(@"time".t·2.@"time".nsec) } + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalBinary (@"time".data·3 []byte "esc:0x0") (? error) + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalJSON (@"time".data·3 []byte "esc:0x0") (@"time".err·1 error) + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalText (@"time".data·3 []byte "esc:0x0") (@"time".err·1 error) + func (@"time".t·2 @"time".Time "esc:0x0") Weekday () (? @"time".Weekday) + func (@"time".t·2 @"time".Time "esc:0x0") Year () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") YearDay () (? int) + func (@"time".t·3 @"time".Time "esc:0x0") Zone () (@"time".name·1 string, @"time".offset·2 int) + func (@"time".t·2 @"time".Time "esc:0x0") @"time".abs () (? uint64) + func (@"time".t·5 @"time".Time "esc:0x0") @"time".date (@"time".full·6 bool) (@"time".year·1 int, @"time".month·2 @"time".Month, @"time".day·3 int, @"time".yday·4 int) + func (@"time".t·4 @"time".Time "esc:0x1") @"time".locabs () (@"time".name·1 string, @"time".offset·2 int, @"time".abs·3 uint64) + type @"crypto/x509".KeyUsage int + type @"crypto/x509/pkix".Extension struct { Id @"encoding/asn1".ObjectIdentifier; Critical bool "asn1:\"optional\""; Value []byte } + type @"crypto/x509".ExtKeyUsage int + import net "net" // indirect + type @"net".IPMask []byte + func (@"net".m·3 @"net".IPMask "esc:0x0") Size () (@"net".ones·1 int, @"net".bits·2 int) + func (@"net".m·2 @"net".IPMask "esc:0x0") String () (? string) + type @"net".IP []byte + func (@"net".ip·2 @"net".IP "esc:0x0") DefaultMask () (? @"net".IPMask) + func (@"net".ip·2 @"net".IP "esc:0x0") Equal (@"net".x·3 @"net".IP "esc:0x0") (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsGlobalUnicast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsInterfaceLocalMulticast () (? bool) { return len(@"net".ip·2) == 0x10 && @"net".ip·2[0x0] == byte(0xFF) && @"net".ip·2[0x1] & byte(0xF) == byte(0x1) } + func (@"net".ip·2 @"net".IP "esc:0x0") IsLinkLocalMulticast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsLinkLocalUnicast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsLoopback () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsMulticast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsUnspecified () (? bool) + func (@"net".ip·3 @"net".IP "esc:0x0") MarshalText () (? []byte, ? error) + func (@"net".ip·2 @"net".IP "esc:0x0") Mask (@"net".mask·3 @"net".IPMask "esc:0x0") (? @"net".IP) + func (@"net".ip·2 @"net".IP "esc:0x0") String () (? string) + func (@"net".ip·2 @"net".IP "esc:0x2") To16 () (? @"net".IP) + func (@"net".ip·2 @"net".IP "esc:0x2") To4 () (? @"net".IP) + func (@"net".ip·2 *@"net".IP "esc:0x0") UnmarshalText (@"net".text·3 []byte "esc:0x0") (? error) + type @"encoding/asn1".RawContent []byte + type @"encoding/asn1".RawValue struct { Class int; Tag int; IsCompound bool; Bytes []byte; FullBytes []byte } + type @"crypto/x509/pkix".AlgorithmIdentifier struct { Algorithm @"encoding/asn1".ObjectIdentifier; Parameters @"encoding/asn1".RawValue "asn1:\"optional\"" } + type @"crypto/x509/pkix".RevokedCertificate struct { SerialNumber *@"math/big".Int; RevocationTime @"time".Time; Extensions []@"crypto/x509/pkix".Extension "asn1:\"optional\"" } + type @"crypto/x509/pkix".TBSCertificateList struct { Raw @"encoding/asn1".RawContent; Version int "asn1:\"optional,default:2\""; Signature @"crypto/x509/pkix".AlgorithmIdentifier; Issuer @"crypto/x509/pkix".RDNSequence; ThisUpdate @"time".Time; NextUpdate @"time".Time "asn1:\"optional\""; RevokedCertificates []@"crypto/x509/pkix".RevokedCertificate "asn1:\"optional\""; Extensions []@"crypto/x509/pkix".Extension "asn1:\"tag:0,optional,explicit\"" } + type @"encoding/asn1".BitString struct { Bytes []byte; BitLength int } + func (@"encoding/asn1".b·2 @"encoding/asn1".BitString "esc:0x0") At (@"encoding/asn1".i·3 int) (? int) { if @"encoding/asn1".i·3 < 0x0 || @"encoding/asn1".i·3 >= @"encoding/asn1".b·2.BitLength { return 0x0 }; var @"encoding/asn1".x·4 int; ; @"encoding/asn1".x·4 = @"encoding/asn1".i·3 / 0x8; var @"encoding/asn1".y·5 uint; ; @"encoding/asn1".y·5 = 0x7 - uint(@"encoding/asn1".i·3 % 0x8); return int(@"encoding/asn1".b·2.Bytes[@"encoding/asn1".x·4] >> @"encoding/asn1".y·5) & 0x1 } + func (@"encoding/asn1".b·2 @"encoding/asn1".BitString "esc:0x2") RightAlign () (? []byte) + type @"crypto/x509/pkix".CertificateList struct { TBSCertList @"crypto/x509/pkix".TBSCertificateList; SignatureAlgorithm @"crypto/x509/pkix".AlgorithmIdentifier; SignatureValue @"encoding/asn1".BitString } + func (@"crypto/x509/pkix".certList·2 *@"crypto/x509/pkix".CertificateList "esc:0x0") HasExpired (@"crypto/x509/pkix".now·3 @"time".Time "esc:0x0") (? bool) + type @"io".Reader interface { Read(@"io".p []byte) (@"io".n int, @"io".err error) } + type @"crypto/x509".CertPool struct { @"crypto/x509".bySubjectKeyId map[string][]int; @"crypto/x509".byName map[string][]int; @"crypto/x509".certs []*@"crypto/x509".Certificate } + func (@"crypto/x509".s·1 *@"crypto/x509".CertPool) AddCert (@"crypto/x509".cert·2 *@"crypto/x509".Certificate) + func (@"crypto/x509".s·2 *@"crypto/x509".CertPool) AppendCertsFromPEM (@"crypto/x509".pemCerts·3 []byte) (@"crypto/x509".ok·1 bool) + func (@"crypto/x509".s·2 *@"crypto/x509".CertPool "esc:0x0") Subjects () (@"crypto/x509".res·1 [][]byte) + func (@"crypto/x509".s·4 *@"crypto/x509".CertPool "esc:0x0") @"crypto/x509".findVerifiedParents (@"crypto/x509".cert·5 *@"crypto/x509".Certificate) (@"crypto/x509".parents·1 []int, @"crypto/x509".errCert·2 *@"crypto/x509".Certificate, @"crypto/x509".err·3 error) + type @"crypto/x509".VerifyOptions struct { DNSName string; Intermediates *@"crypto/x509".CertPool; Roots *@"crypto/x509".CertPool; CurrentTime @"time".Time; KeyUsages []@"crypto/x509".ExtKeyUsage } + type @"crypto/x509".Certificate struct { Raw []byte; RawTBSCertificate []byte; RawSubjectPublicKeyInfo []byte; RawSubject []byte; RawIssuer []byte; Signature []byte; SignatureAlgorithm @"crypto/x509".SignatureAlgorithm; PublicKeyAlgorithm @"crypto/x509".PublicKeyAlgorithm; PublicKey interface {}; Version int; SerialNumber *@"math/big".Int; Issuer @"crypto/x509/pkix".Name; Subject @"crypto/x509/pkix".Name; NotBefore @"time".Time; NotAfter @"time".Time; KeyUsage @"crypto/x509".KeyUsage; Extensions []@"crypto/x509/pkix".Extension; ExtraExtensions []@"crypto/x509/pkix".Extension; ExtKeyUsage []@"crypto/x509".ExtKeyUsage; UnknownExtKeyUsage []@"encoding/asn1".ObjectIdentifier; BasicConstraintsValid bool; IsCA bool; MaxPathLen int; MaxPathLenZero bool; SubjectKeyId []byte; AuthorityKeyId []byte; OCSPServer []string; IssuingCertificateURL []string; DNSNames []string; EmailAddresses []string; IPAddresses []@"net".IP; PermittedDNSDomainsCritical bool; PermittedDNSDomains []string; CRLDistributionPoints []string; PolicyIdentifiers []@"encoding/asn1".ObjectIdentifier } + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate) CheckCRLSignature (@"crypto/x509".crl·3 *@"crypto/x509/pkix".CertificateList) (@"crypto/x509".err·1 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate) CheckSignature (@"crypto/x509".algo·3 @"crypto/x509".SignatureAlgorithm, @"crypto/x509".signed·4 []byte, @"crypto/x509".signature·5 []byte) (@"crypto/x509".err·1 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate) CheckSignatureFrom (@"crypto/x509".parent·3 *@"crypto/x509".Certificate) (@"crypto/x509".err·1 error) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate) CreateCRL (@"crypto/x509".rand·4 @"io".Reader, @"crypto/x509".priv·5 interface {}, @"crypto/x509".revokedCerts·6 []@"crypto/x509/pkix".RevokedCertificate, @"crypto/x509".now·7 @"time".Time, @"crypto/x509".expiry·8 @"time".Time) (@"crypto/x509".crlBytes·1 []byte, @"crypto/x509".err·2 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate "esc:0x0") Equal (@"crypto/x509".other·3 *@"crypto/x509".Certificate "esc:0x0") (? bool) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate) Verify (@"crypto/x509".opts·4 @"crypto/x509".VerifyOptions "esc:0x4") (@"crypto/x509".chains·1 [][]*@"crypto/x509".Certificate, @"crypto/x509".err·2 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate "esc:0x2") VerifyHostname (@"crypto/x509".h·3 string "esc:0x2") (? error) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate) @"crypto/x509".buildChains (@"crypto/x509".cache·4 map[int][][]*@"crypto/x509".Certificate "esc:0x0", @"crypto/x509".currentChain·5 []*@"crypto/x509".Certificate "esc:0x0", @"crypto/x509".opts·6 *@"crypto/x509".VerifyOptions "esc:0x0") (@"crypto/x509".chains·1 [][]*@"crypto/x509".Certificate, @"crypto/x509".err·2 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate "esc:0x2") @"crypto/x509".isValid (@"crypto/x509".certType·3 int, @"crypto/x509".currentChain·4 []*@"crypto/x509".Certificate "esc:0x0", @"crypto/x509".opts·5 *@"crypto/x509".VerifyOptions "esc:0x0") (? error) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate "esc:0x0") @"crypto/x509".systemVerify (@"crypto/x509".opts·4 *@"crypto/x509".VerifyOptions "esc:0x0") (@"crypto/x509".chains·1 [][]*@"crypto/x509".Certificate, @"crypto/x509".err·2 error) { return nil, nil } + type @"crypto/tls".ConnectionState struct { Version uint16; HandshakeComplete bool; DidResume bool; CipherSuite uint16; NegotiatedProtocol string; NegotiatedProtocolIsMutual bool; ServerName string; PeerCertificates []*@"crypto/x509".Certificate; VerifiedChains [][]*@"crypto/x509".Certificate; TLSUnique []byte } + type @"net/http".Cookie struct { Name string; Value string; Path string; Domain string; Expires @"time".Time; RawExpires string; MaxAge int; Secure bool; HttpOnly bool; Raw string; Unparsed []string } + func (@"net/http".c·2 *@"net/http".Cookie) String () (? string) + import bufio "bufio" // indirect + type @"bufio".Reader struct { @"bufio".buf []byte; @"bufio".rd @"io".Reader; @"bufio".r int; @"bufio".w int; @"bufio".err error; @"bufio".lastByte int; @"bufio".lastRuneSize int } + func (@"bufio".b·2 *@"bufio".Reader "esc:0x0") Buffered () (? int) { return @"bufio".b·2.@"bufio".w - @"bufio".b·2.@"bufio".r } + func (@"bufio".b·3 *@"bufio".Reader) Peek (@"bufio".n·4 int) (? []byte, ? error) + func (@"bufio".b·3 *@"bufio".Reader) Read (@"bufio".p·4 []byte) (@"bufio".n·1 int, @"bufio".err·2 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadByte () (@"bufio".c·1 byte, @"bufio".err·2 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadBytes (@"bufio".delim·4 byte) (@"bufio".line·1 []byte, @"bufio".err·2 error) + func (@"bufio".b·4 *@"bufio".Reader) ReadLine () (@"bufio".line·1 []byte, @"bufio".isPrefix·2 bool, @"bufio".err·3 error) + func (@"bufio".b·4 *@"bufio".Reader) ReadRune () (@"bufio".r·1 rune, @"bufio".size·2 int, @"bufio".err·3 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadSlice (@"bufio".delim·4 byte) (@"bufio".line·1 []byte, @"bufio".err·2 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadString (@"bufio".delim·4 byte) (@"bufio".line·1 string, @"bufio".err·2 error) + func (@"bufio".b·1 *@"bufio".Reader) Reset (@"bufio".r·2 @"io".Reader) + func (@"bufio".b·2 *@"bufio".Reader "esc:0x0") UnreadByte () (? error) + func (@"bufio".b·2 *@"bufio".Reader "esc:0x0") UnreadRune () (? error) { if @"bufio".b·2.@"bufio".lastRuneSize < 0x0 || @"bufio".b·2.@"bufio".r < @"bufio".b·2.@"bufio".lastRuneSize { return @"bufio".ErrInvalidUnreadRune }; @"bufio".b·2.@"bufio".r -= @"bufio".b·2.@"bufio".lastRuneSize; @"bufio".b·2.@"bufio".lastByte = -0x1; @"bufio".b·2.@"bufio".lastRuneSize = -0x1; return nil } + func (@"bufio".b·3 *@"bufio".Reader) WriteTo (@"bufio".w·4 @"io".Writer) (@"bufio".n·1 int64, @"bufio".err·2 error) + func (@"bufio".b·1 *@"bufio".Reader) @"bufio".fill () + func (@"bufio".b·2 *@"bufio".Reader "esc:0x1") @"bufio".readErr () (? error) { var @"bufio".err·3 error; ; @"bufio".err·3 = @"bufio".b·2.@"bufio".err; @"bufio".b·2.@"bufio".err = nil; return @"bufio".err·3 } + func (@"bufio".b·1 *@"bufio".Reader "esc:0x0") @"bufio".reset (@"bufio".buf·2 []byte, @"bufio".r·3 @"io".Reader) { *@"bufio".b·1 = (@"bufio".Reader{ @"bufio".buf:@"bufio".buf·2, @"bufio".rd:@"bufio".r·3, @"bufio".lastByte:-0x1, @"bufio".lastRuneSize:-0x1 }) } + func (@"bufio".b·3 *@"bufio".Reader) @"bufio".writeBuf (@"bufio".w·4 @"io".Writer) (? int64, ? error) + type @"bytes".readOp int + type @"bytes".Buffer struct { @"bytes".buf []byte; @"bytes".off int; @"bytes".runeBytes [4]byte; @"bytes".bootstrap [64]byte; @"bytes".lastRead @"bytes".readOp } + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x1") Bytes () (? []byte) { return @"bytes".b·2.@"bytes".buf[@"bytes".b·2.@"bytes".off:] } + func (@"bytes".b·1 *@"bytes".Buffer) Grow (@"bytes".n·2 int) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") Len () (? int) { return len(@"bytes".b·2.@"bytes".buf) - @"bytes".b·2.@"bytes".off } + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x1") Next (@"bytes".n·3 int) (? []byte) + func (@"bytes".b·3 *@"bytes".Buffer) Read (@"bytes".p·4 []byte "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) ReadByte () (@"bytes".c·1 byte, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x0") ReadBytes (@"bytes".delim·4 byte) (@"bytes".line·1 []byte, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) ReadFrom (@"bytes".r·4 @"io".Reader) (@"bytes".n·1 int64, @"bytes".err·2 error) + func (@"bytes".b·4 *@"bytes".Buffer) ReadRune () (@"bytes".r·1 rune, @"bytes".size·2 int, @"bytes".err·3 error) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x0") ReadString (@"bytes".delim·4 byte) (@"bytes".line·1 string, @"bytes".err·2 error) + func (@"bytes".b·1 *@"bytes".Buffer) Reset () + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") String () (? string) { if @"bytes".b·2 == nil { return "" }; return string(@"bytes".b·2.@"bytes".buf[@"bytes".b·2.@"bytes".off:]) } + func (@"bytes".b·1 *@"bytes".Buffer) Truncate (@"bytes".n·2 int) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") UnreadByte () (? error) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") UnreadRune () (? error) + func (@"bytes".b·3 *@"bytes".Buffer) Write (@"bytes".p·4 []byte "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·2 *@"bytes".Buffer) WriteByte (@"bytes".c·3 byte) (? error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteRune (@"bytes".r·4 rune) (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteString (@"bytes".s·4 string "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteTo (@"bytes".w·4 @"io".Writer) (@"bytes".n·1 int64, @"bytes".err·2 error) + func (@"bytes".b·2 *@"bytes".Buffer) @"bytes".grow (@"bytes".n·3 int) (? int) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x1") @"bytes".readSlice (@"bytes".delim·4 byte) (@"bytes".line·1 []byte, @"bytes".err·2 error) + type @"mime/multipart".Part struct { Header @"net/textproto".MIMEHeader; @"mime/multipart".buffer *@"bytes".Buffer; @"mime/multipart".mr *@"mime/multipart".Reader; @"mime/multipart".bytesRead int; @"mime/multipart".disposition string; @"mime/multipart".dispositionParams map[string]string; @"mime/multipart".r @"io".Reader } + func (@"mime/multipart".p·2 *@"mime/multipart".Part) Close () (? error) + func (@"mime/multipart".p·2 *@"mime/multipart".Part "esc:0x0") FileName () (? string) + func (@"mime/multipart".p·2 *@"mime/multipart".Part "esc:0x0") FormName () (? string) + func (@"mime/multipart".p·3 *@"mime/multipart".Part) Read (@"mime/multipart".d·4 []byte) (@"mime/multipart".n·1 int, @"mime/multipart".err·2 error) + func (@"mime/multipart".p·1 *@"mime/multipart".Part "esc:0x0") @"mime/multipart".parseContentDisposition () + func (@"mime/multipart".bp·2 *@"mime/multipart".Part) @"mime/multipart".populateHeaders () (? error) + type @"mime/multipart".Reader struct { @"mime/multipart".bufReader *@"bufio".Reader; @"mime/multipart".currentPart *@"mime/multipart".Part; @"mime/multipart".partsRead int; @"mime/multipart".nl []byte; @"mime/multipart".nlDashBoundary []byte; @"mime/multipart".dashBoundaryDash []byte; @"mime/multipart".dashBoundary []byte } + func (@"mime/multipart".r·3 *@"mime/multipart".Reader) NextPart () (? *@"mime/multipart".Part, ? error) + func (@"mime/multipart".r·3 *@"mime/multipart".Reader) ReadForm (@"mime/multipart".maxMemory·4 int64) (@"mime/multipart".f·1 *@"mime/multipart".Form, @"mime/multipart".err·2 error) + func (@"mime/multipart".mr·2 *@"mime/multipart".Reader) @"mime/multipart".isBoundaryDelimiterLine (@"mime/multipart".line·3 []byte "esc:0x0") (@"mime/multipart".ret·1 bool) + func (@"mime/multipart".mr·2 *@"mime/multipart".Reader "esc:0x0") @"mime/multipart".isFinalBoundary (@"mime/multipart".line·3 []byte "esc:0x0") (? bool) + func (@"mime/multipart".mr·2 *@"mime/multipart".Reader "esc:0x0") @"mime/multipart".peekBufferIsEmptyPart (@"mime/multipart".peek·3 []byte "esc:0x0") (? bool) + type @"net/http".Request struct { Method string; URL *@"net/url".URL; Proto string; ProtoMajor int; ProtoMinor int; Header @"net/http".Header; Body @"io".ReadCloser; ContentLength int64; TransferEncoding []string; Close bool; Host string; Form @"net/url".Values; PostForm @"net/url".Values; MultipartForm *@"mime/multipart".Form; Trailer @"net/http".Header; RemoteAddr string; RequestURI string; TLS *@"crypto/tls".ConnectionState } + func (@"net/http".r·1 *@"net/http".Request "esc:0x0") AddCookie (@"net/http".c·2 *@"net/http".Cookie) + func (@"net/http".r·4 *@"net/http".Request "esc:0x0") BasicAuth () (@"net/http".username·1 string, @"net/http".password·2 string, @"net/http".ok·3 bool) + func (@"net/http".r·3 *@"net/http".Request "esc:0x0") Cookie (@"net/http".name·4 string "esc:0x0") (? *@"net/http".Cookie, ? error) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") Cookies () (? []*@"net/http".Cookie) + func (@"net/http".r·4 *@"net/http".Request) FormFile (@"net/http".key·5 string "esc:0x0") (? @"mime/multipart".File, ? *@"mime/multipart".FileHeader, ? error) + func (@"net/http".r·2 *@"net/http".Request) FormValue (@"net/http".key·3 string "esc:0x0") (? string) + func (@"net/http".r·3 *@"net/http".Request) MultipartReader () (? *@"mime/multipart".Reader, ? error) + func (@"net/http".r·2 *@"net/http".Request) ParseForm () (? error) + func (@"net/http".r·2 *@"net/http".Request) ParseMultipartForm (@"net/http".maxMemory·3 int64) (? error) + func (@"net/http".r·2 *@"net/http".Request) PostFormValue (@"net/http".key·3 string "esc:0x0") (? string) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") ProtoAtLeast (@"net/http".major·3 int, @"net/http".minor·4 int) (? bool) { return @"net/http".r·2.ProtoMajor > @"net/http".major·3 || @"net/http".r·2.ProtoMajor == @"net/http".major·3 && @"net/http".r·2.ProtoMinor >= @"net/http".minor·4 } + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") Referer () (? string) + func (@"net/http".r·1 *@"net/http".Request "esc:0x0") SetBasicAuth (@"net/http".username·2 string "esc:0x0", @"net/http".password·3 string "esc:0x0") + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") UserAgent () (? string) + func (@"net/http".r·2 *@"net/http".Request) Write (@"net/http".w·3 @"io".Writer) (? error) + func (@"net/http".r·2 *@"net/http".Request) WriteProxy (@"net/http".w·3 @"io".Writer) (? error) + func (@"net/http".r·1 *@"net/http".Request) @"net/http".closeBody () + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") @"net/http".expectsContinue () (? bool) + func (@"net/http".r·3 *@"net/http".Request) @"net/http".multipartReader () (? *@"mime/multipart".Reader, ? error) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") @"net/http".wantsClose () (? bool) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") @"net/http".wantsHttp10KeepAlive () (? bool) + func (@"net/http".req·2 *@"net/http".Request) @"net/http".write (@"net/http".w·3 @"io".Writer, @"net/http".usingProxy·4 bool, @"net/http".extraHeaders·5 @"net/http".Header "esc:0x0") (? error) + type @"net/http".Handler interface { ServeHTTP(? @"net/http".ResponseWriter, ? *@"net/http".Request) } + type @"".RouteMatch struct { Route *@"".Route; Handler @"net/http".Handler; Vars map[string]string } + type @"".matcher interface { Match(? *@"net/http".Request, ? *@"".RouteMatch) (? bool) } + import syntax "regexp/syntax" // indirect + type @"regexp/syntax".InstOp uint8 + func (@"regexp/syntax".i·2 @"regexp/syntax".InstOp) String () (? string) { if uint(@"regexp/syntax".i·2) >= uint(len(@"regexp/syntax".instOpNames)) { return "" }; return @"regexp/syntax".instOpNames[@"regexp/syntax".i·2] } + type @"regexp/syntax".Inst struct { Op @"regexp/syntax".InstOp; Out uint32; Arg uint32; Rune []rune } + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") MatchEmptyWidth (@"regexp/syntax".before·3 rune, @"regexp/syntax".after·4 rune) (? bool) + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") MatchRune (@"regexp/syntax".r·3 rune) (? bool) + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") MatchRunePos (@"regexp/syntax".r·3 rune) (? int) + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") String () (? string) + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") @"regexp/syntax".op () (? @"regexp/syntax".InstOp) + type @"regexp/syntax".EmptyOp uint8 + type @"regexp/syntax".Prog struct { Inst []@"regexp/syntax".Inst; Start int; NumCap int } + func (@"regexp/syntax".p·3 *@"regexp/syntax".Prog "esc:0x0") Prefix () (@"regexp/syntax".prefix·1 string, @"regexp/syntax".complete·2 bool) + func (@"regexp/syntax".p·2 *@"regexp/syntax".Prog "esc:0x0") StartCond () (? @"regexp/syntax".EmptyOp) + func (@"regexp/syntax".p·2 *@"regexp/syntax".Prog "esc:0x0") String () (? string) + func (@"regexp/syntax".p·3 *@"regexp/syntax".Prog "esc:0x1") @"regexp/syntax".skipNop (@"regexp/syntax".pc·4 uint32) (? *@"regexp/syntax".Inst, ? uint32) + type @"regexp".onePassInst struct { ? @"regexp/syntax".Inst; Next []uint32 } + type @"regexp".onePassProg struct { Inst []@"regexp".onePassInst; Start int; NumCap int } + import sync "sync" // indirect + type @"sync".Mutex struct { @"sync".state int32; @"sync".sema uint32 } + func (@"sync".m·1 *@"sync".Mutex) Lock () + func (@"sync".m·1 *@"sync".Mutex) Unlock () + type @"regexp".thread struct { @"regexp".inst *@"regexp/syntax".Inst; @"regexp".cap []int } + type @"regexp".entry struct { @"regexp".pc uint32; @"regexp".t *@"regexp".thread } + type @"regexp".queue struct { @"regexp".sparse []uint32; @"regexp".dense []@"regexp".entry } + type @"regexp".inputBytes struct { @"regexp".str []byte } + func (@"regexp".i·2 *@"regexp".inputBytes "esc:0x0") @"regexp".canCheckPrefix () (? bool) { return true } + func (@"regexp".i·2 *@"regexp".inputBytes "esc:0x0") @"regexp".context (@"regexp".pos·3 int) (? @"regexp/syntax".EmptyOp) + func (@"regexp".i·2 *@"regexp".inputBytes "esc:0x0") @"regexp".hasPrefix (@"regexp".re·3 *@"regexp".Regexp "esc:0x0") (? bool) + func (@"regexp".i·2 *@"regexp".inputBytes "esc:0x0") @"regexp".index (@"regexp".re·3 *@"regexp".Regexp "esc:0x0", @"regexp".pos·4 int) (? int) + func (@"regexp".i·3 *@"regexp".inputBytes "esc:0x0") @"regexp".step (@"regexp".pos·4 int) (? rune, ? int) + type @"regexp".inputString struct { @"regexp".str string } + func (@"regexp".i·2 *@"regexp".inputString "esc:0x0") @"regexp".canCheckPrefix () (? bool) { return true } + func (@"regexp".i·2 *@"regexp".inputString "esc:0x0") @"regexp".context (@"regexp".pos·3 int) (? @"regexp/syntax".EmptyOp) + func (@"regexp".i·2 *@"regexp".inputString "esc:0x0") @"regexp".hasPrefix (@"regexp".re·3 *@"regexp".Regexp "esc:0x0") (? bool) + func (@"regexp".i·2 *@"regexp".inputString "esc:0x0") @"regexp".index (@"regexp".re·3 *@"regexp".Regexp "esc:0x0", @"regexp".pos·4 int) (? int) + func (@"regexp".i·3 *@"regexp".inputString "esc:0x0") @"regexp".step (@"regexp".pos·4 int) (? rune, ? int) + type @"io".RuneReader interface { ReadRune() (@"io".r rune, @"io".size int, @"io".err error) } + type @"regexp".inputReader struct { @"regexp".r @"io".RuneReader; @"regexp".atEOT bool; @"regexp".pos int } + func (@"regexp".i·2 *@"regexp".inputReader "esc:0x0") @"regexp".canCheckPrefix () (? bool) { return false } + func (@"regexp".i·2 *@"regexp".inputReader "esc:0x0") @"regexp".context (@"regexp".pos·3 int) (? @"regexp/syntax".EmptyOp) { return @"regexp/syntax".EmptyOp(0x0) } + func (@"regexp".i·2 *@"regexp".inputReader "esc:0x0") @"regexp".hasPrefix (@"regexp".re·3 *@"regexp".Regexp "esc:0x0") (? bool) { return false } + func (@"regexp".i·2 *@"regexp".inputReader "esc:0x0") @"regexp".index (@"regexp".re·3 *@"regexp".Regexp "esc:0x0", @"regexp".pos·4 int) (? int) { return -0x1 } + func (@"regexp".i·3 *@"regexp".inputReader) @"regexp".step (@"regexp".pos·4 int) (? rune, ? int) + type @"regexp".input interface { @"regexp".canCheckPrefix() (? bool); @"regexp".context(@"regexp".pos int) (? @"regexp/syntax".EmptyOp); @"regexp".hasPrefix(@"regexp".re *@"regexp".Regexp) (? bool); @"regexp".index(@"regexp".re *@"regexp".Regexp, @"regexp".pos int) (? int); @"regexp".step(@"regexp".pos int) (@"regexp".r rune, @"regexp".width int) } + type @"regexp".machine struct { @"regexp".re *@"regexp".Regexp; @"regexp".p *@"regexp/syntax".Prog; @"regexp".op *@"regexp".onePassProg; @"regexp".q0 @"regexp".queue; @"regexp".q1 @"regexp".queue; @"regexp".pool []*@"regexp".thread; @"regexp".matched bool; @"regexp".matchcap []int; @"regexp".inputBytes @"regexp".inputBytes; @"regexp".inputString @"regexp".inputString; @"regexp".inputReader @"regexp".inputReader } + func (@"regexp".m·2 *@"regexp".machine) @"regexp".add (@"regexp".q·3 *@"regexp".queue, @"regexp".pc·4 uint32, @"regexp".pos·5 int, @"regexp".cap·6 []int "esc:0x0", @"regexp".cond·7 @"regexp/syntax".EmptyOp, @"regexp".t·8 *@"regexp".thread) (? *@"regexp".thread) + func (@"regexp".m·2 *@"regexp".machine) @"regexp".alloc (@"regexp".i·3 *@"regexp/syntax".Inst) (? *@"regexp".thread) + func (@"regexp".m·1 *@"regexp".machine) @"regexp".clear (@"regexp".q·2 *@"regexp".queue) + func (@"regexp".m·1 *@"regexp".machine) @"regexp".free (@"regexp".t·2 *@"regexp".thread) { @"regexp".m·1.@"regexp".inputBytes.@"regexp".str = nil; @"regexp".m·1.@"regexp".inputString.@"regexp".str = ""; @"regexp".m·1.@"regexp".inputReader.@"regexp".r = nil; @"regexp".m·1.@"regexp".pool = append(@"regexp".m·1.@"regexp".pool, @"regexp".t·2) } + func (@"regexp".m·1 *@"regexp".machine) @"regexp".init (@"regexp".ncap·2 int) + func (@"regexp".m·2 *@"regexp".machine) @"regexp".match (@"regexp".i·3 @"regexp".input, @"regexp".pos·4 int) (? bool) + func (@"regexp".m·2 *@"regexp".machine "esc:0x2") @"regexp".newInputBytes (@"regexp".b·3 []byte) (? @"regexp".input) { @"regexp".m·2.@"regexp".inputBytes.@"regexp".str = @"regexp".b·3; return &@"regexp".m·2.@"regexp".inputBytes } + func (@"regexp".m·2 *@"regexp".machine "esc:0x2") @"regexp".newInputReader (@"regexp".r·3 @"io".RuneReader) (? @"regexp".input) { @"regexp".m·2.@"regexp".inputReader.@"regexp".r = @"regexp".r·3; @"regexp".m·2.@"regexp".inputReader.@"regexp".atEOT = false; @"regexp".m·2.@"regexp".inputReader.@"regexp".pos = 0x0; return &@"regexp".m·2.@"regexp".inputReader } + func (@"regexp".m·2 *@"regexp".machine "esc:0x2") @"regexp".newInputString (@"regexp".s·3 string) (? @"regexp".input) { @"regexp".m·2.@"regexp".inputString.@"regexp".str = @"regexp".s·3; return &@"regexp".m·2.@"regexp".inputString } + func (@"regexp".m·2 *@"regexp".machine) @"regexp".onepass (@"regexp".i·3 @"regexp".input, @"regexp".pos·4 int) (? bool) + func (@"regexp".m·1 *@"regexp".machine) @"regexp".step (@"regexp".runq·2 *@"regexp".queue, @"regexp".nextq·3 *@"regexp".queue, @"regexp".pos·4 int, @"regexp".nextPos·5 int, @"regexp".c·6 rune, @"regexp".nextCond·7 @"regexp/syntax".EmptyOp) + type @"regexp".Regexp struct { @"regexp".expr string; @"regexp".prog *@"regexp/syntax".Prog; @"regexp".onepass *@"regexp".onePassProg; @"regexp".prefix string; @"regexp".prefixBytes []byte; @"regexp".prefixComplete bool; @"regexp".prefixRune rune; @"regexp".prefixEnd uint32; @"regexp".cond @"regexp/syntax".EmptyOp; @"regexp".numSubexp int; @"regexp".subexpNames []string; @"regexp".longest bool; @"regexp".mu @"sync".Mutex; @"regexp".machine []*@"regexp".machine } + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") Expand (@"regexp".dst·3 []byte "esc:0x2", @"regexp".template·4 []byte "esc:0x0", @"regexp".src·5 []byte "esc:0x0", @"regexp".match·6 []int "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") ExpandString (@"regexp".dst·3 []byte "esc:0x2", @"regexp".template·4 string, @"regexp".src·5 string "esc:0x0", @"regexp".match·6 []int "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) Find (@"regexp".b·3 []byte) (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) FindAll (@"regexp".b·3 []byte, @"regexp".n·4 int) (? [][]byte) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllIndex (@"regexp".b·3 []byte, @"regexp".n·4 int) (? [][]int) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllString (@"regexp".s·3 string, @"regexp".n·4 int) (? []string) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllStringIndex (@"regexp".s·3 string, @"regexp".n·4 int) (? [][]int) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllStringSubmatch (@"regexp".s·3 string, @"regexp".n·4 int) (? [][]string) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllStringSubmatchIndex (@"regexp".s·3 string, @"regexp".n·4 int) (? [][]int) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllSubmatch (@"regexp".b·3 []byte, @"regexp".n·4 int) (? [][][]byte) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllSubmatchIndex (@"regexp".b·3 []byte, @"regexp".n·4 int) (? [][]int) + func (@"regexp".re·2 *@"regexp".Regexp) FindIndex (@"regexp".b·3 []byte) (@"regexp".loc·1 []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindReaderIndex (@"regexp".r·3 @"io".RuneReader) (@"regexp".loc·1 []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindReaderSubmatchIndex (@"regexp".r·3 @"io".RuneReader) (? []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindString (@"regexp".s·3 string) (? string) + func (@"regexp".re·2 *@"regexp".Regexp) FindStringIndex (@"regexp".s·3 string) (@"regexp".loc·1 []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindStringSubmatch (@"regexp".s·3 string) (? []string) + func (@"regexp".re·2 *@"regexp".Regexp) FindStringSubmatchIndex (@"regexp".s·3 string) (? []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindSubmatch (@"regexp".b·3 []byte) (? [][]byte) + func (@"regexp".re·2 *@"regexp".Regexp) FindSubmatchIndex (@"regexp".b·3 []byte) (? []int) + func (@"regexp".re·3 *@"regexp".Regexp "esc:0x1") LiteralPrefix () (@"regexp".prefix·1 string, @"regexp".complete·2 bool) { return @"regexp".re·3.@"regexp".prefix, @"regexp".re·3.@"regexp".prefixComplete } + func (@"regexp".re·1 *@"regexp".Regexp "esc:0x0") Longest () { @"regexp".re·1.@"regexp".longest = true } + func (@"regexp".re·2 *@"regexp".Regexp) Match (@"regexp".b·3 []byte) (? bool) + func (@"regexp".re·2 *@"regexp".Regexp) MatchReader (@"regexp".r·3 @"io".RuneReader) (? bool) + func (@"regexp".re·2 *@"regexp".Regexp) MatchString (@"regexp".s·3 string) (? bool) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") NumSubexp () (? int) { return @"regexp".re·2.@"regexp".numSubexp } + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAll (@"regexp".src·3 []byte, @"regexp".repl·4 []byte "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllFunc (@"regexp".src·3 []byte, @"regexp".repl·4 func(? []byte) (? []byte) "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllLiteral (@"regexp".src·3 []byte, @"regexp".repl·4 []byte "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllLiteralString (@"regexp".src·3 string, @"regexp".repl·4 string "esc:0x0") (? string) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllString (@"regexp".src·3 string, @"regexp".repl·4 string) (? string) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllStringFunc (@"regexp".src·3 string, @"regexp".repl·4 func(? string) (? string) "esc:0x0") (? string) + func (@"regexp".re·2 *@"regexp".Regexp) Split (@"regexp".s·3 string, @"regexp".n·4 int) (? []string) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x1") String () (? string) { return @"regexp".re·2.@"regexp".expr } + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x1") SubexpNames () (? []string) { return @"regexp".re·2.@"regexp".subexpNames } + func (@"regexp".re·1 *@"regexp".Regexp) @"regexp".allMatches (@"regexp".s·2 string, @"regexp".b·3 []byte, @"regexp".n·4 int, @"regexp".deliver·5 func(? []int) "esc:0x0") + func (@"regexp".re·2 *@"regexp".Regexp) @"regexp".doExecute (@"regexp".r·3 @"io".RuneReader, @"regexp".b·4 []byte, @"regexp".s·5 string, @"regexp".pos·6 int, @"regexp".ncap·7 int) (? []int) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") @"regexp".expand (@"regexp".dst·3 []byte "esc:0x2", @"regexp".template·4 string, @"regexp".bsrc·5 []byte "esc:0x0", @"regexp".src·6 string "esc:0x0", @"regexp".match·7 []int "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) @"regexp".get () (? *@"regexp".machine) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") @"regexp".pad (@"regexp".a·3 []int "esc:0x2") (? []int) + func (@"regexp".re·1 *@"regexp".Regexp) @"regexp".put (@"regexp".z·2 *@"regexp".machine) + func (@"regexp".re·2 *@"regexp".Regexp) @"regexp".replaceAll (@"regexp".bsrc·3 []byte, @"regexp".src·4 string, @"regexp".nmatch·5 int, @"regexp".repl·6 func(@"regexp".dst []byte, @"regexp".m []int) (? []byte) "esc:0x0") (? []byte) + type @"".routeRegexp struct { @"".template string; @"".matchHost bool; @"".matchQuery bool; @"".strictSlash bool; @"".regexp *@"regexp".Regexp; @"".reverse string; @"".varsN []string; @"".varsR []*@"regexp".Regexp } + func (@"".r·2 *@"".routeRegexp) Match (@"".req·3 *@"net/http".Request, @"".match·4 *@"".RouteMatch "esc:0x0") (? bool) + func (@"".r·2 *@"".routeRegexp "esc:0x0") @"".getUrlQuery (@"".req·3 *@"net/http".Request) (? string) + func (@"".r·2 *@"".routeRegexp) @"".matchQueryString (@"".req·3 *@"net/http".Request) (? bool) + func (@"".r·3 *@"".routeRegexp) @"".url (@"".values·4 map[string]string "esc:0x0") (? string, ? error) + type @"".routeRegexpGroup struct { @"".host *@"".routeRegexp; @"".path *@"".routeRegexp; @"".queries []*@"".routeRegexp } + func (@"".v·1 *@"".routeRegexpGroup) @"".setMatch (@"".req·2 *@"net/http".Request, @"".m·3 *@"".RouteMatch "esc:0x0", @"".r·4 *@"".Route "esc:0x0") + type @"".BuildVarsFunc func(? map[string]string) (? map[string]string) + type @"".MatcherFunc func(? *@"net/http".Request, ? *@"".RouteMatch) (? bool) + func (@"".m·2 @"".MatcherFunc "esc:0x0") Match (@"".r·3 *@"net/http".Request, @"".match·4 *@"".RouteMatch) (? bool) + type @"".Route struct { @"".parent @"".parentRoute; @"".handler @"net/http".Handler; @"".matchers []@"".matcher; @"".regexp *@"".routeRegexpGroup; @"".strictSlash bool; @"".buildOnly bool; @"".name string; @"".err error; @"".buildVarsFunc @"".BuildVarsFunc } + func (@"".r·2 *@"".Route "esc:0x2") BuildOnly () (? *@"".Route) { @"".r·2.@"".buildOnly = true; return @"".r·2 } + func (@"".r·2 *@"".Route "esc:0x2") BuildVarsFunc (@"".f·3 @"".BuildVarsFunc) (? *@"".Route) { @"".r·2.@"".buildVarsFunc = @"".f·3; return @"".r·2 } + func (@"".r·2 *@"".Route "esc:0x1") GetError () (? error) { return @"".r·2.@"".err } + func (@"".r·2 *@"".Route "esc:0x1") GetHandler () (? @"net/http".Handler) { return @"".r·2.@"".handler } + func (@"".r·2 *@"".Route "esc:0x1") GetName () (? string) { return @"".r·2.@"".name } + func (@"".r·2 *@"".Route "esc:0x2") Handler (@"".handler·3 @"net/http".Handler) (? *@"".Route) { if @"".r·2.@"".err == nil { @"".r·2.@"".handler = @"".handler·3 }; return @"".r·2 } + func (@"".r·2 *@"".Route "esc:0x2") HandlerFunc (@"".f·3 func(? @"net/http".ResponseWriter, ? *@"net/http".Request)) (? *@"".Route) + func (@"".r·2 *@"".Route) Headers (@"".pairs·3 ...string) (? *@"".Route) + func (@"".r·2 *@"".Route) HeadersRegexp (@"".pairs·3 ...string) (? *@"".Route) + func (@"".r·2 *@"".Route) Host (@"".tpl·3 string) (? *@"".Route) + func (@"".r·2 *@"".Route) Match (@"".req·3 *@"net/http".Request, @"".match·4 *@"".RouteMatch) (? bool) + func (@"".r·2 *@"".Route) MatcherFunc (@"".f·3 @"".MatcherFunc) (? *@"".Route) + func (@"".r·2 *@"".Route) Methods (@"".methods·3 ...string) (? *@"".Route) + func (@"".r·2 *@"".Route) Name (@"".name·3 string) (? *@"".Route) + func (@"".r·2 *@"".Route) Path (@"".tpl·3 string) (? *@"".Route) + func (@"".r·2 *@"".Route) PathPrefix (@"".tpl·3 string) (? *@"".Route) + func (@"".r·2 *@"".Route) Queries (@"".pairs·3 ...string) (? *@"".Route) + func (@"".r·2 *@"".Route) Schemes (@"".schemes·3 ...string) (? *@"".Route) + func (@"".r·2 *@"".Route) Subrouter () (? *@"".Router) + func (@"".r·3 *@"".Route) URL (@"".pairs·4 ...string) (? *@"net/url".URL, ? error) + func (@"".r·3 *@"".Route) URLHost (@"".pairs·4 ...string) (? *@"net/url".URL, ? error) + func (@"".r·3 *@"".Route) URLPath (@"".pairs·4 ...string) (? *@"net/url".URL, ? error) + func (@"".r·2 *@"".Route) @"".addMatcher (@"".m·3 @"".matcher) (? *@"".Route) { if @"".r·2.@"".err == nil { @"".r·2.@"".matchers = append(@"".r·2.@"".matchers, @"".m·3) }; return @"".r·2 } + func (@"".r·2 *@"".Route) @"".addRegexpMatcher (@"".tpl·3 string, @"".matchHost·4 bool, @"".matchPrefix·5 bool, @"".matchQuery·6 bool) (? error) + func (@"".r·2 *@"".Route) @"".buildVars (@"".m·3 map[string]string) (? map[string]string) + func (@"".r·2 *@"".Route) @"".getNamedRoutes () (? map[string]*@"".Route) + func (@"".r·2 *@"".Route) @"".getRegexpGroup () (? *@"".routeRegexpGroup) + func (@"".r·3 *@"".Route) @"".prepareVars (@"".pairs·4 ...string) (? map[string]string, ? error) + type @"".parentRoute interface { @"".buildVars(? map[string]string) (? map[string]string); @"".getNamedRoutes() (? map[string]*@"".Route); @"".getRegexpGroup() (? *@"".routeRegexpGroup) } + type @"".WalkFunc func(@"".route *@"".Route, @"".router *@"".Router, @"".ancestors []*@"".Route) (? error) + type @"".Router struct { NotFoundHandler @"net/http".Handler; @"".parent @"".parentRoute; @"".routes []*@"".Route; @"".namedRoutes map[string]*@"".Route; @"".strictSlash bool; KeepContext bool } + func (@"".r·2 *@"".Router) BuildVarsFunc (@"".f·3 @"".BuildVarsFunc) (? *@"".Route) + func (@"".r·2 *@"".Router) Get (@"".name·3 string "esc:0x0") (? *@"".Route) + func (@"".r·2 *@"".Router) GetRoute (@"".name·3 string "esc:0x0") (? *@"".Route) + func (@"".r·2 *@"".Router) Handle (@"".path·3 string, @"".handler·4 @"net/http".Handler) (? *@"".Route) + func (@"".r·2 *@"".Router) HandleFunc (@"".path·3 string, @"".f·4 func(? @"net/http".ResponseWriter, ? *@"net/http".Request)) (? *@"".Route) + func (@"".r·2 *@"".Router) Headers (@"".pairs·3 ...string) (? *@"".Route) + func (@"".r·2 *@"".Router) Host (@"".tpl·3 string) (? *@"".Route) + func (@"".r·2 *@"".Router "esc:0x0") Match (@"".req·3 *@"net/http".Request, @"".match·4 *@"".RouteMatch) (? bool) + func (@"".r·2 *@"".Router) MatcherFunc (@"".f·3 @"".MatcherFunc) (? *@"".Route) + func (@"".r·2 *@"".Router) Methods (@"".methods·3 ...string) (? *@"".Route) + func (@"".r·2 *@"".Router) NewRoute () (? *@"".Route) { var @"".route·3 *@"".Route; @"".route·3 = (&@"".Route{ @"".parent:@"".r·2, @"".strictSlash:@"".r·2.@"".strictSlash }); @"".r·2.@"".routes = append(@"".r·2.@"".routes, @"".route·3); return @"".route·3 } + func (@"".r·2 *@"".Router) Path (@"".tpl·3 string) (? *@"".Route) + func (@"".r·2 *@"".Router) PathPrefix (@"".tpl·3 string) (? *@"".Route) + func (@"".r·2 *@"".Router) Queries (@"".pairs·3 ...string) (? *@"".Route) + func (@"".r·2 *@"".Router) Schemes (@"".schemes·3 ...string) (? *@"".Route) + func (@"".r·1 *@"".Router) ServeHTTP (@"".w·2 @"net/http".ResponseWriter, @"".req·3 *@"net/http".Request) + func (@"".r·2 *@"".Router "esc:0x2") StrictSlash (@"".value·3 bool) (? *@"".Router) { @"".r·2.@"".strictSlash = @"".value·3; return @"".r·2 } + func (@"".r·2 *@"".Router) Walk (@"".walkFn·3 @"".WalkFunc "esc:0x0") (? error) + func (@"".r·2 *@"".Router) @"".buildVars (@"".m·3 map[string]string) (? map[string]string) + func (@"".r·2 *@"".Router) @"".getNamedRoutes () (? map[string]*@"".Route) + func (@"".r·2 *@"".Router) @"".getRegexpGroup () (? *@"".routeRegexpGroup) + func (@"".r·2 *@"".Router) @"".walk (@"".walkFn·3 @"".WalkFunc "esc:0x0", @"".ancestors·4 []*@"".Route) (? error) + func @"".NewRouter () (? *@"".Router) { return (&@"".Router{ @"".namedRoutes:make(map[string]*@"".Route, 0x0), KeepContext:false }) } + var @"".SkipRouter error + func @"".Vars (@"".r·2 *@"net/http".Request "esc:0x0") (? map[string]string) + func @"".CurrentRoute (@"".r·2 *@"net/http".Request "esc:0x0") (? *@"".Route) + func @"".init () + var @"time".months [12]string + var @"time".days [7]string + var @"time".Local *@"time".Location + var @"time".UTC *@"time".Location + var @"bufio".ErrInvalidUnreadRune error + var @"regexp/syntax".instOpNames []string + +$$ +_go_.6 0 0 0 644 311105 ` +go object darwin amd64 go1.4.2 X:precisestack + +! +go13lderrors.a +fmt.anet/http.a path.aregexp.a�github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.abytes.anet/url.astrconv.astrings.a�"".NewRouter��eH� %H;aw���H��(H�H�$H�D$�H�\$H�\$ H�H�$�H�L$H��H��tI1��H�L$H� $H�<$t)H�$8H�\$ H�\$�H�D$1�@�hAH�D$0H��(É%�Ή� + 0runtime.morestack_noctxt:2type.map[string]*"".Route^runtime.makemap�type."".Router�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptrP"".autotmp_0002type.*"".Router"".autotmp_00012type.map[string]*"".Route "".~r0type.*"".RouterP�OP� &�.9/Tgclocals·37da6a443256db8ec55c7210d030a9b0Tgclocals·f6dcde45bff02c6c4b088b594fd52a4c�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�$"".(*Router).Match��eH� %H;aw���H��PH�\$XH����H�S H�C(H�k0H�l$H1�H�D$@H�D$ H�T$8H��H�l$ H9�}YH�D$0H�(H�L$(H�,$H�\$`H�\$H�\$hH�\$��\$��t +�D$pH��P�H�D$0H�L$(H��H��H�l$ H9�|��D$pH��PÉ�g��� + 0runtime.morestack_noctxt�""".(*Route).Match@�"".autotmp_0007?type.**"".Route"".autotmp_0006_type.int"".autotmp_0005Otype.int"".autotmp_0003/ type.[]*"".Route "".~r20type.bool"".match &type.*"".RouteMatch "".req,type.*net/http.Request"".rtype.*"".Router$�{��$���rK' + + +  +}STgclocals·d69c4140875de858f5dc9e2e8acb0bc0Tgclocals·29f0050a5ee7c2b9348a75428171d7de�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�,"".(*Router).ServeHTTP��eH� %H�D$�H;Aw���H��H��$ H�~H����H�w8H�<$H�H��H�L$H�T$H�L$HH��$ H�kH���lH�M8H��$�H�E@H��$�H9��bH�l$HH�,$H�T$PH�T$H�L$H�D$�H�T$P�\$ ���-H�H�$�H�D$H�D$XH�D$`H��$H�$H��$ H�\$H�D$0H�D$�H�L$0�\$����H�iH�l$XH�iH�l$`H��$ H�$H�iH�H�D$hH�D$H�l$pH�l$�H��$ H�$H�\$0H�+H�H�D$hH�D$H�l$pH�l$�H�|$X��H��$H���+H�+H�l$XH�kH�l$`H�|$XuVH�D$8H�D$@H�H�\$(H� 1�H9���H�T$(H�L$xH��$�H�L$8H�L$XH�T$@H�T$`H��$�]A��u"H��$ H�$H� Qj�YYH��uIH��$H�\$H��$H�\$H��$ H�\$H�\$`H�$H�\$XH�[ �Ӑ�H��Ð�H���H�H�$H�H�\$H�H�\$�H�L$��������������H��$ H�wH����H��$��H�\$HH��$�H�T$PH��$�H��$�H�$�H�T$H�L$H�T$HH�L$PH��$H�$H��$H�[ ��H�t$H�4$H�5H�l$H��H�H�H�\$HH�\$H�\$PH�\$ �H�D$-H��$H�$H��$H�[0�Ӑ�H��É�-����E������O���: +*0runtime.morestack_noctxt�"".cleanPath� runtime.eqstring�$type."".RouteMatch�"runtime.newobject�$"".(*Router).Match�,type.map[string]string�"".setVars�type.*"".Route�$"".setCurrentRoute�(net/http.NotFound·f�Zgo.itab.net/http.HandlerFunc.net/http.Handler� �github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.Clear·f� "runtime.deferproc� + +� +&runtime.deferreturn� +&runtime.deferreturn� +2type.net/http.HandlerFunc� *type.net/http.Handler� Zgo.itab.net/http.HandlerFunc.net/http.Handler�  runtime.typ2Itab� � runtime.duffcopy� *net/url.(*URL).String� +� (go.string."Location"�&net/http.Header.Set� +�&runtime.deferreturn@�"".autotmp_0015�*type.net/http.Handler"".autotmp_0014type.string"".autotmp_0013�2type.net/http.HandlerFunc"".autotmp_0010type.string"".autotmp_0009�type.string"".&match�&type.*"".RouteMatch "".~r0�*type.net/http.Handler"".handler�*type.net/http.Handler "".url� type.net/url.URL"".p�type.string "".req0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".rtype.*"".RouterD"��M�� ������v�"�;04 #V";2#%O#  2Dh#?��Ad12STgclocals·7a383875e23784cb158d762414ce6278Tgclocals·4c1561a135d5ed5147fd4ff64ff73c94�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go� "".(*Router).Get��eH� %H;aw���H��8H�\$@H�$�H�D$H�T$HH�L$PH�H�$H�D$H�T$(H�T$H�L$0H�L$�H�\$ H�+H�l$XH��8� + + 0runtime.morestack_noctxtH6"".(*Router).getNamedRoutest2type.map[string]*"".Route�4runtime.mapaccess1_faststr@p"".autotmp_0023type.string "".~r10type.*"".Route"".nametype.string"".rtype.*"".RouterpWo� �f +#]Tgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�*"".(*Router).GetRoute��eH� %H;aw���H��8H�\$@H�$�H�D$H�T$HH�L$PH�H�$H�D$H�T$(H�T$H�L$0H�L$�H�\$ H�+H�l$XH��8� + + 0runtime.morestack_noctxtH6"".(*Router).getNamedRoutest2type.map[string]*"".Route�4runtime.mapaccess1_faststr@p"".autotmp_0026type.string "".~r10type.*"".Route"".nametype.string"".rtype.*"".RouterpWo� �f +#]Tgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�0"".(*Router).StrictSlash@(H�D$�l$@�h@H�D$�0 "".~r1 type.*"".Router"".valuetype.bool"".rtype.*"".Router  � Tgclocals·64b411f0f44be3f38c26e84fc3239091Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�6"".(*Router).getNamedRoutes��eH� %H;aw���H��(H�D$0H�X81�H9�uLH�xt\H�HH�hH�l$ H�,$H�L$H�Y(��H�D$H�\$0H�$H�<$t"H�$8H�D$�H�D$0H�h8H�l$8H��(É%��H�H�$H�D$�H�D$H�\$0H�$H�<$tH�$8H�D$�H�D$0륉%�� + 0runtime.morestack_noctxt� +�.runtime.writebarrierptr�2type.map[string]*"".Route�runtime.makemap�.runtime.writebarrierptr P"".autotmp_00292type.map[string]*"".Route "".~r02type.map[string]*"".Route"".rtype.*"".RouterPiOP\�(� E + B K�Tgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�6"".(*Router).getRegexpGroup��eH� %H;aw���H�� H�D$(H�xt+H�HH�hH�l$H�,$H�L$H�Y0��H�\$H�\$0H�� �H�D$0H�� � + 0runtime.morestack_noctxt� + @ "".~r02type.*"".routeRegexpGroup"".rtype.*"".Router@6?@ ?`�+ +@ Tgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�,"".(*Router).buildVars��eH� %H;aw���H��(H�L$0H�D$8H�yt&H�QH�iH�D$H�l$ H�,$H�T$H�Z ��H�D$H�D$@H��(� + 0runtime.morestack_noctxt� +0P "".~r1 ,type.map[string]string"".m,type.map[string]string"".rtype.*"".RouterP@O`�$& +JTgclocals·d3486bc7ce1948dc22d7ad1c0be2887aTgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�*"".(*Router).NewRoute��eH� %H�D$�H;Aw���H��H��$�H�\$PH�H�$�H�|$H��H�|$HH����1��H�1�H9��wH� $H�<$�\H�L$PH�D$XH�D$H�L$`H�L$�H��$�H�D$HH��� �k@@�h@H�D$@H�S H�K(H�[0H��$�H��$�H��$�H��H)�H��}FH�H�$H�T$hH�T$H�L$pH�L$H�D$xH�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$pH�D$xH�T$hH��H�$H�\$@H�\$�H�T$hH�L$pH�D$xH��$�H�$H�<$tFH�$ H��$�H�T$H��$�H�L$H��$�H�D$�H�\$@H��$�H�ĘÉ%뱉������%����H�H�$H�H�\$H�H�\$�H�L$HH�D$�R�����2��� +*0runtime.morestack_noctxtdtype."".Routev"runtime.newobject�� runtime.duffzero�Bgo.itab.*"".Router."".parentRoute�2runtime.writebarrieriface� type.[]*"".Route�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice�type.*"".Router�&type."".parentRoute�Bgo.itab.*"".Router."".parentRoute� runtime.typ2Itab �"".autotmp_0040_ type.[]*"".Route"".autotmp_0039/ type.[]*"".Route"".autotmp_0037�type.*"".Route"".autotmp_0036 type.[]*"".Route"".autotmp_0035�type.*"".Router"".route�type.*"".Route "".~r0type.*"".Route"".rtype.*"".Router"����f� �"�� ] :]�;SY'Tgclocals·0d6246443c3fddb7ffb759a83afd407dTgclocals·691c0cb9316c0a5f7d8580c74ac115f2�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�&"".(*Router).Handle� � eH� %H�D$�H;Aw���H��H��$�H�D$PH�D$`H�H�$�H�|$H��H�|$XH���a1��H�1�H9��H� $H�<$��H�L$`H�D$xH�D$H��$�H�L$�H�\$PH�D$XH�����k@@�h@H�D$@H�S H�K(H�[0H��$�H��$�H��$�H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��$�H��H�$H�\$@H�\$�H��$�H��$�H��$�H�\$PH�$H�<$��H�$ H��$�H�T$H��$�H�L$H��$�H�D$�H�\$@H�$H��$�H�\$H��$�H�\$�H�D$H��$�H��$�H�xXu3H�D$HH�$H�<$t3H�$H�T$hH�T$H�L$pH�L$�H�D$HH��$�H�ĸÉ%�ĉ%�3�����?����%�����H�H�$H�H�\$H�H�\$�H�L$XH�D$���������" +*0runtime.morestack_noctxtntype."".Route�"runtime.newobject�� runtime.duffzero�Bgo.itab.*"".Router."".parentRoute�2runtime.writebarrieriface� type.[]*"".Route�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice� "".(*Route).Path� 2runtime.writebarrieriface� +type.*"".Router� +&type."".parentRoute� +Bgo.itab.*"".Router."".parentRoute� + runtime.typ2Itab`�"".autotmp_0056_ type.[]*"".Route"".autotmp_0055/ type.[]*"".Route"".autotmp_0053�type.*"".Route"".autotmp_0052type.*"".Route"".autotmp_0051 type.[]*"".Route"".autotmp_0050�type.*"".Router"".handler�*type.net/http.Handler"".r�type.*"".Route"".route�type.*"".Route"".r�type.*"".Router "".~r2Ptype.*"".Route"".handler0*type.net/http.Handler"".pathtype.string"".rtype.*"".Router"����s��"�(?`�D]( Je (Tgclocals·259efa0f9d5b5ab4cbb1f7201749d3e1Tgclocals·8cdbdba615b2fb90357456ca3f2cb9a4�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�."".(*Router).HandleFunc� +� eH� %H�D$�H;Aw���H��H��$�H�D$HH�D$XH�H�$�H�|$H��H�|$PH���1��H�1�H9���H� $H�<$��H�L$XH�D$`H�D$H�L$hH�L$�H�\$HH�D$PH���p�k@@�h@H�D$@H�S H�K(H�[0H��$�H��$�H��$�H��H)�H��}IH�H�$H�T$pH�T$H�L$xH�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$xH��$�H�T$pH��H�$H�\$@H�\$�H�T$pH�L$xH��$�H�\$HH�$H�<$��H�$ H��$�H�T$H��$�H�L$H��$�H�D$�H�\$@H�$H��$�H�\$H��$�H�\$�H�\$H�$H��$�H�\$�H�\$H��$�H�ĠÉ%�k���������%�K���H�H�$H�H�\$H�H�\$�H�L$PH�D$����������" +*0runtime.morestack_noctxtntype."".Route�"runtime.newobject�� runtime.duffzero�Bgo.itab.*"".Router."".parentRoute�2runtime.writebarrieriface� type.[]*"".Route�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice� "".(*Route).Path�."".(*Route).HandlerFunc�type.*"".Router� &type."".parentRoute� Bgo.itab.*"".Router."".parentRoute�  runtime.typ2ItabP�"".autotmp_0073_ type.[]*"".Route"".autotmp_0072/ type.[]*"".Route"".autotmp_0070�type.*"".Route"".autotmp_0069type.*"".Route"".autotmp_0068type.*"".Route"".autotmp_0067 type.[]*"".Route"".autotmp_0066�type.*"".Router"".route�type.*"".Route"".r�type.*"".Router "".~r2@type.*"".Route"".f0jtype.func(net/http.ResponseWriter, *net/http.Request)"".pathtype.string"".rtype.*"".Router"����g��"�$?]�>W( w%Tgclocals·1c7793dad628d89b0b03aa7a6b5e8ac7Tgclocals·f24e5ae57611d01ccf1f96d64c337e04�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�("".(*Router).Headers� � eH� %H�D$�H;Aw���H��H��$�H�D$HH�D$XH�H�$�H�|$H��H�|$PH���1��H�1�H9���H� $H�<$��H�L$XH�D$`H�D$H�L$hH�L$�H�\$HH�D$PH���^�k@@�h@H�D$@H�S H�K(H�[0H��$�H��$�H��$�H��H)�H��}IH�H�$H�T$pH�T$H�L$xH�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$xH��$�H�T$pH��H�$H�\$@H�\$�H�T$pH�L$xH��$�H�\$HH�$H�<$t{H�$ H��$�H�T$H��$�H�L$H��$�H�D$�H�\$@H�$H��$�H�\$H��$�H�\$H��$�H�\$�H�\$ H��$�H�ĠÉ%�y���������%�]���H�H�$H�H�\$H�H�\$�H�L$PH�D$���������� +*0runtime.morestack_noctxtntype."".Route�"runtime.newobject�� runtime.duffzero�Bgo.itab.*"".Router."".parentRoute�2runtime.writebarrieriface� type.[]*"".Route�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice�&"".(*Route).Headers�type.*"".Router�&type."".parentRoute� Bgo.itab.*"".Router."".parentRoute�  runtime.typ2ItabP�"".autotmp_0089_ type.[]*"".Route"".autotmp_0088/ type.[]*"".Route"".autotmp_0086�type.*"".Route"".autotmp_0085type.*"".Route"".autotmp_0084 type.[]*"".Route"".autotmp_0083�type.*"".Router"".route�type.*"".Route"".r�type.*"".Router "".~r1@type.*"".Route"".pairstype.[]string"".rtype.*"".Router"����i��"�$?]�>S5 \'Tgclocals·466dbe9b6d0b019671e1c2db1c9f0ba0Tgclocals·f24e5ae57611d01ccf1f96d64c337e04�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�""".(*Router).Host� � eH� %H�D$�H;Aw���H��H��$�H�D$HH�D$XH�H�$�H�|$H��H�|$PH����1��H�1�H9���H� $H�<$��H�L$XH�D$`H�D$H�L$hH�L$�H�\$HH�D$PH���N�k@@�h@H�D$@H�S H�K(H�[0H��$�H��$�H��$�H��H)�H��}IH�H�$H�T$pH�T$H�L$xH�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$xH��$�H�T$pH��H�$H�\$@H�\$�H�T$pH�L$xH��$�H�\$HH�$H�<$tnH�$ H��$�H�T$H��$�H�L$H��$�H�D$�H�\$@H�$H��$�H�\$H��$�H�\$�H�\$H��$�H�ĠÉ%뉉�����%�m���H�H�$H�H�\$H�H�\$�H�L$PH�D$�'�������� +*0runtime.morestack_noctxtntype."".Route�"runtime.newobject�� runtime.duffzero�Bgo.itab.*"".Router."".parentRoute�2runtime.writebarrieriface� type.[]*"".Route�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice� "".(*Route).Host�type.*"".Router�&type."".parentRoute�Bgo.itab.*"".Router."".parentRoute� runtime.typ2Itab@�"".autotmp_0105_ type.[]*"".Route"".autotmp_0104/ type.[]*"".Route"".autotmp_0102�type.*"".Route"".autotmp_0101type.*"".Route"".autotmp_0100 type.[]*"".Route"".autotmp_0099�type.*"".Router"".route�type.*"".Route"".r�type.*"".Router "".~r10type.*"".Route "".tpltype.string"".rtype.*"".Router"����f��"�$?]�>S( Y'Tgclocals·ccff1a4364f53102a1b73e3274c6c0d4Tgclocals·f24e5ae57611d01ccf1f96d64c337e04�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�0"".(*Router).MatcherFunc� � eH� %H�D$�H;Aw���H��H��$�H�D$HH�D$XH�H�$�H�|$H��H�|$PH����1��H�1�H9���H� $H�<$�zH�L$XH�D$`H�D$H�L$hH�L$�H�\$HH�D$PH���A�k@@�h@H�D$@H�S H�K(H�[0H��$�H��$�H��$�H��H)�H��}IH�H�$H�T$pH�T$H�L$xH�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$xH��$�H�T$pH��H�$H�\$@H�\$�H�T$pH�L$xH��$�H�\$HH�$H�<$taH�$ H��$�H�T$H��$�H�L$H��$�H�D$�H�\$@H�$H��$�H�\$�H�\$H��$�H�ĠÉ%떉�����%�z���H�H�$H�H�\$H�H�\$�H�L$PH�D$�4�������� +*0runtime.morestack_noctxtntype."".Route�"runtime.newobject�� runtime.duffzero�Bgo.itab.*"".Router."".parentRoute�2runtime.writebarrieriface� type.[]*"".Route�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice�."".(*Route).MatcherFunc�type.*"".Router�&type."".parentRoute�Bgo.itab.*"".Router."".parentRoute� runtime.typ2Itab0�"".autotmp_0121_ type.[]*"".Route"".autotmp_0120/ type.[]*"".Route"".autotmp_0118�type.*"".Route"".autotmp_0117type.*"".Route"".autotmp_0116 type.[]*"".Route"".autotmp_0115�type.*"".Router"".route�type.*"".Route"".r�type.*"".Router "".~r1 type.*"".Route"".f&type."".MatcherFunc"".rtype.*"".Router"����c��"�$?]�>S Y$Tgclocals·6b6fbfacf80ed81d2be06478c8f1790dTgclocals·f24e5ae57611d01ccf1f96d64c337e04�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�("".(*Router).Methods� � eH� %H�D$�H;Aw���H��H��$�H�D$HH�D$XH�H�$�H�|$H��H�|$PH���1��H�1�H9���H� $H�<$��H�L$XH�D$`H�D$H�L$hH�L$�H�\$HH�D$PH���^�k@@�h@H�D$@H�S H�K(H�[0H��$�H��$�H��$�H��H)�H��}IH�H�$H�T$pH�T$H�L$xH�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$xH��$�H�T$pH��H�$H�\$@H�\$�H�T$pH�L$xH��$�H�\$HH�$H�<$t{H�$ H��$�H�T$H��$�H�L$H��$�H�D$�H�\$@H�$H��$�H�\$H��$�H�\$H��$�H�\$�H�\$ H��$�H�ĠÉ%�y���������%�]���H�H�$H�H�\$H�H�\$�H�L$PH�D$���������� +*0runtime.morestack_noctxtntype."".Route�"runtime.newobject�� runtime.duffzero�Bgo.itab.*"".Router."".parentRoute�2runtime.writebarrieriface� type.[]*"".Route�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice�&"".(*Route).Methods�type.*"".Router�&type."".parentRoute� Bgo.itab.*"".Router."".parentRoute�  runtime.typ2ItabP�"".autotmp_0137_ type.[]*"".Route"".autotmp_0136/ type.[]*"".Route"".autotmp_0134�type.*"".Route"".autotmp_0133type.*"".Route"".autotmp_0132 type.[]*"".Route"".autotmp_0131�type.*"".Router"".route�type.*"".Route"".r�type.*"".Router "".~r1@type.*"".Route"".methodstype.[]string"".rtype.*"".Router"����i��"�$?]�>S5 \'Tgclocals·466dbe9b6d0b019671e1c2db1c9f0ba0Tgclocals·f24e5ae57611d01ccf1f96d64c337e04�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�""".(*Router).Path� � eH� %H�D$�H;Aw���H��H��$�H�D$HH�D$XH�H�$�H�|$H��H�|$PH����1��H�1�H9���H� $H�<$��H�L$XH�D$`H�D$H�L$hH�L$�H�\$HH�D$PH���N�k@@�h@H�D$@H�S H�K(H�[0H��$�H��$�H��$�H��H)�H��}IH�H�$H�T$pH�T$H�L$xH�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$xH��$�H�T$pH��H�$H�\$@H�\$�H�T$pH�L$xH��$�H�\$HH�$H�<$tnH�$ H��$�H�T$H��$�H�L$H��$�H�D$�H�\$@H�$H��$�H�\$H��$�H�\$�H�\$H��$�H�ĠÉ%뉉�����%�m���H�H�$H�H�\$H�H�\$�H�L$PH�D$�'�������� +*0runtime.morestack_noctxtntype."".Route�"runtime.newobject�� runtime.duffzero�Bgo.itab.*"".Router."".parentRoute�2runtime.writebarrieriface� type.[]*"".Route�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice� "".(*Route).Path�type.*"".Router�&type."".parentRoute�Bgo.itab.*"".Router."".parentRoute� runtime.typ2Itab@�"".autotmp_0153_ type.[]*"".Route"".autotmp_0152/ type.[]*"".Route"".autotmp_0150�type.*"".Route"".autotmp_0149type.*"".Route"".autotmp_0148 type.[]*"".Route"".autotmp_0147�type.*"".Router"".route�type.*"".Route"".r�type.*"".Router "".~r10type.*"".Route "".tpltype.string"".rtype.*"".Router"����f��"�$?]�>S( Y'Tgclocals·ccff1a4364f53102a1b73e3274c6c0d4Tgclocals·f24e5ae57611d01ccf1f96d64c337e04�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�."".(*Router).PathPrefix� � eH� %H�D$�H;Aw���H��H��$�H�D$HH�D$XH�H�$�H�|$H��H�|$PH����1��H�1�H9���H� $H�<$��H�L$XH�D$`H�D$H�L$hH�L$�H�\$HH�D$PH���N�k@@�h@H�D$@H�S H�K(H�[0H��$�H��$�H��$�H��H)�H��}IH�H�$H�T$pH�T$H�L$xH�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$xH��$�H�T$pH��H�$H�\$@H�\$�H�T$pH�L$xH��$�H�\$HH�$H�<$tnH�$ H��$�H�T$H��$�H�L$H��$�H�D$�H�\$@H�$H��$�H�\$H��$�H�\$�H�\$H��$�H�ĠÉ%뉉�����%�m���H�H�$H�H�\$H�H�\$�H�L$PH�D$�'�������� +*0runtime.morestack_noctxtntype."".Route�"runtime.newobject�� runtime.duffzero�Bgo.itab.*"".Router."".parentRoute�2runtime.writebarrieriface� type.[]*"".Route�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice�,"".(*Route).PathPrefix�type.*"".Router�&type."".parentRoute�Bgo.itab.*"".Router."".parentRoute� runtime.typ2Itab@�"".autotmp_0169_ type.[]*"".Route"".autotmp_0168/ type.[]*"".Route"".autotmp_0166�type.*"".Route"".autotmp_0165type.*"".Route"".autotmp_0164 type.[]*"".Route"".autotmp_0163�type.*"".Router"".route�type.*"".Route"".r�type.*"".Router "".~r10type.*"".Route "".tpltype.string"".rtype.*"".Router"����f��"�$?]�>S( Y'Tgclocals·ccff1a4364f53102a1b73e3274c6c0d4Tgclocals·f24e5ae57611d01ccf1f96d64c337e04�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�("".(*Router).Queries� � eH� %H�D$�H;Aw���H��H��$�H�D$HH�D$XH�H�$�H�|$H��H�|$PH���1��H�1�H9���H� $H�<$��H�L$XH�D$`H�D$H�L$hH�L$�H�\$HH�D$PH���^�k@@�h@H�D$@H�S H�K(H�[0H��$�H��$�H��$�H��H)�H��}IH�H�$H�T$pH�T$H�L$xH�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$xH��$�H�T$pH��H�$H�\$@H�\$�H�T$pH�L$xH��$�H�\$HH�$H�<$t{H�$ H��$�H�T$H��$�H�L$H��$�H�D$�H�\$@H�$H��$�H�\$H��$�H�\$H��$�H�\$�H�\$ H��$�H�ĠÉ%�y���������%�]���H�H�$H�H�\$H�H�\$�H�L$PH�D$���������� +*0runtime.morestack_noctxtntype."".Route�"runtime.newobject�� runtime.duffzero�Bgo.itab.*"".Router."".parentRoute�2runtime.writebarrieriface� type.[]*"".Route�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice�&"".(*Route).Queries�type.*"".Router�&type."".parentRoute� Bgo.itab.*"".Router."".parentRoute�  runtime.typ2ItabP�"".autotmp_0185_ type.[]*"".Route"".autotmp_0184/ type.[]*"".Route"".autotmp_0182�type.*"".Route"".autotmp_0181type.*"".Route"".autotmp_0180 type.[]*"".Route"".autotmp_0179�type.*"".Router"".route�type.*"".Route"".r�type.*"".Router "".~r1@type.*"".Route"".pairstype.[]string"".rtype.*"".Router"����i��"�$?]�>S5 \'Tgclocals·466dbe9b6d0b019671e1c2db1c9f0ba0Tgclocals·f24e5ae57611d01ccf1f96d64c337e04�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�("".(*Router).Schemes� � eH� %H�D$�H;Aw���H��H��$�H�D$HH�D$XH�H�$�H�|$H��H�|$PH���1��H�1�H9���H� $H�<$��H�L$XH�D$`H�D$H�L$hH�L$�H�\$HH�D$PH���^�k@@�h@H�D$@H�S H�K(H�[0H��$�H��$�H��$�H��H)�H��}IH�H�$H�T$pH�T$H�L$xH�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$xH��$�H�T$pH��H�$H�\$@H�\$�H�T$pH�L$xH��$�H�\$HH�$H�<$t{H�$ H��$�H�T$H��$�H�L$H��$�H�D$�H�\$@H�$H��$�H�\$H��$�H�\$H��$�H�\$�H�\$ H��$�H�ĠÉ%�y���������%�]���H�H�$H�H�\$H�H�\$�H�L$PH�D$���������� +*0runtime.morestack_noctxtntype."".Route�"runtime.newobject�� runtime.duffzero�Bgo.itab.*"".Router."".parentRoute�2runtime.writebarrieriface� type.[]*"".Route�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice�&"".(*Route).Schemes�type.*"".Router�&type."".parentRoute� Bgo.itab.*"".Router."".parentRoute�  runtime.typ2ItabP�"".autotmp_0201_ type.[]*"".Route"".autotmp_0200/ type.[]*"".Route"".autotmp_0198�type.*"".Route"".autotmp_0197type.*"".Route"".autotmp_0196 type.[]*"".Route"".autotmp_0195�type.*"".Router"".route�type.*"".Route"".r�type.*"".Router "".~r1@type.*"".Route"".schemestype.[]string"".rtype.*"".Router"����i��"�$?]�>S5 \'Tgclocals·466dbe9b6d0b019671e1c2db1c9f0ba0Tgclocals·f24e5ae57611d01ccf1f96d64c337e04�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�4"".(*Router).BuildVarsFunc� � eH� %H�D$�H;Aw���H��H��$�H�D$PH�D$`H�H�$�H�|$H��H�|$XH��� 1��H�1�H9���H� $H�<$��H�L$`H�D$hH�D$H�L$pH�L$�H�\$PH�D$XH���g�k@@�h@H�D$@H�S H�K(H�[0H��$�H��$�H��$�H��H)�H��}LH�H�$H�T$xH�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H�T$xH��H�$H�\$@H�\$�H�T$xH��$�H��$�H�\$PH�$H�<$t{H�$ H��$�H�T$H��$�H�L$H��$�H�D$�H�L$@H��$�H�L$HH� $H�<$t$H�$hH�D$�H�\$HH��$�H�ĨÉ%�Ӊ%�y���������%�T���H�H�$H�H�\$H�H�\$�H�L$XH�D$���������� +*0runtime.morestack_noctxtntype."".Route�"runtime.newobject�� runtime.duffzero�Bgo.itab.*"".Router."".parentRoute�2runtime.writebarrieriface� type.[]*"".Route�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice�.runtime.writebarrierptr�type.*"".Router�&type."".parentRoute� Bgo.itab.*"".Router."".parentRoute�  runtime.typ2Itab0�"".autotmp_0216_ type.[]*"".Route"".autotmp_0215/ type.[]*"".Route"".autotmp_0213�type.*"".Route"".autotmp_0212 type.[]*"".Route"".autotmp_0211�type.*"".Router"".r�type.*"".Route"".route�type.*"".Route"".r�type.*"".Router "".~r1 type.*"".Route"".f*type."".BuildVarsFunc"".rtype.*"".Router"����i��"�$?]�AV,e Tgclocals·776d19cc6eced68e652f85d577f321c6Tgclocals·38f35918b64660b95e0269a6592b7ed4�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�""".(*Router).Walk��eH� %H;aw���H��PH�D$hH�D$pH�H�$�H�l$H��tSH�\$X1�1�H�$H�\$`H�\$H�l$8H�l$H�T$@H�T$H�L$HH�L$ �H�L$(H�D$0H�L$hH�D$pH��PÉE� + + 0runtime.morestack_noctxt^"type.[0]*"".Routep"runtime.newobject�""".(*Router).walk@�"".autotmp_0227/ type.[]*"".Route "".~r1 type.error"".walkFn type."".WalkFunc"".rtype.*"".Router���� �,t +7iTgclocals·e0dd5664695c71438932a711825a98a4Tgclocals·0528ab8f76149a707fd2f0025c2178a3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�""".(*Router).walk��eH� %H��$h���H;Aw���H��HDŽ$HHDŽ$PH��$ H����H�K H�C(H�k0H��$H�D$XH��$H�D$PH��$H�L$xH�\$XH�l$PH9��H�\$xH�H�X81�H9���H�h8H�]1�H9���H�h8L�EI�XH����H�D$`H�$H��$ H�\$H��$0H�\$H��$8H�\$H��$@H�\$ H��$(H���H�D$(H�L$0H��$�H��$�H�-H9�u~H�$H�L$H�-H�l$H�-H�l$��\$ ��tNH�\$xH��H�\$xH�\$XH��H�\$XH�\$XH�l$PH9������HDŽ$HHDŽ$PH���H�\$`H���H�S H�C(H�k0H��$�1�H��$�H�D$@H��$�H��H�l$@H9���H�D$pH����H�H�hH�L$HH��$�H��$�H�H�$H��$�H�T$H��$�H�l$�H�L$�\$ ��tuH� $H��$(H�\$H��$0H�\$H��$8H�\$H��$@H�\$ �H�L$(H�D$0H��$�H��H��$�tH��$HH��$PH���H�D$pH�L$HH��H��H�l$@H9�����H�H�$H�|$`H����H�oH�|$H��H�H��H�\$H�\$h�\$ ���(���H��$0H��$8H��$@H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��$�H��H�$H�\$`H�\$�H��$�H��$�H��$�H�\$hH�$H��$(H�\$H��$0H�T$H��$8H�L$H��$@H�D$ �H�L$(H�D$0H��$�H��H��$�tH��$HH��$PH���H��$8H��H��$@H9�r%H��$0H��$0H��$8H��$@����� ��N�����D�����������d���" +00runtime.morestack_noctxt� +�"".SkipRouter�"".SkipRouter�"".SkipRouter�runtime.ifaceeq� type.*"".Router� $runtime.assertI2T2� +""".(*Router).walk� type.*"".Router� $runtime.assertI2T2�  type.[]*"".Route�"runtime.growslice�.runtime.writebarrierptr�""".(*Router).walk�$runtime.panicslicep�8"".autotmp_0254type.uint64"".autotmp_0253type.uint64"".autotmp_0252type.int"".autotmp_0249type.int"".autotmp_0248type.int"".autotmp_0247� type.[]*"".Route"".autotmp_0246�type."".matcher"".autotmp_0245� type.*"".matcher"".autotmp_0244�type.int"".autotmp_0243�type.int"".autotmp_0241�type.**"".Route"".autotmp_0240�type.int"".autotmp_0239�type.int"".autotmp_0238type.int"".autotmp_0237type.error"".autotmp_0235type.error"".autotmp_0234_"type.[]"".matcher"".autotmp_0232/ type.[]*"".Route "".err�type.error"".h�type.*"".Router "".err�type.error +"".sr�type."".matcher "".err�type.error"".t�type.*"".Route "".~r2Ptype.error"".ancestors  type.[]*"".Route"".walkFn type."".WalkFunc"".rtype.*"".Router6%����������e� f�=d8\D .4 !v9OG�T8-.  (���|DZz (Tgclocals·02f53cdec99f366e42fb544f32ed9035Tgclocals·e1370d8c0370fc841121204684c0e45d�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�"".Vars��eH� %H;aw���H��@H�D$(H�H�$H�\$(H�\$�H�\$H�l$H��H��H�H�H�\$HH�$�H�D$H�L$ H��t3H�H�$H�D$0H�D$H�L$8H�L$�H�\$H�\$PH��@�H�D$PH��@� + 0runtime.morestack_noctxtL$type."".contextKeyrruntime.convT2E��github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.Get�,type.map[string]string�"runtime.assertE2T �"".autotmp_0261/$type."".contextKey +"".rv"type.interface {} "".~r1,type.map[string]string"".r,type.*net/http.Request��� ��U3 +8xTgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�"".CurrentRoute��eH� %H;aw���H��@H�D$(H�H�$H�\$(H�\$�H�\$H�l$H��H��H�H�H�\$HH�$�H�D$H�L$ H��t3H�H�$H�D$0H�D$H�L$8H�L$�H�\$H�\$PH��@�H�D$PH��@� + 0runtime.morestack_noctxtL$type."".contextKeyrruntime.convT2E��github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.Get�type.*"".Route�"runtime.assertE2T �"".autotmp_0263/$type."".contextKey +"".rv"type.interface {} "".~r1type.*"".Route"".r,type.*net/http.Request��� ��U3 +8xTgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�"".setVars��eH� %H;aw���H��0H�D$(H�H�$H�\$(H�\$�H�\$H�l$H��H��H�H�H�\$8H�$H�\$@H�\$H�\$HH�\$ �H��0� + + 0runtime.morestack_noctxtL$type."".contextKeyrruntime.convT2E��github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.Set0`"".autotmp_0265$type."".contextKey "".val"type.interface {}"".r,type.*net/http.Request`]_ ��Y  +8HTgclocals·9b807a1de79759fa48658b2ca8ff7282Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�$"".setCurrentRoute��eH� %H;aw���H��0H�D$(H�H�$H�\$(H�\$�H�\$H�l$H��H��H�H�H�\$8H�$H�\$@H�\$H�\$HH�\$ �H��0� + + 0runtime.morestack_noctxtL$type."".contextKeyrruntime.convT2E��github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.Set0`"".autotmp_0266$type."".contextKey "".val"type.interface {}"".r,type.*net/http.Request`]_ ��Y  +8HTgclocals·9b807a1de79759fa48658b2ca8ff7282Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�"".cleanPath��eH� %H;aw���H��PH�L$XH�D$`H�D$hH�D$pH��uH�H�+H�l$hH�kH�l$pH��P�H������/t.H�H�,$H��H��H�H�H�L$H�D$�H�L$ H�D$(H�L$XH� $H�D$`H�D$�H�t$`H�T$H�D$H��H��H�\$XH9���H�+���/uHH��uQH�T$0H�$H�D$8H�D$H�-L�D$L��H��H�H��H�T$0H�D$8�\$ ��tH�T$hH�D$pH��P�H�T$0H�D$8H�T$@H�$H�D$HH�D$H�H�l$H��H��H�H��H�T$ H�D$(�� �  + 0runtime.morestack_noctxt~go.string."/"�go.string."/"�*runtime.concatstring2�path.Clean�go.string."/"� runtime.eqstring�go.string."/"�*runtime.concatstring2�$runtime.panicindex�$runtime.panicindex@� +"".autotmp_0269type.string"".autotmp_0267type.string +"".np?type.string "".~r1 type.string"".ptype.string&�>�����Y�4�6.'hB ��[&Tgclocals·771157e6981a4b26b64a947269cc9ecbTgclocals·29f0050a5ee7c2b9348a75428171d7de�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�"".uniqueVars� � eH� %H��$p���H;Aw���H��HDŽ$HHDŽ$PL��$H��$ H��$(H��$E1�H��$H�D$HL��$�H�l$HI9��AL�D$`I���YI�I�xL�L$PH�L$xH��$0H��$8H��$@H��$�1�H��$�H�D$8H��$�H�l$8H9���H�L$XH����H�1H�AH�T$@H��$�H��$�H9��~H�l$xH�,$H��$�H�|$H�t$hH�t$H�D$pH�D$�L�L$PL�D$`H��$�H�T$@H�L$X�\$ ���%H�\$hH��$�H�\$pH��$�H��$�H�H�CH��$�H����H��H��H��$�H��$�H��$�H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$HH��$PH��É����H��H��H�l$8H9��9���I��I��H�l$HI9������HDŽ$HHDŽ$PH��É� ���A����� +00runtime.morestack_noctxt� runtime.eqstring�type.string�runtime.convT2E�2runtime.writebarrieriface�Zgo.string."mux: duplicated route variable %q"� fmt.Errorf��&"".autotmp_0285�"type.interface {}"".autotmp_0283�&type.[]interface {}"".autotmp_0282type.string"".autotmp_0281�type.*string"".autotmp_0280�type.int"".autotmp_0279�type.int"".autotmp_0278�type.string"".autotmp_0277�type.*string"".autotmp_0276�type.int"".autotmp_0275�type.int"".autotmp_0273�type.string"".autotmp_0272�(type.[1]interface {}"".autotmp_0271_type.[]string"".autotmp_0270/type.[]string +"".v2�type.string +"".v1�type.string "".~r2`type.error +"".s20type.[]string +"".s1type.[]string&%����P���.�=fpb� ��v�Tgclocals·90aaa11a3c4e552027084aaae119235bTgclocals·a4b09b32f70466d9a6c07b8385c51f8a�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�"".checkPairs��eH� %H�D$�H;Aw���H��H��$�HDŽ$�HDŽ$�H��H��?H��H�T$8H)�H��H�H���H��$�H�\$xH��$�H��$�H��$�H�\$PH�H�CH�\$PH����H��H��H�\$`H�T$hH�L$pH�H�$H�\$xH�\$�H�L$H�D$H�\$`H�$H�L$@H�L$H�D$HH�D$�H�H�,$H��H��H�H�H�\$`H�\$H�\$hH�\$H�\$pH�\$ �H�L$(H�D$0H�\$8H��$�H��$�H��$�H�ĐÉ�/���H��$�HDŽ$�HDŽ$�H�Đ� +*0runtime.morestack_noctxt�type.[]string�runtime.convT2E�2runtime.writebarrieriface��go.string."mux: number of parameters must be multiple of 2, got %v"�fmt.Errorf`�"".autotmp_0296�"type.interface {}"".autotmp_0294_&type.[]interface {}"".autotmp_0293type.int"".autotmp_0291/type.[]string"".autotmp_0290(type.[1]interface {}"".length�type.int "".~r2@type.error "".~r10type.int"".pairstype.[]string "����.� ��B#�2�dmTgclocals·6d3fa487f5e45db9cb9199d2a5e0e216Tgclocals·7876b70d8da64fa07ca2fd3ecc71f905�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�."".mapFromPairsToString��eH� %H;aw���H��XHDŽ$�HDŽ$�H�\$`H�$H�\$hH�\$H�\$pH�\$�H�L$H�D$ H�T$(H�T$PH��H�D$HtH�D$xH��$�H��$�H��X�H�H�$H��H��?H��H�L$0H)�H��H��H�\$�H�T$`H�L$hH�\$H�\$@1�H�l$0H9�}mH�H�$H�\$@H�\$H��H��H9�s{Hk�H�H�\$H��H�D$8H��H��H9�sUHk�H�H�l$�H�T$`H�L$hH�D$8H��H�l$0H9�|�H�\$@H�\$xHDŽ$�HDŽ$�H��X�� �  + 0runtime.morestack_noctxt�"".checkPairs�,type.map[string]string�runtime.makemap�,type.map[string]string�$runtime.mapassign1�$runtime.panicindex�$runtime.panicindex`�"".i?type.int"".m/,type.map[string]string "".errtype.error"".lengthOtype.int "".~r2@type.error "".~r10,type.map[string]string"".pairstype.[]string&�v������,�26 A Z'O�ITgclocals·ca1ebfc68aaed1d083688775167e5178Tgclocals·61e2515c69061b8fed0e66ece719f936�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�,"".mapFromPairsToRegex��eH� %H;aw���H��pHDŽ$�HDŽ$�H�\$xH�$H��$�H�\$H��$�H�\$�H�L$H�D$ H�T$(H�T$XH��H�D$Pt!HDŽ$�H��$�H��$�H��p�H�H�$H��H��?H��H�L$0H)�H��H��H�\$�H�\$H�\$@1�H�l$0H9���H��H�D$8H��H�l$xL��$�L9���Hk�H�H�$H��H��H�H��H�D$H�L$H�T$ H�T$hH��H�L$`t!HDŽ$�H��$�H��$�H��p�H�D$HH�H�$H�\$@H�\$H�\$xH�l$8L��$�L9�s\Hk�H�H�\$H�\$HH�\$�H�D$8H��H�l$0H9��"���H�\$@H��$�HDŽ$�HDŽ$�H��p�� �  + 0runtime.morestack_noctxt�"".checkPairs�go.itab.*bytes.Buffer.io.Writer�� runtime.duffzero�type.string�runtime.convT2E�2runtime.writebarrieriface�type.string�runtime.convT2E�2runtime.writebarrieriface�type.string�runtime.convT2E� 2runtime.writebarrieriface�!0go.string."%s(?P<%s>%s)"�"fmt.Fprintf�#>go.itab.*bytes.Buffer.io.Writer�$type.string�%runtime.convT2E�&2runtime.writebarrieriface�&"go.string."%s%%s"�'fmt.Fprintf�(4runtime.writebarrierstring�*type.string�+runtime.convT2E�+2runtime.writebarrieriface�+ go.string."^%s$"�,fmt.Sprintf�-regexp.Compile�..runtime.writebarrierptr�1 regexp.QuoteMeta�26bytes.(*Buffer).WriteString�2 go.string."[/]?"�36bytes.(*Buffer).WriteString�3go.string."="�4strings.SplitN�56bytes.(*Buffer).WriteString�62bytes.(*Buffer).WriteByte�66bytes.(*Buffer).WriteString�72bytes.(*Buffer).WriteByte�8"go.string.""�8regexp.Compile�;"go.string.""�;&type."".routeRegexp�;"runtime.newobject�<4runtime.writebarrierstring�>.runtime.writebarrierptr�?4runtime.writebarrierstring�@2runtime.writebarrierslice�A2runtime.writebarrierslice�E2runtime.slicebytetostring�F$runtime.panicslice�G2runtime.slicebytetostring�H$runtime.panicslice�H$runtime.panicindex�H$runtime.panicslice�H$runtime.panicindex�H$runtime.panicindex�H$type.*bytes.Buffer�Htype.io.Writer�I>go.itab.*bytes.Buffer.io.Writer�I runtime.typ2Itab�I$type.*bytes.Buffer�Jtype.io.Writer�J>go.itab.*bytes.Buffer.io.Writer�J runtime.typ2Itab�Mtype.string�Mruntime.convT2E�N2runtime.writebarrieriface�N\go.string."mux: missing name or pattern in %q"�Ofmt.Errorf�P$runtime.panicslice�P$runtime.panicindex�P$runtime.panicindex�Q$runtime.panicindex�Q$runtime.panicslice�Q$runtime.panicindex�Q$runtime.panicindex�Q$runtime.panicslice�Q$runtime.panicindex�R$runtime.panicslice�R$runtime.panicslice�S"go.string."[^.]+"`� �"".autotmp_0416� (type.*"".routeRegexp"".autotmp_0415type.uint64"".autotmp_0414type.uint64"".autotmp_0413type.uint64"".autotmp_0412type.string"".autotmp_0411type.uint64"".autotmp_0410type.uint64"".autotmp_0409type.uint64"".autotmp_0408type.string"".autotmp_0407type.uint64"".autotmp_0406type.uint64"".autotmp_0405type.uint64"".autotmp_0403"type.interface {}"".autotmp_0402*type.*[1]interface {}"".autotmp_0401&type.[]interface {}"".autotmp_0400"type.interface {}"".autotmp_0399*type.*[1]interface {}"".autotmp_0398&type.[]interface {}"".autotmp_0397type.*uint8"".autotmp_0396"type.interface {}"".autotmp_0395"type.interface {}"".autotmp_0394"type.interface {}"".autotmp_0392&type.[]interface {}"".autotmp_0391�type.*uint8"".autotmp_0390�"type.interface {}"".autotmp_0388�&type.[]interface {}"".autotmp_0387type.uint64"".autotmp_0386type.uint64"".autotmp_0385type.uint64"".autotmp_0384type.uint64"".autotmp_0383type.uint64"".autotmp_0382type.uint64"".autotmp_0381type.int"".autotmp_0380type.int"".autotmp_0378type.uint64"".autotmp_0377type.uint64"".autotmp_0376$type.*bytes.Buffer"".autotmp_0375$type.*bytes.Buffer"".autotmp_0374�$type.*bytes.Buffer"".autotmp_0373$type.*bytes.Buffer"".autotmp_0372type.int"".autotmp_0371type.int"".autotmp_0370type.uint64"".autotmp_0369type.uint64"".autotmp_0368type.int"".autotmp_0365type.uint64"".autotmp_0363�type.string"".autotmp_0361type.[]string"".autotmp_0360type.string"".autotmp_0359type.int"".autotmp_0358type.string"".autotmp_0357type.string"".autotmp_0356(type.[1]interface {}"".autotmp_0355type.string"".autotmp_0354$type.*bytes.Buffer"".autotmp_0353(type.[1]interface {}"".autotmp_0352�type.string"".autotmp_0351�type.string"".autotmp_0350type.string"".autotmp_0349$type.*bytes.Buffer"".autotmp_0348_(type.[3]interface {}"".autotmp_0346�type.string"".autotmp_0345�(type.[1]interface {}"".autotmp_0344type.int"".autotmp_0343type.[]string"".autotmp_0342type.int"".autotmp_0340type.int"".autotmp_0338type.int"".autotmp_0337type.int "".~r0�type.stringbytes.b·2� $type.*bytes.Buffer "".~r0�type.stringbytes.s·2�type.stringbytes.s·2�type.string"strings.suffix·3�type.stringstrings.s·2�type.string"".errCompile�type.error "".reg� &type.*regexp.Regexp"".queryVal�type.string "".raw�type.string"".varIdx� type.int"".patt�type.string"".name�type.string"".parts�type.[]string "".raw�type.string"".i� type.int "".err�type.error "".end� type.int"".reverse� $type.*bytes.Buffer"".pattern� $type.*bytes.Buffer"".varsR�*type.[]*regexp.Regexp"".varsN�type.[]string"".endSlash� +type.bool""".defaultPattern�type.string"".template�type.string"".errBraces�type.error"".idxs�type.[]int "".~r6@type.error "".~r50(type.*"".routeRegexp"".strictSlash&type.bool"".matchQuery$type.bool"".matchPrefix"type.bool"".matchHost type.bool "".tpltype.stringT%� �� � �� � �� � �� � �� � ��)�8=w$  �VV�� G�! ( +��H�,1:!S +$w+ +({ 4V�09��  AA�      �V��O<P(<��(����H�vw��Hq�Cg;HD� 45\4$3('1A�v:'�Tgclocals·794bc2a224f980dced8624445883a1f1Tgclocals·33531776c15af84406e52705f6739a4b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/regexp.go�."".(*routeRegexp).Match��eH� %H;aw���H��0H�L$@H�D$8�X��u^�X��tH�$H�L$��\$�\$PH��0�H�hH�,$H�yH��t#H�_8H�|$H��H�H���\$�\$PH��0É��H� $�H�L$H�D$H�\$8H�kH�,$H�L$ H�L$H�D$(H�D$��\$�\$PH��0� + 0runtime.morestack_noctxt�D"".(*routeRegexp).matchQueryString�8regexp.(*Regexp).MatchString�"".getHost�8regexp.(*Regexp).MatchString@`"".autotmp_0457type.bool"".autotmp_0456type.string"".autotmp_0455type.bool "".~r20type.bool"".match &type.*"".RouteMatch "".req,type.*net/http.Request"".r(type.*"".routeRegexp`7_`4_`J_��$  9U ?�Tgclocals·8d11a518189555fd7f3bac3cc6ad264cTgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/regexp.go�*"".(*routeRegexp).url�!�!eH� %H��$����H;Aw���H��HDŽ$�HDŽ$�HDŽ$�HDŽ$�H��$�H�k8H�H�$H�l$H�l$�H�T$H�L$ H�D$(H��$ H��$(H��$0H��$�H����H�S0H�K8H�k@H��$`1�H��$XH�L$HH��$PH��H�l$HH9��fH�L$XH���;H�H�iH�D$PH�D$@H��$�H��$�H�H�$H��$�H�\$H��$�H�T$H��$�H�l$�H�L$ �\$(H����H�H�T$pH�iH�l$x���CH��$�H��$H��$�H��$H��$H�H�CH��$H����H��H��H��$hH��$pH��$xH�H�$H��$H�\$�H�L$H�D$H��$hH�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$hH�\$H��$pH�\$H��$xH�\$ �H�L$(H�D$0HDŽ$�HDŽ$�H��$�H��$�H�ĠÉ����H��$H��$H�H�$H��$H�\$�H�\$H�l$H��H��H�H�H��$ H�l$@L��$(L9��Hk�H�H�$�H�L$XH�D$PH��H��H�l$HH9������H��$�H����H�o H�<$H��H�H�H��$ H�\$H��$(H�\$H��$0H�\$ �H�L$(H�D$0H��$�H�kH�,$H��$�H�L$H��$�H�D$��\$���9H��$�H��� H�S0H�K8H�k@H��$`1�H��$XH�L$HH��$PH��H�l$HH9���H�L$XH����H�H�iH�D$PH�D$8H��$�H��$�H�H�$H��$�H�\$H��$�H�T$H��$�H�l$�H�\$ H���`H�3H�SH��$�H���@H�KHH�CPH�kXH��$HH��$8H�l$8H��$@H9��H��H�+H�,$H��$�H�t$H��$�H�T$��\$���pH��$�H��$�H�H�$H��$�H�\$H��$�H�L$H��$�H�D$�H�\$ H���H�+H��$H�kH��$H��$�H����H�KHH�CPH�kXH��$HH��$8H�l$8H��$@H9���H��H�H����H� H�kH�L$`H��$�H�l$hH��$�H��$�1��H��$�H���IH��H��H��$hH��$pH��$xH�H�$H��$H�\$�H�L$H�D$H��$hH�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$�H�L$H�D$H��$hH��H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$hH�\$H��$pH�\$H��$xH�\$ �H�L$(H�D$0HDŽ$�HDŽ$�H��$�H��$�H�ĠÉ������g���� �����������H�L$XH�D$PH��H��H�l$HH9��x���H��$�H��$�H��$�H��$�HDŽ$�HDŽ$�H�Ġ�� ������������+���������뛉�<���� ��1����������l���B +00runtime.morestack_noctxt�&type.[]interface {}�"runtime.makeslice�,type.map[string]string�4runtime.mapaccess2_faststr�type.string�runtime.convT2E� 2runtime.writebarrieriface� Tgo.string."mux: missing route variable %q"� +fmt.Errorf� type.string� runtime.convT2E� 2runtime.writebarrieriface�fmt.Sprintf�8regexp.(*Regexp).MatchString�,type.map[string]string�4runtime.mapaccess1_faststr�8regexp.(*Regexp).MatchString�,type.map[string]string�4runtime.mapaccess1_faststr�� runtime.duffzero�type.string�runtime.convT2E�2runtime.writebarrieriface�type.string�runtime.convT2E�2runtime.writebarrieriface�ngo.string."mux: variable %q doesn't match, expected %q"�fmt.Errorf�$runtime.panicindex� $runtime.panicindex� $runtime.panicindex`�R"".autotmp_0492"type.interface {}"".autotmp_0491"type.interface {}"".autotmp_0489&type.[]interface {}"".autotmp_0488type.string"".autotmp_0487type.*string"".autotmp_0486type.int"".autotmp_0485type.int"".autotmp_0484�"type.interface {}"".autotmp_0482&type.[]interface {}"".autotmp_0480�type.string"".autotmp_0479�type.*string"".autotmp_0478�type.int"".autotmp_0477type.int"".autotmp_0476type.error"".autotmp_0475�type.string"".autotmp_0474type.string"".autotmp_0473�type.string"".autotmp_0472?(type.[2]interface {}"".autotmp_0470�type.string"".autotmp_0469type.string"".autotmp_0468type.[]string"".autotmp_0466type.string"".autotmp_0465type.string"".autotmp_0463�type.string"".autotmp_0462�(type.[1]interface {}"".autotmp_0461type.string"".autotmp_0460�type.[]string"".autotmp_0459o&type.[]interface {}"".autotmp_0458�type.int "".~r0�type.string"".v�type.string"".k�type.int +"".rv�type.string"".value�type.string"".v�type.string"".k�type.int"".urlValues�&type.[]interface {} "".~r2@type.error "".~r1 type.string"".values,type.map[string]string"".r(type.*"".routeRegexp6%����� �����P�t�UMnl �o W=n�m�'�@ Lv��vn�9 �� ��Mt , +$Tgclocals·87979038f036a055d96e4dae0820fce3Tgclocals·016ee616535ff5ac97db8c28536faf6d�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/regexp.go�:"".(*routeRegexp).getUrlQuery� +� +eH� %H�D$�H;Aw���H���H��$�HDŽ$�HDŽ$�X��u HDŽ$�HDŽ$H����H�(H�$H��H��H�H�H�H�l$H��H��H�H�H�D$ �H�L$(H�D$0H�\$8H����H�)H�l$@H�iH�l$HH��$�H�kH�,$�H�L$H��$�1��H�H�$H�L$H��$�H�\$�L�D$HH��$�1�H9��6H��$�H���KH�;H�sH�SH��$�H���'H� H�L$PH�CH�|$xH�|$`H��$�H�t$hH��$�H�T$pH�D$XL9���H� $H�D$H�l$@H�l$L�D$�H�L$h�\$ ��ttH��~nH�\$PH�$H�t$XH�t$H�5H�l$H��H�H�H�\$`H��v6H�l$ H��H��H�H��H�\$0H��$�H�\$8H��$H����� H��$�H�$�L�D$HH��$�1�H9������HDŽ$�HDŽ$H���É�����������  +*0runtime.morestack_noctxt�go.string."="�strings.SplitN�(net/url.(*URL).Query�� runtime.duffzero�&type.net/url.Values�&runtime.mapiterinit� runtime.eqstring�go.string."="�*runtime.concatstring3�$runtime.panicindex� &runtime.mapiternext� +$runtime.panicindex@�"".autotmp_0506type.[]string"".autotmp_0504�:type.map.iter[string][]string"".autotmp_0501�type.[]string"".vals�type.[]string "".key�type.string"".templateKey�type.string "".~r1 type.string "".req,type.*net/http.Request"".r(type.*"".routeRegexp2"�H�����O���2�B  _�@n) + &�?6�Z:KTgclocals·f99f470b4e8bf0bbfec1c215fb234ac7Tgclocals·d0cd7946f7c85d974217a4cbfbe17824�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/regexp.go�D"".(*routeRegexp).matchQueryString��eH� %H;aw���H��0H�\$8H�$H�\$@H�\$�H�L$H�D$H�\$8H�kH�,$H�L$ H�L$H�D$(H�D$��\$�\$HH��0� + 0runtime.morestack_noctxt\:"".(*routeRegexp).getUrlQuery�8regexp.(*Regexp).MatchString0`"".autotmp_0507type.string "".~r1 type.bool "".req,type.*net/http.Request"".r(type.*"".routeRegexp`U_p �V +-CTgclocals·d3486bc7ce1948dc22d7ad1c0be2887aTgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/regexp.go�"".braceIndices��eH� %H�D$�H;Aw���H���HDŽ$�HDŽ$�HDŽ$HDŽ$HDŽ$H�D$@H�D$HH�H�$H�D$H�D$�L��$�L��$�H�D$@H�|$L�T$ H�t$(H��$�H��$�L��$�L��$�H��$�H��$�1�L9�}0L9���I� �+@��{��H��H��uH�L$HH��L9�|�H���0L�\$pL�L$xH��$�H�H�CH��$�H����H��H��H��$�H��$�H��$�H�H�$H�\$pH�\$�H�L$H�D$H��$�H�$H�L$`H�L$H�D$hH�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0HDŽ$�HDŽ$�HDŽ$H��$H��$H���É����H��$�L��$�H��$HDŽ$HDŽ$H����@��}�|���H��H�D$@H����H��H�L$PH��H�\$XL��H��L)�H��}iH�H�$H��$�H�|$H��$�H�T$H��$�H�t$H�D$ �L��$�L��$�H�L$PH�D$@H�|$(H�T$0H�t$8I��I��H��H�l$HH�+H��H��H�l$XH�+H��$�L��$�H��$�H��$�L��$�H��$�����H�������L�\$pL�L$xH��$�H�H�CH��$�H����H��H��H��$�H��$�H��$�H�H�$H�\$pH�\$�H�L$H�D$H��$�H�$H�L$`H�L$H�D$hH�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0HDŽ$�HDŽ$�HDŽ$H��$H��$H���É����� " +*0runtime.morestack_noctxt�type.[]int�"runtime.makeslice�type.string�runtime.convT2E�2runtime.writebarrieriface�Pgo.string."mux: unbalanced braces in %q"�fmt.Errorf� type.[]int� "runtime.growslice�type.string�runtime.convT2E�2runtime.writebarrieriface�Pgo.string."mux: unbalanced braces in %q"�fmt.Errorf�$runtime.panicindexp�."".autotmp_0533"type.interface {}"".autotmp_0532*type.*[1]interface {}"".autotmp_0531&type.[]interface {}"".autotmp_0530�"type.interface {}"".autotmp_0528_&type.[]interface {}"".autotmp_0523type.[]int"".autotmp_0522type.int"".autotmp_0520type.error"".autotmp_0519type.string"".autotmp_0518(type.[1]interface {}"".autotmp_0515�type.string"".autotmp_0514�(type.[1]interface {}"".autotmp_0512type.int"".autotmp_0511�type.int"".autotmp_0510type.int"".autotmp_0509/type.[]int"".i�type.int"".idxs�type.[]int "".idx�type.int"".level�type.int "".~r2Ptype.error "".~r1 type.[]int"".stype.string4"����>������ V�^v +  +�8 +� +� *��p��pR Tgclocals·9680905063a74374258fdae79a25b518Tgclocals·36d420fa591ebaf09d3a180d960e2a08�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/regexp.go�"".varGroupName��eH� %H;aw���H��@H�D$PH�D$XH�\$HH�$�H�L$H�D$H�H�,$H��H��H�H�H�L$0H�L$H�D$8H�D$�H�\$ H�\$PH�\$(H�\$XH��@� + + 0runtime.morestack_noctxtlstrconv.Itoa�go.string."v"�*runtime.concatstring20�"".autotmp_0543type.string "".~r1type.string "".idxtype.int�p� �,d +5[Tgclocals·a73fd2a0c6f832642aa9216fd9c5e6beTgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/regexp.go�>"".(*routeRegexpGroup).setMatch�8�8eH� %H��$����H;Aw���H��H��$�H�]1�H9���H��$�H�$�H�L$H�D$H��$�H�H�kH�,$H��$XH�L$H��$`H�D$�H�T$H�L$ H�D$(H��$0H��$8H��H��$(�KH��$�H�H�[H���Q H�S`H�KhH�kpH��$hH��$pH��$x1�H��$�H��$�H��� + H��$�H��H��H��H��H��H��tH��H��$�1�H��$xH�T$xH��$pH��$�H�l$xH9���H��$�H���� H� H�kH�t$pH�t$`H��$XH��$H��$`H��$ H���*H�|$@H�<$�H�|$@H�t$pH��$ H�T$H��$XH�D$H��$`H9���H��$H�,$H�L$H�T$H�D$�H�|$@H�t$p�\$ ����H�H�$H��$�H�kH�l$H��$�H�+H���� H�M0H�E8H�]@H��$PH��H��$@H��H��$HH9��r Hk�H�H�\$H�\$`H��H��$(L��$0L9��> Hk�H�H�l$�H�t$pH�|$@H��H��$�H��H��$�H��H�l$xH9��d���H��$�H�]1�H9�� H��$�H�[H�kH�,$H��$�H�~H���� +H�w8H�|$H�H��H�T$H�L$ H�D$(H��$H��$ H��H��$��H��$�H�[H�[H���O +H�S`H�KhH�kpH��$�H��$�H��$�1�H��$�H��$�H��� +H��$�H��H��H��H��H��H��tH��H��$P1�H��$HH��$�H��$@H��$�H��$�H9���H��$�H���� H� H�kH�|$pH�|$PH��$XH��$8H��$`H��$@H���+H�t$8H�4$�H�|$pH�t$8H��$@H�T$H��$XH�D$H��$`H9���H��$8H�,$H�L$H�T$H�D$�H�|$pH�t$8�\$ ����H�H�$H��$�H�kH�l$H��$�H�kH����H�M0H�E8H�]@H��$�H��H��$pH��H��$xH9��iHk�H�H�\$H�\$PH��H��$L��$L9��5Hk�H�H�l$�H�|$pH�t$8H��H��$�H��H��$�H��H��$�H9��`���H��$�H�m�]���/H��$�H�kH����L�E8L��$�H�M@H�H�;H��$�H�CH��$�H��$H9��kH��H)�H��H9��`L��H)�H��t H��H�H��H��$XH��$`H9��+H� $H�t$H�|$H�D$��\$ ���H���D$7H��$�H�kH����L�EL��$�H�MH�H�;H��$H�CH��$�H��$H9���H��H)�H��H9���L��H)�H��t H��H�H��H��$XH��$`H9��OH� $H�t$H�|$H�D$��\$ ���)H���\$78���H��$�H�kH�,$�H�L$H�D$H��$XH� $H��$`H�D$�H�D$�|$7�/H�H@H��H�P@H9��H�P8H��$XH�P8H��$`H�H@H�$�H�L$H�D$H��$XH��$�H��$`H��$�H�D$h-HDŽ$�HDŽ$�H�H�$�H�D$H��$�H�$H�<$�lH��$�H�\$H��$�H�\$�H��$�H�l$hH�hH��$�H�1�H9���H��$�H��$HH��$PH��$�H�$H�<$��H�$H��$�H�D$H��$�H�L$�H��$�H���xH�SH�CH�k H��$h1�H��$`H�D$pH��$XH��H�l$pH9�� +H��$�H�(H��$�H��$�H�,$H��$�H�\$�H�L$H�D$H��$�H�kH�,$H��$XH�L$H��$`H�D$�H�T$H�L$ H�D$(H��$H��$H��H��$��JH��$�H�[H����H�S`H�KhH�kpH��$�H��$�H��$�1�H��$�H��$�H���=H��$�H��H��H��H��H��H��tH��H��$P1�H��$HH��$�H��$@H��$�H��$�H9���H��$�H����H� H�kH�|$xH�|$XH��$XH��$(H��$`H��$0H���#H�t$HH�4$�H�|$xH�t$HH��$0H�T$H��$XH�D$H��$`H9���H��$(H�,$H�L$H�T$H�D$�H�|$xH�t$H�\$ ����H�H�$H��$�H�kH�l$H��$�H����H�K0H�C8H�k@H��$�H��H��$pH��H��$xH9���Hk�H�H�\$H�\$XH��H��$�L��$L9�ssHk�H�H�l$�H�|$xH�t$HH��H��$�H��H��$�H��H��$�H9��h���H��$�H��$�H��H��H�l$pH9������H�Ĉ�� � �������1���� ��u���������%�?���H�H�$H�H�\$H�H�\$�H�D$������%����� H��$�H����H�H8H�x@H��$XH� $H��$`H�|$H�H�|$H��H�H��H�\$ H�l$H��H��H�H�H��$�H�$H�<$tH�$8�H��$��x����%����y���1������� �E����1������� �E�@���� � �E�U�����f���� �������F���� � �E�L�����^���� �����d +00runtime.morestack_noctxt�"".getHost�Fregexp.(*Regexp).FindStringSubmatch�"".varGroupName� runtime.eqstring�,type.map[string]string� $runtime.mapassign1� Fregexp.(*Regexp).FindStringSubmatch�"".varGroupName� runtime.eqstring�,type.map[string]string�$runtime.mapassign1�go.string."/"� runtime.eqstring�go.string."/"� runtime.eqstring�*net/url.(*URL).String�net/url.Parse� *net/url.(*URL).String�!:type.net/http.redirectHandler�!"runtime.newobject�"4runtime.writebarrierstring�#dgo.itab.*net/http.redirectHandler.net/http.Handler�$2runtime.writebarrieriface�&:"".(*routeRegexp).getUrlQuery�'Fregexp.(*Regexp).FindStringSubmatch�,"".varGroupName�- runtime.eqstring�.,type.map[string]string�0$runtime.mapassign1�2$runtime.panicindex�2$runtime.panicindex�2$runtime.panicslice�3���(�� B 0� av:���( %  J�   ^H�U��U�� ;~ �U�` M +Z �Tgclocals·4fc7bdfae1004e17f25f57b4d9db22c7Tgclocals·d36b9b8893a5ece0bc46499614098043�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/regexp.go�"".getHost��eH� %H;aw���H��8H�L$@H�D$HH�D$PH�iH�]H����1�<t&H�iH��tH�](H�\$HH�]0H�\$PH��8ÉE��H��H��tfH�IxH���H�L$(H� $H�|$0H�|$H�5H�|$H�H��H�T$(H�L$0H�D$ H���tH9�rH��H�T$HH�L$PH��8�� ��H���W��� + + 0runtime.morestack_noctxt�go.string.":"�strings.Index�$runtime.panicslice0p"".hosttype.string "".~r1type.string"".r,type.*net/http.RequestpOoplop�,�1&=�1Tgclocals·d7e8a62d22b1cde6d92b17a55c33fe8fTgclocals·8d600a433c6aaa81a4fe446d95c5546b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/regexp.go�""".(*Route).Match� � eH� %H;aw���H��pH��$�H�l$x�]A���&H�\$xH�{X�H�\$xH���H�S H�C(H�k0H�l$h1�H�D$`H�D$ H�T$XH��H�l$ H9���H�D$0H����H�H�hH�L$(H�T$HH�l$PH��$�H�\$H�t$H�l$@H�,$H�T$8H�Z ��H��$��\$��u Ƅ$�H��p�H�D$0H�L$(H��H��H�l$ H9��w���H�1�H9�u&H�4$H�<$�H�\$xH�\$�H��$�H�~u@H�4$H�<$��H�$H�|$xH����H�oH�|$H��H�H��H��$�H�^1�H9�uHH�H�$H�D$�H�D$H��$�H�$H�<$tbH�$H�D$�H��$�H�l$xH�]81�H9�t.H�\$xH�k8H�,$H��$�H�\$H�t$H�\$xH�\$�Ƅ$�H��pÉ%땉�5����%�����%�������B���������Ƅ$�H��p� + 0runtime.morestack_noctxt� +�.runtime.writebarrierptr�2runtime.writebarrieriface�,type.map[string]string�runtime.makemap�.runtime.writebarrierptr�>"".(*routeRegexpGroup).setMatch@�"".autotmp_0644Otype."".matcher"".autotmp_0643 type.*"".matcher"".autotmp_0642�type.int"".autotmp_0641�type.int"".autotmp_0638/"type.[]"".matcher"".motype."".matcher "".~r20type.bool"".match &type.*"".RouteMatch "".req,type.*net/http.Request"".rtype.*"".Route.�������B� +�RR""g8  +&@ H.  �c�Tgclocals·d69c4140875de858f5dc9e2e8acb0bc0Tgclocals·4398bb51467914f29637b614067b995f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�("".(*Route).GetError`DH�\$H��tH�kXH�l$H�k`H�l$É��0 "".~r0type.error"".rtype.*"".Route00�0Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�*"".(*Route).BuildOnly@,H�D$H��@�hAH�D$�  "".~r0type.*"".Route"".rtype.*"".Route  � Tgclocals·06cab038d51064a089bda21fa03e00f7Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�&"".(*Route).Handler��eH� %H;aw���H��H�D$ H�xXu.H�$H�<$t-H�$H�\$(H�\$H�\$0H�\$�H�D$ H�D$8H��É%�� + 0runtime.morestack_noctxt�2runtime.writebarrieriface@0 "".~r10type.*"".Route"".handler*type.net/http.Handler"".rtype.*"".Route0C/0p�. + +J&Tgclocals·433981679ca6b8ba029d40d9f4c7048cTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�."".(*Route).HandlerFunc��eH� %H;aw���H��PH�t$XH�t$ H�\$`H�\$(H�1�H9�tWH�T$(H�D$@H�T$HH�~Xu.H�4$H�<$t-H�$H�D$0H�D$H�T$8H�T$�H�t$ H�t$hH��PÉ%��H�H�$H�H�\$H�H�\$�H�t$ H�D$�r��� + 0runtime.morestack_noctxtbZgo.itab.net/http.HandlerFunc.net/http.Handler�2runtime.writebarrieriface�2type.net/http.HandlerFunc�*type.net/http.Handler�Zgo.itab.net/http.HandlerFunc.net/http.Handler� runtime.typ2Itab0�"".autotmp_0646*type.net/http.Handler"".autotmp_0645O2type.net/http.HandlerFunc"".handler?*type.net/http.Handler"".r_type.*"".Route "".~r1 type.*"".Route"".fjtype.func(net/http.ResponseWriter, *net/http.Request)"".rtype.*"".Route�o��F���v@Tgclocals·3cd76c4f8d01c613585e17871258aa07Tgclocals·deb2efaee408e42e52d8a1e8a4a979e1�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�,"".(*Route).GetHandler`DH�\$H��tH�kH�l$H�kH�l$É��0 "".~r0*type.net/http.Handler"".rtype.*"".Route00�0Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go� "".(*Route).Name� +� +eH� %H�D$�H;Aw���H��H��$�H�]PH���zH��$�H�\$`H��$�H�\$hH��$�1��H��$�H���H��H��H�\$pH�T$xH��$�H�H�$H�l$H�|$��H�D$H�H�L$H�D$H�\$pH�$H�L$@H�L$H�D$HH�D$�H�H�$H�\$`H�\$�H�L$H�D$H�\$pH��H�$H�L$@H�L$H�D$HH�D$�H�H�,$H��H��H�H�H�\$pH�\$H�\$xH�\$H��$�H�\$ �H�L$(H�D$0H��$�H�$H�<$��H�$XH�L$PH�L$H�D$XH�D$�H��$�H�}X��H�,$H�<$��H�$HH��$�H�\$H��$�H�\$�H��$�H�$�H�D$H��$�H�\$`H��$�H�\$hH��$�H�\$8H�H�$H�D$H�\$`H�\$H�\$8H�\$�H��$�H��$�H�ĨÉ%�R����%�����%���������� +*0runtime.morestack_noctxt�� runtime.duffzero�type.string�runtime.convT2E�2runtime.writebarrieriface�type.string�runtime.convT2E�2runtime.writebarrieriface�pgo.string."mux: route already has name %q, can't set %q"�fmt.Errorf�2runtime.writebarrieriface�4runtime.writebarrierstring�4"".(*Route).getNamedRoutes�2type.map[string]*"".Route� $runtime.mapassign1@�"".autotmp_0658"type.interface {}"".autotmp_0657�"type.interface {}"".autotmp_0655o&type.[]interface {}"".autotmp_0654�type.*"".Route"".autotmp_0653type.string"".autotmp_0651�type.error"".autotmp_0650�type.string"".autotmp_0649?(type.[2]interface {} "".~r10type.*"".Route"".nametype.string"".rtype.*"".Route"����2�*�*� 3n &���fOTgclocals·4205cab2470caaf976442750814b93e4Tgclocals·609fbbd38973bf0432058b6e5d6645e9�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�&"".(*Route).GetName`DH�\$H��tH�kHH�l$H�kPH�l$É��0 "".~r0type.string"".rtype.*"".Route00�0Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�,"".(*Route).addMatcher��eH� %H;aw���H��pH�D$xH�xX��H�P H�H(H�X0H�T$XH�L$`H�\$hH��H)�H��}FH�H�$H�T$@H�T$H�L$HH�L$H�D$PH�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$HH�D$PH��H�T$@H��Hk�H�H�$H��$�H�\$H��$�H�\$�H�T$@H�L$HH�D$PH�\$xH�$H�<$t:H�$ H�T$XH�T$H�L$`H�L$H�D$hH�D$�H�D$xH��$�H��pÉ%� + 0runtime.morestack_noctxt�"type.[]"".matcher�"runtime.growslice�2runtime.writebarrieriface�2runtime.writebarrierslice@� "".autotmp_0664_"type.[]"".matcher"".autotmp_0663/"type.[]"".matcher"".autotmp_0662"type.[]"".matcher "".~r10type.*"".Route"".mtype."".matcher"".rtype.*"".Route���� �� �  �TG"Tgclocals·a69e79957b5150531998200513ab99eeTgclocals·23c4785fa8abd7e258acfe91c9f325f3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�8"".(*Route).addRegexpMatcher�%�%eH� %H��$(���H;Aw���H��XH��$`HDŽ$�HDŽ$�H�xXt H�hXH��$�H�h`H��$�H��X�H�$�H�D$H��$`H�$H�<$��H�$8H�D$�H��$`H��$hH��$p��$x����$z��H���;H���L���/�%H�n8H�]1�H9���H�v8H�~H����H�7H�<$H�H�H�H�l$H��H��H�H��H�L$ H�D$(H��$�H� $H��$�H�D$H��$hH�\$H��$pH�\$�H��$`H�L$ H�D$(H��$hH� $H��$pH�D$��$x�\$��$y�\$��$z�\$�n@@�l$�L��$`H�t$H�t$PH�D$ H�L$(H��$�H��H��$�tH��$�H��$�H��X�I�h8H����H�UH�EH�] H��$P1�H��$HH�D$@H��$@H��H�l$@H9���H�D$hH�H�L$HH���zH�n0H�$H��H��H�H�H�H���UH�h0H�\$H��H��H�H�H��L��$`H�t$PH�L$0H�D$8H��$�H��H��$�tH��$�H��$�H��X�H�D$hH�L$HH��H��H�l$@H9��M�����$x��I�h8H�]1�H9���H����H�n0H�$H��H��H�H�H�I�p8H�~H���]H�w0H�|$H�H�H��L��$`H�t$PH�D$0H�L$8H��$�H��H��$�tH��$�H��$�H��X�I�h8H�,$H�<$��H�t$�H��$`H�T$XH�\$PH�\$`H�1�H9��{H�L$`H��$�H�D$pH��$�H�L$xH�zX�(H��H�R H�K(H�[0H��$�H��$H��$H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H��$�H��Hk�H�H�$H�\$pH�\$H�\$xH�\$�H��$�H��$�H��$�H�\$XH�$H�<$tVH�$ H��$�H�T$H��$H�L$H��$H�D$�H�T$XHDŽ$�HDŽ$�H��XÉ%�H�H�$H�H�\$H�H�\$�H�T$XH�D$�N����%�����������o���I�h8H�]1�H9���H����H�n0H�$H��H��H�H�H�I�p8H�>H����H�w0H�|$H�H�H��L��$`H�t$PH�D$0H�L$8H��$�H��H��$�tH��$�H��$�H��XÀ�$z�8I�h8H���"H�UH�MH�] H��$@H��$HH��$PH��H)�H��}TH�H�$H��$H�T$H��$H�L$H��$ H�D$H�D$ �H�t$PH�T$(H�L$0H�D$8H��H��H��$H��$ H��$H��H�$H�t$�H��$H��$H��$ H��$`H�k8H�,$H�<$t6H�$H��$@H�T$H��$HH�L$H��$PH�D$��;����%���E�����I�h8H�,$H�<$tH�$H�t$������%���.�������������������E�$��������H��$�H��$�H��$�H�H�CH��$�H����H��H��H��$(H��$0H��$8H�H�$H��$�H�\$�H�L$H�D$H��$(H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$(H�\$H��$0H�\$H��$8H�\$ �H�L$(H�D$0H��$�H��$�H��XÉ����� �%�J���@ +00runtime.morestack_noctxt�4"".(*Route).getRegexpGroup�.runtime.writebarrierptr�go.string."/"�"strings.TrimRight�*runtime.concatstring2�""".newRouteRegexp� +"".uniqueVars�"".uniqueVars�.runtime.writebarrierptr�Dgo.itab.*"".routeRegexp."".matcher�"type.[]"".matcher�"runtime.growslice�2runtime.writebarrieriface�2runtime.writebarrierslice�(type.*"".routeRegexp�type."".matcher�Dgo.itab.*"".routeRegexp."".matcher� runtime.typ2Itab�"".uniqueVars�,type.[]*"".routeRegexp�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice�.runtime.writebarrierptr�!type.string�"runtime.convT2E�#2runtime.writebarrieriface�#jgo.string."mux: path must start with a slash, got %q"�$fmt.Errorf�$$runtime.panicindex`�L"".autotmp_0705type.uint64"".autotmp_0704type.uint64"".autotmp_0703type.int"".autotmp_0702type.int"".autotmp_0701�"type.[]"".matcher"".autotmp_0700�"type.[]"".matcher"".autotmp_0698�type."".matcher"".autotmp_0695type.int"".autotmp_0694type.int"".autotmp_0693�,type.[]*"".routeRegexp"".autotmp_0692,type.[]*"".routeRegexp"".autotmp_0691�(type.*"".routeRegexp"".autotmp_0690�*type.**"".routeRegexp"".autotmp_0689type.int"".autotmp_0688type.int"".autotmp_0687�"type.interface {}"".autotmp_0685_&type.[]interface {}"".autotmp_0684"type.[]"".matcher"".autotmp_0683(type.*"".routeRegexp"".autotmp_0682,type.[]*"".routeRegexp"".autotmp_0681type.error"".autotmp_0680type.error"".autotmp_0679type.error"".autotmp_0678/,type.[]*"".routeRegexp"".autotmp_0677�type.string"".autotmp_0675�type.string"".autotmp_0674�(type.[1]interface {}"".autotmp_0673�type.int"".m�type."".matcher"".r�type.*"".Route "".err�type.error +"".rr�(type.*"".routeRegexp "".~r4@type.error"".matchQuery4type.bool"".matchPrefix2type.bool"".matchHost0type.bool "".tpltype.string"".rtype.*"".Routel%�F��������������������#���E L �oWo +q� @  p� +   �Dp���WYV�`�v.Tgclocals·a579fb2cc990573d92ac647761c8f48eTgclocals·49f894b96db65eb6fa4537f9009e3618�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�,"".headerMatcher.Match��eH� %H;aw���H�� H�\$(H�$H�\$0H�k8H�l$�D$��\$�\$@H�� � + 0runtime.morestack_noctxtn*"".matchMapWithString@@ "".~r20type.bool"".match &type.*"".RouteMatch"".r,type.*net/http.Request"".m*type."".headerMatcher@.?P �6 +6Tgclocals·8d11a518189555fd7f3bac3cc6ad264cTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�&"".(*Route).Headers� +� +eH� %H�D$�H;Aw���H��H��$�H�xX�HH��$�H�$H��$�H�\$H��$�H�\$�H�\$H�\$HH�L$ H�D$(H��$�H�$H�<$��H�$XH�L$xH�L$H��$�H�D$�H��$�H�T$@H�\$HH�\$PH�1�H9��eH�L$PH�D$hH�D$XH�L$pH�L$`H�zX�(H��H�R H�K(H�[0H��$�H��$�H��$�H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H��$�H��Hk�H�H�$H�\$XH�\$H�\$`H�\$�H��$�H��$�H��$�H�\$@H�$H�<$tFH�$ H��$�H�T$H��$�H�L$H��$�H�D$�H�T$@H��$�H�ĸÉ%�H�H�$H�H�\$H�H�\$�H�T$@H�D$�d����%����H��$�H�ĸ� +*0runtime.morestack_noctxt�."".mapFromPairsToString�2runtime.writebarrieriface�Fgo.itab."".headerMatcher."".matcher�"type.[]"".matcher�"runtime.growslice�2runtime.writebarrieriface�2runtime.writebarrierslice�*type."".headerMatcher� type."".matcher� Fgo.itab."".headerMatcher."".matcher�  runtime.typ2ItabP�"".autotmp_0723_"type.[]"".matcher"".autotmp_0722/"type.[]"".matcher"".autotmp_0720�type."".matcher"".autotmp_0719type.error"".autotmp_0718"type.[]"".matcher"".autotmp_0717�*type."".headerMatcher"".m�type."".matcher"".r�type.*"".Route"".headers�,type.map[string]string "".~r1@type.*"".Route"".pairstype.[]string"".rtype.*"".Route "����[���* w�  [L�WYF3Tgclocals·61dac2719f307a892a4a15123f2e6a2dTgclocals·d0bf09074369fc3c2a2033c0e9216dd2�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�6"".headerRegexMatcher.Match��eH� %H;aw���H�� H�\$(H�$H�\$0H�k8H�l$�D$��\$�\$@H�� � + 0runtime.morestack_noctxtn("".matchMapWithRegex@@ "".~r20type.bool"".match &type.*"".RouteMatch"".r,type.*net/http.Request"".m4type."".headerRegexMatcher@.?P �6 +6Tgclocals·8d11a518189555fd7f3bac3cc6ad264cTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�2"".(*Route).HeadersRegexp� +� +eH� %H�D$�H;Aw���H��H��$�H�xX�HH��$�H�$H��$�H�\$H��$�H�\$�H�\$H�\$HH�L$ H�D$(H��$�H�$H�<$��H�$XH�L$xH�L$H��$�H�D$�H��$�H�T$@H�\$HH�\$PH�1�H9��eH�L$PH�D$hH�D$XH�L$pH�L$`H�zX�(H��H�R H�K(H�[0H��$�H��$�H��$�H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H��H��$�H��Hk�H�H�$H�\$XH�\$H�\$`H�\$�H��$�H��$�H��$�H�\$@H�$H�<$tFH�$ H��$�H�T$H��$�H�L$H��$�H�D$�H�T$@H��$�H�ĸÉ%�H�H�$H�H�\$H�H�\$�H�T$@H�D$�d����%����H��$�H�ĸ� +*0runtime.morestack_noctxt�,"".mapFromPairsToRegex�2runtime.writebarrieriface�Pgo.itab."".headerRegexMatcher."".matcher�"type.[]"".matcher�"runtime.growslice�2runtime.writebarrieriface�2runtime.writebarrierslice�4type."".headerRegexMatcher� type."".matcher� Pgo.itab."".headerRegexMatcher."".matcher�  runtime.typ2ItabP�"".autotmp_0739_"type.[]"".matcher"".autotmp_0738/"type.[]"".matcher"".autotmp_0736�type."".matcher"".autotmp_0735type.error"".autotmp_0734"type.[]"".matcher"".autotmp_0733�4type."".headerRegexMatcher"".m�type."".matcher"".r�type.*"".Route"".headers��B,"G $C $q-� ��a{2�Tgclocals·59dbf976b94cece68fb6f0f44435318fTgclocals·92e9b440ac71050dc24736a398ce1eac�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�&"".(*Route).URLPath� � eH� %H�D$�H;Aw���H��H��$�HDŽ$�HDŽ$�H�xXt,HDŽ$�H�hXH��$�H�h`H��$�H�Ĉ�H�X81�H9���H�h8H�]1�H9��vH�$H��$�H�\$H��$�H�\$H��$�H�\$�H�L$ H�D$(H�T$0H��$�H��H�D$xt$HDŽ$�H��$�H��$�H�Ĉ�H��$�H�[8H�kH�,$H�L$�H�\$H�\$hH�\$H�\$pH�D$ H�L$(H��$�H��H�D$xt$HDŽ$�H��$�H��$�H�Ĉ�H�H�$�H�|$H��H��tk1��H�L$8H� $H�<$tKH�$8H�\$hH�\$H�\$pH�\$�H�\$8H��$�HDŽ$�HDŽ$�H�ĈÉ%묉�H�H�+H�l$XH�kH�l$`H�D$HH�D$PH�H�$�H�D$H�D$@H�$H�<$��H�\$XH�\$H�\$`H�\$�H�\$@H�\$@H�1�H9�t)H�L$@HDŽ$�H��$�H��$�H�Ĉ�H�H�$H�H�\$H�H�\$�H�D$먉%�k���" +*0runtime.morestack_noctxt�."".(*Route).prepareVars�*"".(*routeRegexp).url� type.net/url.URL�"runtime.newobject�� runtime.duffzero�4runtime.writebarrierstring�Tgo.string."mux: route doesn't have a path"�.type.errors.errorString� "runtime.newobject� 4runtime.writebarrierstring� +Bgo.itab.*errors.errorString.error� +0type.*errors.errorString� type.error� Bgo.itab.*errors.errorString.error�  runtime.typ2Itabp�"".autotmp_0852�"type.*net/url.URL"".autotmp_0849�0type.*errors.errorString"".autotmp_08480type.*errors.errorString "".~r0type.errorerrors.text·2_type.string"".path?type.string "".errtype.error "".~r2Ptype.error "".~r1@"type.*net/url.URL"".pairstype.[]string"".rtype.*"".RouteL"�R�����r�������F�>�B,"G $D $S- � ��Ct2�Tgclocals·59dbf976b94cece68fb6f0f44435318fTgclocals·92e9b440ac71050dc24736a398ce1eac�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�."".(*Route).prepareVars��eH� %H;aw���H��@H�D$pH�D$xH�\$PH�$H�\$XH�\$H�\$`H�\$�H�L$H�D$ H�T$(H�T$8H��H�D$0tH�D$hH�D$pH�T$xH��@�H�\$HH�$H�L$�H�\$H�\$hH�D$pH�D$xH��@� + 0runtime.morestack_noctxt�."".mapFromPairsToString�*"".(*Route).buildVarsp� + "".errtype.error "".~r2Ptype.error "".~r1@,type.map[string]string"".pairstype.[]string"".rtype.*"".Route�j�3��,6 ; +IwTgclocals·9877a4ef732a0f966b889793f9b99b87Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�*"".(*Route).buildVars��eH� %H;aw���H��(H�L$0H�D$8H�9t*H�H�iH�D$H�l$ H�,$H�T$H�Z ��H�L$0H�D$H�Yh1�H9�tH�D$8H�$H�QhH���H�D$H�D$@H��(� + 0runtime.morestack_noctxt� +� +0P"".autotmp_0856,type.map[string]string "".~r1 ,type.map[string]string"".m,type.map[string]string"".rtype.*"".RoutePeO��$*  + +H8Tgclocals·d3486bc7ce1948dc22d7ad1c0be2887aTgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�4"".(*Route).getNamedRoutes��eH� %H;aw���H��@H�\$HH�;��H�H�$H�D$�H�\$H�\$(H�H�$�H�L$H��H����1��H�L$ H� $H�<$��H�$8H�\$(H�\$�H�D$ 1�@�hAH�D$ H�1�H9�tpH�\$HH�$H�<$tWH�L$ H�D$0H�D$H�L$8H�L$�H�\$HH��t*H�H�kH�l$8H�,$H�D$0H�X(��H�\$H�\$PH��@É�҉%�H�H�$H�H�\$H�H�\$�H�D$�^����%� ��������� + 0runtime.morestack_noctxtX2type.map[string]*"".Route|runtime.makemap�type."".Router�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�Bgo.itab.*"".Router."".parentRoute�2runtime.writebarrieriface� +�type.*"".Router�&type."".parentRoute�Bgo.itab.*"".Router."".parentRoute� runtime.typ2Itab �"".autotmp_0862?type.*"".Router"".autotmp_0861type.*"".Router"".autotmp_08602type.map[string]*"".Route"".autotmp_0859type.*"".Router"".autotmp_0858/2type.map[string]*"".Route "".~r02type.map[string]*"".Route"".rtype.*"".Route���R�� �9N=ALj"Tgclocals·31b90725c9a885e731df361f51db8f0dTgclocals·db0987207386230beda65332b07cbe03�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�4"".(*Route).getRegexpGroup� � eH� %H;aw���H��PH�l$XH�]81�H9��FH�\$XH�;��H�H�$H�D$�H�\$H�\$8H�H�$�H�L$H��H���T1��H�L$(H� $H�<$�-H�$8H�\$8H�\$�H�D$(1�@�hAH�D$(H�1�H9���H�\$XH�$H�<$��H�L$(H�D$@H�D$H�L$HH�L$�H�\$XH���qH�H�kH�l$HH�,$H�D$@H�X0��H�D$H�D$ 1�H9�uPH�H�$�H�D$H�\$XH�$H�<$t"H�$8H�D$�H�\$XH�k8H�l$`H��PÉ%��H�H�$�H�D$H�D$0H�$H�<$��H�\$ H�+H�l$�H�\$0H�$H�<$��H�$H�\$ H�kH�l$�H�\$0H�$H�<$t]H�$H�|$ H��tIH�oH�|$H��H�H�H��H�\$XH�$H�<$tH�$8H�\$0H�\$������%�މ볉%뚉%�f����%�4���������%�O���H�H�$H�H�\$H�H�\$�H�D$� ����%����������0 + 0runtime.morestack_noctxt�2type.map[string]*"".Route�runtime.makemap�type."".Router�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�Bgo.itab.*"".Router."".parentRoute�2runtime.writebarrieriface� +�0type."".routeRegexpGroup�"runtime.newobject�.runtime.writebarrierptr�0type."".routeRegexpGroup�"runtime.newobject�.runtime.writebarrierptr�.runtime.writebarrierptr�2runtime.writebarrierslice� .runtime.writebarrierptr� +type.*"".Router� +&type."".parentRoute� +Bgo.itab.*"".Router."".parentRoute� + runtime.typ2Itab �"".autotmp_08742type.*"".routeRegexpGroup"".autotmp_0872Otype.*"".Router"".autotmp_0871type.*"".Router"".autotmp_08702type.*"".routeRegexpGroup"".autotmp_0869?2type.*"".routeRegexpGroup"".autotmp_0868type.*"".Router"".autotmp_0867/2type.map[string]*"".Route"".regexp_2type.*"".routeRegexpGroup "".~r02type.*"".routeRegexpGroup"".rtype.*"".Route������H� �/ 4 �$ %T,QAT�+c$ n%Tgclocals·0d6246443c3fddb7ffb759a83afd407dTgclocals·3ed9fdb07b75789a32dd7844090b13c9�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�"".init��eH� %H;aw���H��0���t���uH��0�� ����������H�H�,$H��H��H�H��H�L$H�D$H�H�$H�L$ H�L$H�D$(H�D$��H��0�( + 0runtime.morestack_noctxt:"".initdone·R"".initdone·p"runtime.throwinit�"".initdone·�strings.init�strconv.init�net/url.init�bytes.init��github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.init�regexp.init�path.init�net/http.init�fmt.init�8go.string."skip this router"�errors.New�"".SkipRouter�2runtime.writebarrieriface�"".initdone·`"".autotmp_0878type.error`_`�_ rH� r�H� 7�Tgclocals·3280bececceccd33cb74587feedb1f9fTgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go�0"".parentRoute.buildVars��eH� %H;aw���H��H�Y H��t H�|$ H9;uH�#H�\$0H�\$H�\$(H�$H�\$ H�[ ��H�\$H�\$8H��� + 0runtime.morestack_noctxt� +@0 "".~r10,type.map[string]string""..anon0 ,type.map[string]string""..this&type."".parentRoute0B/`` +LTgclocals·eeb28990c0dc813022336c3780186218Tgclocals·3280bececceccd33cb74587feedb1f9f�:"".parentRoute.getNamedRoutes��eH� %H;aw���H��H�Y H��t H�|$H9;uH�#H�\$ H�$H�\$H�[(��H�\$H�\$(H��� + 0runtime.morestack_noctxt� +0  "".~r0 2type.map[string]*"".Route""..this&type."".parentRoute 8`` +BTgclocals·8cb639c12a4a13c6ace27031b0f83707Tgclocals·3280bececceccd33cb74587feedb1f9f�:"".parentRoute.getRegexpGroup��eH� %H;aw���H��H�Y H��t H�|$H9;uH�#H�\$ H�$H�\$H�[0��H�\$H�\$(H��� + 0runtime.morestack_noctxt� +0  "".~r0 2type.*"".routeRegexpGroup""..this&type."".parentRoute 8`` +BTgclocals·8cb639c12a4a13c6ace27031b0f83707Tgclocals·3280bececceccd33cb74587feedb1f9f�(type..hash.[8]string��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�runtime.strhash@` "".autotmp_0884type.int"".autotmp_0883type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*[8]string`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/doc.go�$type..eq.[8]string��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$`H����H��Hk�H�H�3H�KH�\$hH��tvH��Hk�H�H�H�CH9�uVH�t$HH�4$H�L$PH�L$H�T$8H�T$H�D$@H�D$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt� runtime.eqstring@�"".autotmp_0888?type.string"".autotmp_0887type.string"".autotmp_0886_type.int"".autotmp_0885Otype.int "".~r30type.bool"".s type.uintptr"".qtype.*[8]string"".ptype.*[8]string&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/doc.go� "".matcher.Match��eH� %H;aw���H�� H�Y H��t H�|$(H9;uH�#H�\$8H�\$H�\$@H�\$H�\$0H�$H�\$(H�[ ���\$�\$HH�� � + 0runtime.morestack_noctxt� +P@ "".~r2@type.bool""..anon10&type.*"".RouteMatch""..anon0 ,type.*net/http.Request""..thistype."".matcher@K? p +p +VTgclocals·564befda8e2e8cc7f35f6bc1d3c5e0a6Tgclocals·3280bececceccd33cb74587feedb1f9f�."".(*MatcherFunc).Match��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�\$81�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�\$8H�+H�,$H�\$@H�\$H�\$HH�\$��\$�\$PH��0� + 0runtime.morestack_noctxt~go.string."mux"�.go.string."MatcherFunc"�"go.string."Match"�"runtime.panicwrap�("".MatcherFunc.Match@` "".~r20type.bool"".match &type.*"".RouteMatch"".r,type.*net/http.Request""..this(type.*"".MatcherFunc`�_ +� � +}CTgclocals·8d11a518189555fd7f3bac3cc6ad264cTgclocals·3280bececceccd33cb74587feedb1f9f�4type..hash.[1]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_0893type.int"".autotmp_0892type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[1]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/doc.go�0type..eq.[1]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_0897?"type.interface {}"".autotmp_0896"type.interface {}"".autotmp_0895_type.int"".autotmp_0894Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[1]interface {}"".p*type.*[1]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/doc.go�4type..hash.[3]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_0900type.int"".autotmp_0899type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[3]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/doc.go�0type..eq.[3]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_0904?"type.interface {}"".autotmp_0903"type.interface {}"".autotmp_0902_type.int"".autotmp_0901Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[3]interface {}"".p*type.*[3]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/doc.go�4type..hash.[2]interface {}��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�(runtime.nilinterhash@` "".autotmp_0907type.int"".autotmp_0906type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".p*type.*[2]interface {}`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/doc.go�0type..eq.[2]interface {}��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$hH����H��Hk�H�H� H�sH�\$`H��tvH��Hk�H�H�H�SH9�uVH�D$8H�$H�T$@H�T$H�L$HH�L$H�t$PH�t$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt�runtime.efaceeq@�"".autotmp_0911?"type.interface {}"".autotmp_0910"type.interface {}"".autotmp_0909_type.int"".autotmp_0908Otype.int "".~r30type.bool"".s type.uintptr"".q*type.*[2]interface {}"".p*type.*[2]interface {}&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/doc.go�2"".(*headerMatcher).Match��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�\$81�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�\$8H�+H�,$H�\$@H�\$H�\$HH�\$��\$�\$PH��0� + 0runtime.morestack_noctxt~go.string."mux"�2go.string."headerMatcher"�"go.string."Match"�"runtime.panicwrap�,"".headerMatcher.Match@` "".~r20type.bool"".match &type.*"".RouteMatch"".r,type.*net/http.Request""..this,type.*"".headerMatcher`�_ +�� +}CTgclocals·8d11a518189555fd7f3bac3cc6ad264cTgclocals·3280bececceccd33cb74587feedb1f9f�<"".(*headerRegexMatcher).Match��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�\$81�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�\$8H�+H�,$H�\$@H�\$H�\$HH�\$��\$�\$PH��0� + 0runtime.morestack_noctxt~go.string."mux"��2"".(*methodMatcher).Match��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�\$81�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�t$8H��t4H�,$H��H�H�H�H�\$@H�\$H�\$HH�\$ ��\$(�\$PH��0É�� + 0runtime.morestack_noctxt~go.string."mux"�2go.string."methodMatcher"�"go.string."Match"�"runtime.panicwrap�,"".methodMatcher.Match@` "".~r20type.bool"".match &type.*"".RouteMatch"".r,type.*net/http.Request""..this,type.*"".methodMatcher`�_` �� +}STgclocals·8d11a518189555fd7f3bac3cc6ad264cTgclocals·3280bececceccd33cb74587feedb1f9f�2"".(*schemeMatcher).Match��eH� %H;aw���H��0H�Y H��t H�|$8H9;uH�#H�\$81�H9�uHH�H�,$H��H��H�H�H�H�l$H��H��H�H�H�H�l$ H��H��H�H�� H�t$8H��t4H�,$H��H�H�H�H�\$@H�\$H�\$HH�\$ ��\$(�\$PH��0É�� + 0runtime.morestack_noctxt~go.string."mux"�2go.string."schemeMatcher"�"go.string."Match"�"runtime.panicwrap�,"".schemeMatcher.Match@` "".~r20type.bool"".match &type.*"".RouteMatch"".r,type.*net/http.Request""..this,type.*"".schemeMatcher`�_` �� +}STgclocals·8d11a518189555fd7f3bac3cc6ad264cTgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·f6dcde45bff02c6c4b088b594fd52a4c((�Tgclocals·37da6a443256db8ec55c7210d030a9b0((�Tgclocals·29f0050a5ee7c2b9348a75428171d7de �Tgclocals·d69c4140875de858f5dc9e2e8acb0bc0 **�,Zgo.itab.net/http.HandlerFunc.net/http.Handler�(go.string."Location"@2Location (go.string."Location"�Tgclocals·4c1561a135d5ed5147fd4ff64ff73c94��6�� �� ���Tgclocals·7a383875e23784cb158d762414ce6278HH��������Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·64b411f0f44be3f38c26e84fc3239091�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·06cab038d51064a089bda21fa03e00f7�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·06cab038d51064a089bda21fa03e00f7�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·d3486bc7ce1948dc22d7ad1c0be2887a +�,Bgo.itab.*"".Router."".parentRoute�Tgclocals·691c0cb9316c0a5f7d8580c74ac115f2@@ (�Tgclocals·0d6246443c3fddb7ffb759a83afd407d@@�Tgclocals·8cdbdba615b2fb90357456ca3f2cb9a4PP ��""�Tgclocals·259efa0f9d5b5ab4cbb1f7201749d3e1PP ���������Tgclocals·f24e5ae57611d01ccf1f96d64c337e04HH��( + + �Tgclocals·1c7793dad628d89b0b03aa7a6b5e8ac7HH +��������Tgclocals·f24e5ae57611d01ccf1f96d64c337e04HH��( + + �Tgclocals·466dbe9b6d0b019671e1c2db1c9f0ba0HH + + + + + + + +�Tgclocals·f24e5ae57611d01ccf1f96d64c337e04HH��( + + �Tgclocals·ccff1a4364f53102a1b73e3274c6c0d4HH + + + + + + +�Tgclocals·f24e5ae57611d01ccf1f96d64c337e04HH��( + + �Tgclocals·6b6fbfacf80ed81d2be06478c8f1790dHH + + + + + + +�Tgclocals·f24e5ae57611d01ccf1f96d64c337e04HH��( + + �Tgclocals·466dbe9b6d0b019671e1c2db1c9f0ba0HH + + + + + + + +�Tgclocals·f24e5ae57611d01ccf1f96d64c337e04HH��( + + �Tgclocals·ccff1a4364f53102a1b73e3274c6c0d4HH + + + + + + +�Tgclocals·f24e5ae57611d01ccf1f96d64c337e04HH��( + + �Tgclocals·ccff1a4364f53102a1b73e3274c6c0d4HH + + + + + + +�Tgclocals·f24e5ae57611d01ccf1f96d64c337e04HH��( + + �Tgclocals·466dbe9b6d0b019671e1c2db1c9f0ba0HH + + + + + + + +�Tgclocals·f24e5ae57611d01ccf1f96d64c337e04HH��( + + �Tgclocals·466dbe9b6d0b019671e1c2db1c9f0ba0HH + + + + + + + +�Tgclocals·38f35918b64660b95e0269a6592b7ed4PP ��""��Tgclocals·776d19cc6eced68e652f85d577f321c6PP + + + + + + + +�Tgclocals·0528ab8f76149a707fd2f0025c2178a3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·e1370d8c0370fc841121204684c0e45dpp.���� ��Tgclocals·02f53cdec99f366e42fb544f32ed9035@@******�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·06cab038d51064a089bda21fa03e00f7�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·06cab038d51064a089bda21fa03e00f7�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·9b807a1de79759fa48658b2ca8ff7282>�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·9b807a1de79759fa48658b2ca8ff7282>�go.string."/"0$/ go.string."/"�Tgclocals·29f0050a5ee7c2b9348a75428171d7de �Tgclocals·771157e6981a4b26b64a947269cc9ecb �Zgo.string."mux: duplicated route variable %q"pd!mux: duplicated route variable %q Zgo.string."mux: duplicated route variable %q"�Tgclocals·a4b09b32f70466d9a6c07b8385c51f8aPP.* / �Tgclocals·90aaa11a3c4e552027084aaae119235b00������go.string."mux: number of parameters must be multiple of 2, got %v"��7mux: number of parameters must be multiple of 2, got %v �go.string."mux: number of parameters must be multiple of 2, got %v"�Tgclocals·7876b70d8da64fa07ca2fd3ecc71f905((�����Tgclocals·6d3fa487f5e45db9cb9199d2a5e0e216(( �Tgclocals·61e2515c69061b8fed0e66ece719f936 �Tgclocals·ca1ebfc68aaed1d083688775167e5178  �Tgclocals·023baa463c1418986a4dad82d1430d9900  +�Tgclocals·9fcabcff059425eb5732bdd45e48e99f00 �Tgclocals·4398bb51467914f29637b614067b995f �Tgclocals·f608478770c574ea7f894c13fa2c89c9  ���Tgclocals·7212e532adb3b2b744244d8012f4d919PP2��V���V���V�Tgclocals·6302ce6642f568c714fa473870d50e5100�Tgclocals·12a3c69de01d25238dd5def67adcdcd9PP0��U��U +��U�Tgclocals·6302ce6642f568c714fa473870d50e5100�,>go.itab.*bytes.Buffer.io.Writer�"go.string."[^/]+"0,[^/]+ "go.string."[^/]+"�$go.string."[^?&]*"0.[^?&]* $go.string."[^?&]*"�"go.string."[^.]+"0,[^.]+ "go.string."[^.]+"�go.string.":"0$: go.string.":"�\go.string."mux: missing name or pattern in %q"pf"mux: missing name or pattern in %q \go.string."mux: missing name or pattern in %q"�0go.string."%s(?P<%s>%s)"@: %s(?P<%s>%s) 0go.string."%s(?P<%s>%s)"�"go.string."%s%%s"0,%s%%s "go.string."%s%%s"� go.string."^%s$"0*^%s$ go.string."^%s$"� go.string."[/]?"0*[/]? go.string."[/]?"�go.string."="0$= go.string."="�"go.string.""0, "go.string.""�Tgclocals·33531776c15af84406e52705f6739a4b� +� +"�� � � � � � � "�� " ��� " ��� " ���� "(���� ��"���� ��" �� "(���  "���  "�� "��  "��  �"�� +�  �"�"�"�����Tgclocals·794bc2a224f980dced8624445883a1f1��" �Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·8d11a518189555fd7f3bac3cc6ad264c*�Tgo.string."mux: missing route variable %q"`^mux: missing route variable %q Tgo.string."mux: missing route variable %q"�ngo.string."mux: variable %q doesn't match, expected %q"�x+mux: variable %q doesn't match, expected %q ngo.string."mux: variable %q doesn't match, expected %q"�Tgclocals·016ee616535ff5ac97db8c28536faf6d�� +R� ��� ����Tgclocals·87979038f036a055d96e4dae0820fce3`` + + + + + + + + + + +�Tgclocals·d0cd7946f7c85d974217a4cbfbe17824``(��U"��U��U�Tgclocals·f99f470b4e8bf0bbfec1c215fb234ac788 + + + + +�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·d3486bc7ce1948dc22d7ad1c0be2887a +�Pgo.string."mux: unbalanced braces in %q"`Zmux: unbalanced braces in %q Pgo.string."mux: unbalanced braces in %q"�Tgclocals·36d420fa591ebaf09d3a180d960e2a08((  �Tgclocals·9680905063a74374258fdae79a25b518((�go.string."v"0$v go.string."v"�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·a73fd2a0c6f832642aa9216fd9c5e6be�,dgo.itab.*net/http.redirectHandler.net/http.Handler�Tgclocals·d36b9b8893a5ece0bc46499614098043�� ~����((�(�Tgclocals·4fc7bdfae1004e17f25f57b4d9db22c7hh ������������Tgclocals·8d600a433c6aaa81a4fe446d95c5546b �Tgclocals·d7e8a62d22b1cde6d92b17a55c33fe8f �Tgclocals·4398bb51467914f29637b614067b995f �Tgclocals·d69c4140875de858f5dc9e2e8acb0bc0 **�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·06cab038d51064a089bda21fa03e00f7�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·433981679ca6b8ba029d40d9f4c7048c.�Tgclocals·deb2efaee408e42e52d8a1e8a4a979e1(( +�Tgclocals·3cd76c4f8d01c613585e17871258aa07(( + + +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�pgo.string."mux: route already has name %q, can't set %q"�z,mux: route already has name %q, can't set %q pgo.string."mux: route already has name %q, can't set %q"�Tgclocals·609fbbd38973bf0432058b6e5d6645e900����Tgclocals·4205cab2470caaf976442750814b93e400 + + + +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·23c4785fa8abd7e258acfe91c9f325f3  �Tgclocals·a69e79957b5150531998200513ab99ee ..�,Dgo.itab.*"".routeRegexp."".matcher�jgo.string."mux: path must start with a slash, got %q"�t)mux: path must start with a slash, got %q jgo.string."mux: path must start with a slash, got %q"�Tgclocals·49f894b96db65eb6fa4537f9009e3618�� +B � �(  �Tgclocals·a579fb2cc990573d92ac647761c8f48e`` + JJJJJJJJJJ�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·8d11a518189555fd7f3bac3cc6ad264c*�,Fgo.itab."".headerMatcher."".matcher�Tgclocals·d0bf09074369fc3c2a2033c0e9216dd2@@"��Tgclocals·61dac2719f307a892a4a15123f2e6a2d@@ + + + + + + +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·8d11a518189555fd7f3bac3cc6ad264c*�,Pgo.itab."".headerRegexMatcher."".matcher�Tgclocals·d0bf09074369fc3c2a2033c0e9216dd2@@"��Tgclocals·61dac2719f307a892a4a15123f2e6a2d@@ + + + + + + +�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·8d11a518189555fd7f3bac3cc6ad264c*�,Bgo.itab."".MatcherFunc."".matcher�Tgclocals·75f60c0a81a1e5f06874fff822af42bb88 +� �Tgclocals·ab01a2d55089ff50c402006df1039c3988 + + + + +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·7f9f3eadddff48005164209ea06f81db ��,Fgo.itab."".methodMatcher."".matcher�Tgclocals·ea547fb5b4f79d347b1811ec26dee271pp,  �Tgclocals·61dac2719f307a892a4a15123f2e6a2d@@ + + + + + + +�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·e0dd5664695c71438932a711825a98a4 +�Tgclocals·c3929f74b6da126d84d869df0f05d5b9((/�Tgclocals·149f5bf45741ad4d84849674a456615e(( + + + +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·7f9f3eadddff48005164209ea06f81db ��,Fgo.itab."".schemeMatcher."".matcher�Tgclocals·ea547fb5b4f79d347b1811ec26dee271pp,  �Tgclocals·61dac2719f307a892a4a15123f2e6a2d@@ + + + + + + +�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·d3486bc7ce1948dc22d7ad1c0be2887a +�,@go.itab.*"".Route."".parentRoute�,:go.itab.*"".Router."".matcher�Tgclocals·b285e033b8bcfe9b9c1b7a6b8ade4839PP �� * + + +�Tgclocals·d2a701546bccde17a508ee8f261130f1PP�,Bgo.itab.*errors.errorString.error�dgo.string."mux: route doesn't have a host or path"pn&mux: route doesn't have a host or path dgo.string."mux: route doesn't have a host or path"� go.string."http"0*http go.string."http"�Tgclocals·15a9e9c4f9817efd4475d45f753600deXX  �������Tgclocals·918b03c3cf4f7263dd73363217e9a538XX  + + + + + + + + +�Tgo.string."mux: route doesn't have a host"`^mux: route doesn't have a host Tgo.string."mux: route doesn't have a host"�Tgclocals·92e9b440ac71050dc24736a398ce1eac88 �Tgclocals·59dbf976b94cece68fb6f0f44435318f88 + + + + +�Tgo.string."mux: route doesn't have a path"`^mux: route doesn't have a path Tgo.string."mux: route doesn't have a path"�Tgclocals·92e9b440ac71050dc24736a398ce1eac88 �Tgclocals·59dbf976b94cece68fb6f0f44435318f88 + + + + +�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·9877a4ef732a0f966b889793f9b99b87 +�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·d3486bc7ce1948dc22d7ad1c0be2887a +�Tgclocals·db0987207386230beda65332b07cbe03((�Tgclocals·31b90725c9a885e731df361f51db8f0d((�Tgclocals·3ed9fdb07b75789a32dd7844090b13c9@@ �" �Tgclocals·0d6246443c3fddb7ffb759a83afd407d@@�8go.string."skip this router"PBskip this router 8go.string."skip this router"�Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·3280bececceccd33cb74587feedb1f9f�*"".SkipRouter type.error�,"".initdone·type.uint8�"".NewRouter·f"".NewRouter�$runtime.makemap·fruntime.makemap�(runtime.newobject·f"runtime.newobject�4runtime.writebarrierptr·f.runtime.writebarrierptr�,runtime.throwreturn·f&runtime.throwreturn�*"".(*Router).Match·f$"".(*Router).Match�("".(*Route).Match·f""".(*Route).Match�2"".(*Router).ServeHTTP·f,"".(*Router).ServeHTTP�"".cleanPath·f"".cleanPath�&runtime.eqstring·f runtime.eqstring�0net/url.(*URL).String·f*net/url.(*URL).String�,net/http.Header.Set·f&net/http.Header.Set�,runtime.deferreturn·f&runtime.deferreturn�"".setVars·f"".setVars�*"".setCurrentRoute·f$"".setCurrentRoute�(net/http.NotFound·f"net/http.NotFound�&runtime.typ2Itab·f runtime.typ2Itab��github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.Clear·f�github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.Clear�(runtime.deferproc·f"runtime.deferproc�&"".(*Router).Get·f "".(*Router).Get�<"".(*Router).getNamedRoutes·f6"".(*Router).getNamedRoutes�:runtime.mapaccess1_faststr·f4runtime.mapaccess1_faststr�0"".(*Router).GetRoute·f*"".(*Router).GetRoute�6"".(*Router).StrictSlash·f0"".(*Router).StrictSlash�<"".(*Router).getRegexpGroup·f6"".(*Router).getRegexpGroup�2"".(*Router).buildVars·f,"".(*Router).buildVars�0"".(*Router).NewRoute·f*"".(*Router).NewRoute�8runtime.writebarrieriface·f2runtime.writebarrieriface�(runtime.growslice·f"runtime.growslice�8runtime.writebarrierslice·f2runtime.writebarrierslice�,"".(*Router).Handle·f&"".(*Router).Handle�&"".(*Route).Path·f "".(*Route).Path�4"".(*Router).HandleFunc·f."".(*Router).HandleFunc�4"".(*Route).HandlerFunc·f."".(*Route).HandlerFunc�."".(*Router).Headers·f("".(*Router).Headers�,"".(*Route).Headers·f&"".(*Route).Headers�("".(*Router).Host·f""".(*Router).Host�&"".(*Route).Host·f "".(*Route).Host�6"".(*Router).MatcherFunc·f0"".(*Router).MatcherFunc�4"".(*Route).MatcherFunc·f."".(*Route).MatcherFunc�."".(*Router).Methods·f("".(*Router).Methods�,"".(*Route).Methods·f&"".(*Route).Methods�("".(*Router).Path·f""".(*Router).Path�4"".(*Router).PathPrefix·f."".(*Router).PathPrefix�2"".(*Route).PathPrefix·f,"".(*Route).PathPrefix�."".(*Router).Queries·f("".(*Router).Queries�,"".(*Route).Queries·f&"".(*Route).Queries�."".(*Router).Schemes·f("".(*Router).Schemes�,"".(*Route).Schemes·f&"".(*Route).Schemes�:"".(*Router).BuildVarsFunc·f4"".(*Router).BuildVarsFunc�("".(*Router).Walk·f""".(*Router).Walk�("".(*Router).walk·f""".(*Router).walk�$runtime.ifaceeq·fruntime.ifaceeq�*runtime.assertI2T2·f$runtime.assertI2T2�*runtime.panicslice·f$runtime.panicslice�"".Vars·f"".Vars�$runtime.convT2E·fruntime.convT2E��github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.Get·f�github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.Get�(runtime.assertE2T·f"runtime.assertE2T�$"".CurrentRoute·f"".CurrentRoute��github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.Set·f�github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.Set�*runtime.panicindex·f$runtime.panicindex�0runtime.concatstring2·f*runtime.concatstring2�path.Clean·fpath.Clean� "".uniqueVars·f"".uniqueVars�fmt.Errorf·ffmt.Errorf� "".checkPairs·f"".checkPairs�4"".mapFromPairsToString·f."".mapFromPairsToString�*runtime.mapassign1·f$runtime.mapassign1�2"".mapFromPairsToRegex·f,"".mapFromPairsToRegex�"regexp.Compile·fregexp.Compile�$"".matchInArray·f"".matchInArray�0"".matchMapWithString·f*"".matchMapWithString�,runtime.mapiterinit·f&runtime.mapiterinit�,runtime.mapiternext·f&runtime.mapiternext�regexp.(*Regexp).MatchString·f8regexp.(*Regexp).MatchString�("".newRouteRegexp·f""".newRouteRegexp�$"".braceIndices·f"".braceIndices�(runtime.makeslice·f"runtime.makeslice�8runtime.stringtoslicebyte·f2runtime.stringtoslicebyte�8bytes.(*Buffer).WriteByte·f2bytes.(*Buffer).WriteByte�"strings.SplitN·fstrings.SplitN�®exp.QuoteMeta·f regexp.QuoteMeta�$"".varGroupName·f"".varGroupName�fmt.Fprintf·ffmt.Fprintf�:runtime.writebarrierstring·f4runtime.writebarrierstring�fmt.Sprintf·ffmt.Sprintf�"".(*routeRegexpGroup).setMatch�Lregexp.(*Regexp).FindStringSubmatch·fFregexp.(*Regexp).FindStringSubmatch� net/url.Parse·fnet/url.Parse� strings.Index·fstrings.Index�."".(*Route).GetError·f("".(*Route).GetError�0"".(*Route).BuildOnly·f*"".(*Route).BuildOnly�,"".(*Route).Handler·f&"".(*Route).Handler�2"".(*Route).GetHandler·f,"".(*Route).GetHandler�&"".(*Route).Name·f "".(*Route).Name�:"".(*Route).getNamedRoutes·f4"".(*Route).getNamedRoutes�,"".(*Route).GetName·f&"".(*Route).GetName�2"".(*Route).addMatcher·f,"".(*Route).addMatcher�>"".(*Route).addRegexpMatcher·f8"".(*Route).addRegexpMatcher�:"".(*Route).getRegexpGroup·f4"".(*Route).getRegexpGroup�(strings.TrimRight·f"strings.TrimRight�2"".headerMatcher.Match·f,"".headerMatcher.Match�<"".headerRegexMatcher.Match·f6"".headerRegexMatcher.Match�8"".(*Route).HeadersRegexp·f2"".(*Route).HeadersRegexp�."".MatcherFunc.Match·f("".MatcherFunc.Match�2"".methodMatcher.Match·f,"".methodMatcher.Match�$strings.ToUpper·fstrings.ToUpper�$runtime.convT2I·fruntime.convT2I�2"".schemeMatcher.Match·f,"".schemeMatcher.Match�$strings.ToLower·fstrings.ToLower�8"".(*Route).BuildVarsFunc·f2"".(*Route).BuildVarsFunc�0"".(*Route).Subrouter·f*"".(*Route).Subrouter�$"".(*Route).URL·f"".(*Route).URL�4"".(*Route).prepareVars·f."".(*Route).prepareVars�,"".(*Route).URLHost·f&"".(*Route).URLHost�,"".(*Route).URLPath·f&"".(*Route).URLPath�0"".(*Route).buildVars·f*"".(*Route).buildVars�"".init·f"".init�(runtime.throwinit·f"runtime.throwinit�strings.init·fstrings.init�strconv.init·fstrconv.init�net/url.init·fnet/url.init�bytes.init·fbytes.init��github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.init·f�github.com/fsouza/go-dockerclient/external/github.com/gorilla/context.init�regexp.init·fregexp.init�path.init·fpath.init� net/http.init·fnet/http.init�fmt.init·ffmt.init�errors.New·ferrors.New�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·eeb28990c0dc813022336c3780186218+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·8cb639c12a4a13c6ace27031b0f83707 �Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·8cb639c12a4a13c6ace27031b0f83707 �bruntime.gcbits.0x48844400000000000000000000000000 H�D�(go.string."[]string"@2[]string (go.string."[]string"�type.[]string��Ө� + � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P(go.string."[]string"p,go.weak.type.*[]string�"runtime.zerovalue�type.string�:go.typelink.[]string/[]stringtype.[]string�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�0type..hashfunc.[8]string(type..hash.[8]string�,type..eqfunc.[8]string$type..eq.[8]string�&type..alg.[8]string 0type..hashfunc.[8]string,type..eqfunc.[8]string�bruntime.gcbits.0x48484848484848480000000000000000 HHHHHHHH�*go.string."[8]string"@4 [8]string *go.string."[8]string"�type.[8]string���US�> &type..alg.[8]string0bruntime.gcbits.0x48484848484848480000000000000000P*go.string."[8]string"p.go.weak.type.*[8]string�"runtime.zerovalue�type.string�type.[]string�>go.typelink.[8]string/[8]stringtype.[8]string�bruntime.gcbits.0x88000000000000000000000000000000 ��Jgo.string."*map.bucket[string]string"`T*map.bucket[string]string Jgo.string."*map.bucket[string]string"�Y� � runtime.algarray0Btype..gc.map.bucket[string]string@Jtype..gcprog.map.bucket[string]stringPHgo.string."map.bucket[string]string"pLgo.weak.type.*map.bucket[string]string�"runtime.zerovalue��:type.map.bucket[string]string� go.string."keys"�type.[8]string�$go.string."values"�type.[8]string�(go.string."overflow"�go.weak.type.*map[string]string�"runtime.zerovalue�type.string�type.string�:type.map.bucket[string]string�4type.map.hdr[string]string�^go.typelink.map[string]string/map[string]string,type.map[string]string�jgo.string."func(map[string]string) map[string]string"�t)func(map[string]string) map[string]string jgo.string."func(map[string]string) map[string]string"�\type.func(map[string]string) map[string]string��B��$3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pjgo.string."func(map[string]string) map[string]string"pngo.weak.type.*func(map[string]string) map[string]string�"runtime.zerovalue��\type.func(map[string]string) map[string]string��\type.func(map[string]string) map[string]string�,type.map[string]string�,type.map[string]string�Pgo.string."func() map[string]*mux.Route"`Zfunc() map[string]*mux.Route Pgo.string."func() map[string]*mux.Route"�@type.func() map[string]*"".Route�����93 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."func() map[string]*mux.Route"pRgo.weak.type.*func() map[string]*"".Route�"runtime.zerovalue��@type.func() map[string]*"".Route��@type.func() map[string]*"".Route�2type.map[string]*"".Route�8go.string."[]*regexp.Regexp"PB[]*regexp.Regexp 8go.string."[]*regexp.Regexp"�*type.[]*regexp.Regexp����p� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P8go.string."[]*regexp.Regexp"pmux.RouteMatch 4go.string."mux.RouteMatch"�"go.string."Route"0,Route "go.string."Route"�&go.string."Handler"00Handler &go.string."Handler"� go.string."Vars"0*Vars go.string."Vars"�,go.string."RouteMatch"@6 +RouteMatch ,go.string."RouteMatch"�$type."".RouteMatch�� ���@ � runtime.algarray0bruntime.gcbits.0xc8880000000000000000000000000000P4go.string."mux.RouteMatch"p&type.*"".RouteMatch�"runtime.zerovalue��$type."".RouteMatch�"go.string."Route"�type.*"".Route�&go.string."Handler"�*type.net/http.Handler� go.string."Vars"�,type.map[string]string`�$type."".RouteMatch�,go.string."RouteMatch"�"go.importpath."".��$type."".RouteMatch�6go.string."*mux.RouteMatch"@@*mux.RouteMatch 6go.string."*mux.RouteMatch"�&type.*"".RouteMatch���A�}6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."*mux.RouteMatch"p8go.weak.type.**"".RouteMatch�"runtime.zerovalue�$type."".RouteMatch��go.string."func(*mux.routeRegexp, *http.Request, *mux.RouteMatch) bool"��;func(*mux.routeRegexp, *http.Request, *mux.RouteMatch) bool �go.string."func(*mux.routeRegexp, *http.Request, *mux.RouteMatch) bool"��type.func(*"".routeRegexp, *net/http.Request, *"".RouteMatch) bool���`3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*mux.routeRegexp, *http.Request, *mux.RouteMatch) bool"p�go.weak.type.*func(*"".routeRegexp, *net/http.Request, *"".RouteMatch) bool�"runtime.zerovalue���type.func(*"".routeRegexp, *net/http.Request, *"".RouteMatch) bool���type.func(*"".routeRegexp, *net/http.Request, *"".RouteMatch) bool�(type.*"".routeRegexp�,type.*net/http.Request�&type.*"".RouteMatch�type.bool�pgo.string."func(*mux.routeRegexp, *http.Request) string"�z,func(*mux.routeRegexp, *http.Request) string pgo.string."func(*mux.routeRegexp, *http.Request) string"�htype.func(*"".routeRegexp, *net/http.Request) string���u�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ppgo.string."func(*mux.routeRegexp, *http.Request) string"pzgo.weak.type.*func(*"".routeRegexp, *net/http.Request) string�"runtime.zerovalue��htype.func(*"".routeRegexp, *net/http.Request) string��htype.func(*"".routeRegexp, *net/http.Request) string�(type.*"".routeRegexp�,type.*net/http.Request�type.string�lgo.string."func(*mux.routeRegexp, *http.Request) bool"�v*func(*mux.routeRegexp, *http.Request) bool lgo.string."func(*mux.routeRegexp, *http.Request) bool"�dtype.func(*"".routeRegexp, *net/http.Request) bool�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Plgo.string."func(*mux.routeRegexp, *http.Request) bool"pvgo.weak.type.*func(*"".routeRegexp, *net/http.Request) bool�"runtime.zerovalue��dtype.func(*"".routeRegexp, *net/http.Request) bool��dtype.func(*"".routeRegexp, *net/http.Request) bool�(type.*"".routeRegexp�,type.*net/http.Request�type.bool��go.string."func(*mux.routeRegexp, map[string]string) (string, error)"��9func(*mux.routeRegexp, map[string]string) (string, error) �go.string."func(*mux.routeRegexp, map[string]string) (string, error)"�ztype.func(*"".routeRegexp, map[string]string) (string, error)��a]�{3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*mux.routeRegexp, map[string]string) (string, error)"p�go.weak.type.*func(*"".routeRegexp, map[string]string) (string, error)�"runtime.zerovalue��ztype.func(*"".routeRegexp, map[string]string) (string, error)��ztype.func(*"".routeRegexp, map[string]string) (string, error)�(type.*"".routeRegexp�,type.map[string]string�type.string�type.error�"go.string."Match"0,Match "go.string."Match"�jgo.string."func(*http.Request, *mux.RouteMatch) bool"�t)func(*http.Request, *mux.RouteMatch) bool jgo.string."func(*http.Request, *mux.RouteMatch) bool"�btype.func(*net/http.Request, *"".RouteMatch) bool��=ձ3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pjgo.string."func(*http.Request, *mux.RouteMatch) bool"ptgo.weak.type.*func(*net/http.Request, *"".RouteMatch) bool�"runtime.zerovalue��btype.func(*net/http.Request, *"".RouteMatch) bool��btype.func(*net/http.Request, *"".RouteMatch) bool�,type.*net/http.Request�&type.*"".RouteMatch�type.bool�.go.string."getUrlQuery"@8 getUrlQuery .go.string."getUrlQuery"�Lgo.string."func(*http.Request) string"`Vfunc(*http.Request) string Lgo.string."func(*http.Request) string"�Ftype.func(*net/http.Request) string���v t3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PLgo.string."func(*http.Request) string"pXgo.weak.type.*func(*net/http.Request) string�"runtime.zerovalue��Ftype.func(*net/http.Request) string��Ftype.func(*net/http.Request) string�,type.*net/http.Request�type.string�8go.string."matchQueryString"PBmatchQueryString 8go.string."matchQueryString"�Hgo.string."func(*http.Request) bool"`Rfunc(*http.Request) bool Hgo.string."func(*http.Request) bool"�Btype.func(*net/http.Request) bool��e���3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PHgo.string."func(*http.Request) bool"pTgo.weak.type.*func(*net/http.Request) bool�"runtime.zerovalue��Btype.func(*net/http.Request) bool��Btype.func(*net/http.Request) bool�,type.*net/http.Request�type.bool�go.string."url"0(url go.string."url"�fgo.string."func(map[string]string) (string, error)"pp'func(map[string]string) (string, error) fgo.string."func(map[string]string) (string, error)"�Xtype.func(map[string]string) (string, error)��^�aG3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pfgo.string."func(map[string]string) (string, error)"pjgo.weak.type.*func(map[string]string) (string, error)�"runtime.zerovalue��Xtype.func(map[string]string) (string, error)��Xtype.func(map[string]string) (string, error)�,type.map[string]string�type.string�type.error�(type.*"".routeRegexp��T~��6> � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P8go.string."*mux.routeRegexp"p:go.weak.type.**"".routeRegexp�"runtime.zerovalue�&type."".routeRegexp`�(type.*"".routeRegexp��(type.*"".routeRegexp�"go.string."Match"�btype.func(*net/http.Request, *"".RouteMatch) bool��type.func(*"".routeRegexp, *net/http.Request, *"".RouteMatch) bool�."".(*routeRegexp).Match�."".(*routeRegexp).Match�.go.string."getUrlQuery"�"go.importpath."".�Ftype.func(*net/http.Request) string�htype.func(*"".routeRegexp, *net/http.Request) string�:"".(*routeRegexp).getUrlQuery�:"".(*routeRegexp).getUrlQuery�8go.string."matchQueryString"�"go.importpath."".�Btype.func(*net/http.Request) bool�dtype.func(*"".routeRegexp, *net/http.Request) bool�D"".(*routeRegexp).matchQueryString�D"".(*routeRegexp).matchQueryString�go.string."url"�"go.importpath."".�Xtype.func(map[string]string) (string, error)�ztype.func(*"".routeRegexp, map[string]string) (string, error)�*"".(*routeRegexp).url�*"".(*routeRegexp).url�go.weak.type.*[]*"".routeRegexp�"runtime.zerovalue�(type.*"".routeRegexp�`go.typelink.[]*mux.routeRegexp/[]*"".routeRegexp,type.[]*"".routeRegexp�bruntime.gcbits.0x88488488440000000000000000000000 �H��D�@go.string."mux.routeRegexpGroup"PJmux.routeRegexpGroup @go.string."mux.routeRegexpGroup"� go.string."host"0*host go.string."host"� go.string."path"0*path go.string."path"�&go.string."queries"00queries &go.string."queries"�8go.string."routeRegexpGroup"PBrouteRegexpGroup 8go.string."routeRegexpGroup"�0type."".routeRegexpGroup��(�1JF& � runtime.algarray0bruntime.gcbits.0x88488488440000000000000000000000P@go.string."mux.routeRegexpGroup"p2type.*"".routeRegexpGroup�"runtime.zerovalue��0type."".routeRegexpGroup� go.string."host"�"go.importpath."".�(type.*"".routeRegexp� go.string."path"�"go.importpath."".�(type.*"".routeRegexp�&go.string."queries"�"go.importpath."".�,type.[]*"".routeRegexp`�0type."".routeRegexpGroup�8go.string."routeRegexpGroup"�"go.importpath."".��0type."".routeRegexpGroup�Bgo.string."*mux.routeRegexpGroup"PL*mux.routeRegexpGroup Bgo.string."*mux.routeRegexpGroup"��go.string."func(*mux.routeRegexpGroup, *http.Request, *mux.RouteMatch, *mux.Route)"��Gfunc(*mux.routeRegexpGroup, *http.Request, *mux.RouteMatch, *mux.Route) �go.string."func(*mux.routeRegexpGroup, *http.Request, *mux.RouteMatch, *mux.Route)"��type.func(*"".routeRegexpGroup, *net/http.Request, *"".RouteMatch, *"".Route)��'8D3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*mux.routeRegexpGroup, *http.Request, *mux.RouteMatch, *mux.Route)"p�go.weak.type.*func(*"".routeRegexpGroup, *net/http.Request, *"".RouteMatch, *"".Route)�"runtime.zerovalue���type.func(*"".routeRegexpGroup, *net/http.Request, *"".RouteMatch, *"".Route)���type.func(*"".routeRegexpGroup, *net/http.Request, *"".RouteMatch, *"".Route)�2type.*"".routeRegexpGroup�,type.*net/http.Request�&type.*"".RouteMatch�type.*"".Route�(go.string."setMatch"@2setMatch (go.string."setMatch"�xgo.string."func(*http.Request, *mux.RouteMatch, *mux.Route)"��0func(*http.Request, *mux.RouteMatch, *mux.Route) xgo.string."func(*http.Request, *mux.RouteMatch, *mux.Route)"�ntype.func(*net/http.Request, *"".RouteMatch, *"".Route)��C/O�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pxgo.string."func(*http.Request, *mux.RouteMatch, *mux.Route)"p�go.weak.type.*func(*net/http.Request, *"".RouteMatch, *"".Route)�"runtime.zerovalue��ntype.func(*net/http.Request, *"".RouteMatch, *"".Route)��ntype.func(*net/http.Request, *"".RouteMatch, *"".Route)�,type.*net/http.Request�&type.*"".RouteMatch�type.*"".Route�2type.*"".routeRegexpGroup��ρ�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."*mux.routeRegexpGroup"pDgo.weak.type.**"".routeRegexpGroup�"runtime.zerovalue�0type."".routeRegexpGroup`�2type.*"".routeRegexpGroup��2type.*"".routeRegexpGroup�(go.string."setMatch"�"go.importpath."".�ntype.func(*net/http.Request, *"".RouteMatch, *"".Route)��type.func(*"".routeRegexpGroup, *net/http.Request, *"".RouteMatch, *"".Route)�>"".(*routeRegexpGroup).setMatch�>"".(*routeRegexpGroup).setMatch�Pgo.string."func() *mux.routeRegexpGroup"`Zfunc() *mux.routeRegexpGroup Pgo.string."func() *mux.routeRegexpGroup"�@type.func() *"".routeRegexpGroup����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."func() *mux.routeRegexpGroup"pRgo.weak.type.*func() *"".routeRegexpGroup�"runtime.zerovalue��@type.func() *"".routeRegexpGroup��@type.func() *"".routeRegexpGroup�2type.*"".routeRegexpGroup�8go.string."*mux.parentRoute"PB*mux.parentRoute 8go.string."*mux.parentRoute"�(type.*"".parentRoute��Ǧ��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P8go.string."*mux.parentRoute"p:go.weak.type.**"".parentRoute�"runtime.zerovalue�&type."".parentRoute�bruntime.gcbits.0x8c000000000000000000000000000000 ��6go.string."mux.parentRoute"@@mux.parentRoute 6go.string."mux.parentRoute"�*go.string."buildVars"@4 buildVars *go.string."buildVars"�4go.string."getNamedRoutes"@>getNamedRoutes 4go.string."getNamedRoutes"�4go.string."getRegexpGroup"@>getRegexpGroup 4go.string."getRegexpGroup"�.go.string."parentRoute"@8 parentRoute .go.string."parentRoute"�&type."".parentRoute�����& � runtime.algarray0bruntime.gcbits.0x8c000000000000000000000000000000P6go.string."mux.parentRoute"p(type.*"".parentRoute�"runtime.zerovalue��&type."".parentRoute�*go.string."buildVars"�"go.importpath."".�\type.func(map[string]string) map[string]string�4go.string."getNamedRoutes"�"go.importpath."".�@type.func() map[string]*"".Route�4go.string."getRegexpGroup"�"go.importpath."".�@type.func() *"".routeRegexpGroup`�&type."".parentRoute�.go.string."parentRoute"�"go.importpath."".��&type."".parentRoute�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·564befda8e2e8cc7f35f6bc1d3c5e0a6 +��0go.string."*mux.matcher"@: *mux.matcher 0go.string."*mux.matcher"� type.*"".matcher�� +r��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P0go.string."*mux.matcher"p2go.weak.type.**"".matcher�"runtime.zerovalue�type."".matcher�.go.string."mux.matcher"@8 mux.matcher .go.string."mux.matcher"�&go.string."matcher"00matcher &go.string."matcher"�type."".matcher���Fq� � runtime.algarray0bruntime.gcbits.0x8c000000000000000000000000000000P.go.string."mux.matcher"p type.*"".matcher�"runtime.zerovalue��type."".matcher�"go.string."Match"�btype.func(*net/http.Request, *"".RouteMatch) bool`�type."".matcher�&go.string."matcher"�"go.importpath."".��type."".matcher�2go.string."[]mux.matcher"@< []mux.matcher 2go.string."[]mux.matcher"�"type.[]"".matcher��*P�  � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P2go.string."[]mux.matcher"p4go.weak.type.*[]"".matcher�"runtime.zerovalue�type."".matcher�Lgo.typelink.[]mux.matcher/[]"".matcher"type.[]"".matcher�go.weak.type.**"".BuildVarsFunc�"runtime.zerovalue�*type."".BuildVarsFunc�:go.string."mux.BuildVarsFunc"PDmux.BuildVarsFunc :go.string."mux.BuildVarsFunc"�2go.string."BuildVarsFunc"@< BuildVarsFunc 2go.string."BuildVarsFunc"�*type."".BuildVarsFunc���ˬ�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."mux.BuildVarsFunc"p,type.*"".BuildVarsFunc�"runtime.zerovalue��*type."".BuildVarsFunc��*type."".BuildVarsFunc�,type.map[string]string�,type.map[string]string`�*type."".BuildVarsFunc�2go.string."BuildVarsFunc"�"go.importpath."".��*type."".BuildVarsFunc�bruntime.gcbits.0x8c8c488484c488000000000000000000 ��H��Ĉ�*go.string."mux.Route"@4 mux.Route *go.string."mux.Route"�$go.string."parent"0.parent $go.string."parent"�&go.string."handler"00handler &go.string."handler"�(go.string."matchers"@2matchers (go.string."matchers"�*go.string."buildOnly"@4 buildOnly *go.string."buildOnly"� go.string."name"0*name go.string."name"�go.string."err"0(err go.string."err"�2go.string."buildVarsFunc"@< buildVarsFunc 2go.string."buildVarsFunc"�type."".Route��p}0�  8@AHXhJ � runtime.algarray0bruntime.gcbits.0x8c8c488484c488000000000000000000P*go.string."mux.Route"ptype.*"".Route�"runtime.zerovalue��type."".Route�$go.string."parent"�"go.importpath."".�&type."".parentRoute�&go.string."handler"�"go.importpath."".�*type.net/http.Handler�(go.string."matchers"�"go.importpath."".�"type.[]"".matcher�$go.string."regexp"�"go.importpath."".�2type.*"".routeRegexpGroup�.go.string."strictSlash"�"go.importpath."".�type.bool�*go.string."buildOnly"�"go.importpath."".�type.bool� go.string."name"�"go.importpath."".�type.string�go.string."err"�"go.importpath."".�type.error�2go.string."buildVarsFunc"�"go.importpath."".�*type."".BuildVarsFunc`�type."".Route�"go.string."Route"�"go.importpath."".��type."".Route�,go.string."*mux.Route"@6 +*mux.Route ,go.string."*mux.Route"�Ngo.string."func(*mux.Route) *mux.Route"`Xfunc(*mux.Route) *mux.Route Ngo.string."func(*mux.Route) *mux.Route"�type.func(*"".Router) *"".Route������3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."func(*mux.Router) *mux.Route"pPgo.weak.type.*func(*"".Router) *"".Route�"runtime.zerovalue��>type.func(*"".Router) *"".Route��>type.func(*"".Router) *"".Route�type.*"".Router�type.*"".Route��go.string."func(*mux.Router, http.ResponseWriter, *http.Request)"��5func(*mux.Router, http.ResponseWriter, *http.Request) �go.string."func(*mux.Router, http.ResponseWriter, *http.Request)"��type.func(*"".Router, net/http.ResponseWriter, *net/http.Request)��|�t�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*mux.Router, http.ResponseWriter, *http.Request)"p�go.weak.type.*func(*"".Router, net/http.ResponseWriter, *net/http.Request)�"runtime.zerovalue���type.func(*"".Router, net/http.ResponseWriter, *net/http.Request)���type.func(*"".Router, net/http.ResponseWriter, *net/http.Request)�type.*"".Router�8type.net/http.ResponseWriter�,type.*net/http.Request�^go.string."func(*mux.Router, bool) *mux.Router"ph#func(*mux.Router, bool) *mux.Router ^go.string."func(*mux.Router, bool) *mux.Router"�Ltype.func(*"".Router, bool) *"".Router��?�Ч3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P^go.string."func(*mux.Router, bool) *mux.Router"p^go.weak.type.*func(*"".Router, bool) *"".Router�"runtime.zerovalue��Ltype.func(*"".Router, bool) *"".Router��Ltype.func(*"".Router, bool) *"".Router�type.*"".Router�type.bool�type.*"".Router�2go.string."*mux.WalkFunc"@< *mux.WalkFunc 2go.string."*mux.WalkFunc"�"type.*"".WalkFunc��!o6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P2go.string."*mux.WalkFunc"p4go.weak.type.**"".WalkFunc�"runtime.zerovalue� type."".WalkFunc�0go.string."mux.WalkFunc"@: mux.WalkFunc 0go.string."mux.WalkFunc"�(go.string."WalkFunc"@2WalkFunc (go.string."WalkFunc"� type."".WalkFunc��[�E�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P0go.string."mux.WalkFunc"p"type.*"".WalkFunc�"runtime.zerovalue�� type."".WalkFunc�� type."".WalkFunc�type.*"".Route�type.*"".Router� type.[]*"".Route�type.error`� type."".WalkFunc�(go.string."WalkFunc"�"go.importpath."".�� type."".WalkFunc�bgo.string."func(*mux.Router, mux.WalkFunc) error"pl%func(*mux.Router, mux.WalkFunc) error bgo.string."func(*mux.Router, mux.WalkFunc) error"�Ptype.func(*"".Router, "".WalkFunc) error��ӂ�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pbgo.string."func(*mux.Router, mux.WalkFunc) error"pbgo.weak.type.*func(*"".Router, "".WalkFunc) error�"runtime.zerovalue��Ptype.func(*"".Router, "".WalkFunc) error��Ptype.func(*"".Router, "".WalkFunc) error�type.*"".Router� type."".WalkFunc�type.error��go.string."func(*mux.Router, map[string]string) map[string]string"��6func(*mux.Router, map[string]string) map[string]string �go.string."func(*mux.Router, map[string]string) map[string]string"�ttype.func(*"".Router, map[string]string) map[string]string��*��V3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*mux.Router, map[string]string) map[string]string"p�go.weak.type.*func(*"".Router, map[string]string) map[string]string�"runtime.zerovalue��ttype.func(*"".Router, map[string]string) map[string]string��ttype.func(*"".Router, map[string]string) map[string]string�type.*"".Router�,type.map[string]string�,type.map[string]string�fgo.string."func(*mux.Router) map[string]*mux.Route"pp'func(*mux.Router) map[string]*mux.Route fgo.string."func(*mux.Router) map[string]*mux.Route"�Ttype.func(*"".Router) map[string]*"".Route�����%3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pfgo.string."func(*mux.Router) map[string]*mux.Route"pfgo.weak.type.*func(*"".Router) map[string]*"".Route�"runtime.zerovalue��Ttype.func(*"".Router) map[string]*"".Route��Ttype.func(*"".Router) map[string]*"".Route�type.*"".Router�2type.map[string]*"".Route�fgo.string."func(*mux.Router) *mux.routeRegexpGroup"pp'func(*mux.Router) *mux.routeRegexpGroup fgo.string."func(*mux.Router) *mux.routeRegexpGroup"�Ttype.func(*"".Router) *"".routeRegexpGroup���^]3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pfgo.string."func(*mux.Router) *mux.routeRegexpGroup"pfgo.weak.type.*func(*"".Router) *"".routeRegexpGroup�"runtime.zerovalue��Ttype.func(*"".Router) *"".routeRegexpGroup��Ttype.func(*"".Router) *"".routeRegexpGroup�type.*"".Router�2type.*"".routeRegexpGroup�~go.string."func(*mux.Router, mux.WalkFunc, []*mux.Route) error"��3func(*mux.Router, mux.WalkFunc, []*mux.Route) error ~go.string."func(*mux.Router, mux.WalkFunc, []*mux.Route) error"�jtype.func(*"".Router, "".WalkFunc, []*"".Route) error����ms3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P~go.string."func(*mux.Router, mux.WalkFunc, []*mux.Route) error"p|go.weak.type.*func(*"".Router, "".WalkFunc, []*"".Route) error�"runtime.zerovalue��jtype.func(*"".Router, "".WalkFunc, []*"".Route) error��jtype.func(*"".Router, "".WalkFunc, []*"".Route) error�type.*"".Router� type."".WalkFunc� type.[]*"".Route�type.error�\go.string."func(mux.BuildVarsFunc) *mux.Route"pf"func(mux.BuildVarsFunc) *mux.Route \go.string."func(mux.BuildVarsFunc) *mux.Route"�Jtype.func("".BuildVarsFunc) *"".Route���^c�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P\go.string."func(mux.BuildVarsFunc) *mux.Route"p\go.weak.type.*func("".BuildVarsFunc) *"".Route�"runtime.zerovalue��Jtype.func("".BuildVarsFunc) *"".Route��Jtype.func("".BuildVarsFunc) *"".Route�*type."".BuildVarsFunc�type.*"".Route�go.string."Get"0(Get go.string."Get"�Fgo.string."func(string) *mux.Route"PPfunc(string) *mux.Route Fgo.string."func(string) *mux.Route"�6type.func(string) *"".Route���Q��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PFgo.string."func(string) *mux.Route"pHgo.weak.type.*func(string) *"".Route�"runtime.zerovalue��6type.func(string) *"".Route��6type.func(string) *"".Route�type.string�type.*"".Route�(go.string."GetRoute"@2GetRoute (go.string."GetRoute"�$go.string."Handle"0.Handle $go.string."Handle"�bgo.string."func(string, http.Handler) *mux.Route"pl%func(string, http.Handler) *mux.Route bgo.string."func(string, http.Handler) *mux.Route"�Ztype.func(string, net/http.Handler) *"".Route�� z3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pbgo.string."func(string, http.Handler) *mux.Route"plgo.weak.type.*func(string, net/http.Handler) *"".Route�"runtime.zerovalue��Ztype.func(string, net/http.Handler) *"".Route��Ztype.func(string, net/http.Handler) *"".Route�type.string�*type.net/http.Handler�type.*"".Route�,go.string."HandleFunc"@6 +HandleFunc ,go.string."HandleFunc"��go.string."func(string, func(http.ResponseWriter, *http.Request)) *mux.Route"��Afunc(string, func(http.ResponseWriter, *http.Request)) *mux.Route �go.string."func(string, func(http.ResponseWriter, *http.Request)) *mux.Route"��type.func(string, func(net/http.ResponseWriter, *net/http.Request)) *"".Route��Ɋn�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(string, func(http.ResponseWriter, *http.Request)) *mux.Route"p�go.weak.type.*func(string, func(net/http.ResponseWriter, *net/http.Request)) *"".Route�"runtime.zerovalue���type.func(string, func(net/http.ResponseWriter, *net/http.Request)) *"".Route���type.func(string, func(net/http.ResponseWriter, *net/http.Request)) *"".Route�type.string�jtype.func(net/http.ResponseWriter, *net/http.Request)�type.*"".Route�&go.string."Headers"00Headers &go.string."Headers"�Lgo.string."func(...string) *mux.Route"`Vfunc(...string) *mux.Route Lgo.string."func(...string) *mux.Route"�type.func(*"".Router) *"".Route� *"".(*Router).NewRoute� +*"".(*Router).NewRoute� + go.string."Path"� +6type.func(string) *"".Route� +Ntype.func(*"".Router, string) *"".Route� +""".(*Router).Path� +""".(*Router).Path� +,go.string."PathPrefix"� 6type.func(string) *"".Route� Ntype.func(*"".Router, string) *"".Route� ."".(*Router).PathPrefix� ."".(*Router).PathPrefix� &go.string."Queries"� type.func(*"".Route) *"".Router���;�!3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."func(*mux.Route) *mux.Router"pPgo.weak.type.*func(*"".Route) *"".Router�"runtime.zerovalue��>type.func(*"".Route) *"".Router��>type.func(*"".Route) *"".Router�type.*"".Route�type.*"".Router�rgo.string."func(*mux.Route, ...string) (*url.URL, error)"�|-func(*mux.Route, ...string) (*url.URL, error) rgo.string."func(*mux.Route, ...string) (*url.URL, error)"�jtype.func(*"".Route, ...string) (*net/url.URL, error)����Ui3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Prgo.string."func(*mux.Route, ...string) (*url.URL, error)"p|go.weak.type.*func(*"".Route, ...string) (*net/url.URL, error)�"runtime.zerovalue��jtype.func(*"".Route, ...string) (*net/url.URL, error)��jtype.func(*"".Route, ...string) (*net/url.URL, error)�type.*"".Route�type.[]string�"type.*net/url.URL�type.error�hgo.string."func(*mux.Route, mux.matcher) *mux.Route"�r(func(*mux.Route, mux.matcher) *mux.Route hgo.string."func(*mux.Route, mux.matcher) *mux.Route"�Ttype.func(*"".Route, "".matcher) *"".Route�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Phgo.string."func(*mux.Route, mux.matcher) *mux.Route"pfgo.weak.type.*func(*"".Route, "".matcher) *"".Route�"runtime.zerovalue��Ttype.func(*"".Route, "".matcher) *"".Route��Ttype.func(*"".Route, "".matcher) *"".Route�type.*"".Route�type."".matcher�type.*"".Route�xgo.string."func(*mux.Route, string, bool, bool, bool) error"��0func(*mux.Route, string, bool, bool, bool) error xgo.string."func(*mux.Route, string, bool, bool, bool) error"�htype.func(*"".Route, string, bool, bool, bool) error��\7�X3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pxgo.string."func(*mux.Route, string, bool, bool, bool) error"pzgo.weak.type.*func(*"".Route, string, bool, bool, bool) error�"runtime.zerovalue��htype.func(*"".Route, string, bool, bool, bool) error��htype.func(*"".Route, string, bool, bool, bool) error�type.*"".Route�type.string�type.bool�type.bool�type.bool�type.error��go.string."func(*mux.Route, map[string]string) map[string]string"��5func(*mux.Route, map[string]string) map[string]string �go.string."func(*mux.Route, map[string]string) map[string]string"�rtype.func(*"".Route, map[string]string) map[string]string���WG3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*mux.Route, map[string]string) map[string]string"p�go.weak.type.*func(*"".Route, map[string]string) map[string]string�"runtime.zerovalue��rtype.func(*"".Route, map[string]string) map[string]string��rtype.func(*"".Route, map[string]string) map[string]string�type.*"".Route�,type.map[string]string�,type.map[string]string�dgo.string."func(*mux.Route) map[string]*mux.Route"pn&func(*mux.Route) map[string]*mux.Route dgo.string."func(*mux.Route) map[string]*mux.Route"�Rtype.func(*"".Route) map[string]*"".Route��X� 3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pdgo.string."func(*mux.Route) map[string]*mux.Route"pdgo.weak.type.*func(*"".Route) map[string]*"".Route�"runtime.zerovalue��Rtype.func(*"".Route) map[string]*"".Route��Rtype.func(*"".Route) map[string]*"".Route�type.*"".Route�2type.map[string]*"".Route�dgo.string."func(*mux.Route) *mux.routeRegexpGroup"pn&func(*mux.Route) *mux.routeRegexpGroup dgo.string."func(*mux.Route) *mux.routeRegexpGroup"�Rtype.func(*"".Route) *"".routeRegexpGroup��=� +3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pdgo.string."func(*mux.Route) *mux.routeRegexpGroup"pdgo.weak.type.*func(*"".Route) *"".routeRegexpGroup�"runtime.zerovalue��Rtype.func(*"".Route) *"".routeRegexpGroup��Rtype.func(*"".Route) *"".routeRegexpGroup�type.*"".Route�2type.*"".routeRegexpGroup��go.string."func(*mux.Route, ...string) (map[string]string, error)"��6func(*mux.Route, ...string) (map[string]string, error) �go.string."func(*mux.Route, ...string) (map[string]string, error)"�ttype.func(*"".Route, ...string) (map[string]string, error)���xn3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*mux.Route, ...string) (map[string]string, error)"p�go.weak.type.*func(*"".Route, ...string) (map[string]string, error)�"runtime.zerovalue��ttype.func(*"".Route, ...string) (map[string]string, error)��ttype.func(*"".Route, ...string) (map[string]string, error)�type.*"".Route�type.[]string�,type.map[string]string�type.error�*go.string."BuildOnly"@4 BuildOnly *go.string."BuildOnly"�(go.string."GetError"@2GetError (go.string."GetError"�0go.string."func() error"@: func() error 0go.string."func() error"�"type.func() error����ֵ3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P0go.string."func() error"p4go.weak.type.*func() error�"runtime.zerovalue��"type.func() error��"type.func() error�type.error�,go.string."GetHandler"@6 +GetHandler ,go.string."GetHandler"�>go.string."func() http.Handler"PHfunc() http.Handler >go.string."func() http.Handler"�8type.func() net/http.Handler����53 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."func() http.Handler"pJgo.weak.type.*func() net/http.Handler�"runtime.zerovalue��8type.func() net/http.Handler��8type.func() net/http.Handler�*type.net/http.Handler�&go.string."GetName"00GetName &go.string."GetName"�2go.string."func() string"@< func() string 2go.string."func() string"�$type.func() string���m�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P2go.string."func() string"p6go.weak.type.*func() string�"runtime.zerovalue��$type.func() string��$type.func() string�type.string�Rgo.string."func(http.Handler) *mux.Route"`\func(http.Handler) *mux.Route Rgo.string."func(http.Handler) *mux.Route"�Jtype.func(net/http.Handler) *"".Route�� {vN3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PRgo.string."func(http.Handler) *mux.Route"p\go.weak.type.*func(net/http.Handler) *"".Route�"runtime.zerovalue��Jtype.func(net/http.Handler) *"".Route��Jtype.func(net/http.Handler) *"".Route�*type.net/http.Handler�type.*"".Route�.go.string."HandlerFunc"@8 HandlerFunc .go.string."HandlerFunc"��go.string."func(func(http.ResponseWriter, *http.Request)) *mux.Route"��9func(func(http.ResponseWriter, *http.Request)) *mux.Route �go.string."func(func(http.ResponseWriter, *http.Request)) *mux.Route"��type.func(func(net/http.ResponseWriter, *net/http.Request)) *"".Route����5A3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(func(http.ResponseWriter, *http.Request)) *mux.Route"p�go.weak.type.*func(func(net/http.ResponseWriter, *net/http.Request)) *"".Route�"runtime.zerovalue���type.func(func(net/http.ResponseWriter, *net/http.Request)) *"".Route���type.func(func(net/http.ResponseWriter, *net/http.Request)) *"".Route�jtype.func(net/http.ResponseWriter, *net/http.Request)�type.*"".Route�2go.string."HeadersRegexp"@< HeadersRegexp 2go.string."HeadersRegexp"� go.string."Name"0*Name go.string."Name"�*go.string."Subrouter"@4 Subrouter *go.string."Subrouter"�go.weak.type.*func() *"".Router�"runtime.zerovalue��,type.func() *"".Router��,type.func() *"".Router�type.*"".Router�go.string."URL"0(URL go.string."URL"�Zgo.string."func(...string) (*url.URL, error)"pd!func(...string) (*url.URL, error) Zgo.string."func(...string) (*url.URL, error)"�Ttype.func(...string) (*net/url.URL, error)��4I͡3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PZgo.string."func(...string) (*url.URL, error)"pfgo.weak.type.*func(...string) (*net/url.URL, error)�"runtime.zerovalue��Ttype.func(...string) (*net/url.URL, error)��Ttype.func(...string) (*net/url.URL, error)�type.[]string�"type.*net/url.URL�type.error�&go.string."URLHost"00URLHost &go.string."URLHost"�&go.string."URLPath"00URLPath &go.string."URLPath"�,go.string."addMatcher"@6 +addMatcher ,go.string."addMatcher"�Pgo.string."func(mux.matcher) *mux.Route"`Zfunc(mux.matcher) *mux.Route Pgo.string."func(mux.matcher) *mux.Route"�>type.func("".matcher) *"".Route��rą�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."func(mux.matcher) *mux.Route"pPgo.weak.type.*func("".matcher) *"".Route�"runtime.zerovalue��>type.func("".matcher) *"".Route��>type.func("".matcher) *"".Route�type."".matcher�type.*"".Route�8go.string."addRegexpMatcher"PBaddRegexpMatcher 8go.string."addRegexpMatcher"�`go.string."func(string, bool, bool, bool) error"pj$func(string, bool, bool, bool) error `go.string."func(string, bool, bool, bool) error"�Rtype.func(string, bool, bool, bool) error�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P`go.string."func(string, bool, bool, bool) error"pdgo.weak.type.*func(string, bool, bool, bool) error�"runtime.zerovalue��Rtype.func(string, bool, bool, bool) error��Rtype.func(string, bool, bool, bool) error�type.string�type.bool�type.bool�type.bool�type.error�.go.string."prepareVars"@8 prepareVars .go.string."prepareVars"�lgo.string."func(...string) (map[string]string, error)"�v*func(...string) (map[string]string, error) lgo.string."func(...string) (map[string]string, error)"�^type.func(...string) (map[string]string, error)���N�D3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Plgo.string."func(...string) (map[string]string, error)"ppgo.weak.type.*func(...string) (map[string]string, error)�"runtime.zerovalue��^type.func(...string) (map[string]string, error)��^type.func(...string) (map[string]string, error)�type.[]string�,type.map[string]string�type.error�type.*"".Route��a[�6� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."*mux.Route"p.go.weak.type.**"".Route�"runtime.zerovalue�type."".Route`�type.*"".Route��type.*"".Route�*go.string."BuildOnly"�*type.func() *"".Route�type.func(*"".Route) *"".Router�*"".(*Route).Subrouter�*"".(*Route).Subrouter�go.string."URL"�Ttype.func(...string) (*net/url.URL, error)�jtype.func(*"".Route, ...string) (*net/url.URL, error)�"".(*Route).URL�"".(*Route).URL�&go.string."URLHost"�Ttype.func(...string) (*net/url.URL, error)�jtype.func(*"".Route, ...string) (*net/url.URL, error)�&"".(*Route).URLHost�&"".(*Route).URLHost�&go.string."URLPath"�Ttype.func(...string) (*net/url.URL, error)�jtype.func(*"".Route, ...string) (*net/url.URL, error)�&"".(*Route).URLPath�&"".(*Route).URLPath�,go.string."addMatcher"�"go.importpath."".�>type.func("".matcher) *"".Route�Ttype.func(*"".Route, "".matcher) *"".Route�,"".(*Route).addMatcher�,"".(*Route).addMatcher�8go.string."addRegexpMatcher"�"go.importpath."".�Rtype.func(string, bool, bool, bool) error�htype.func(*"".Route, string, bool, bool, bool) error�8"".(*Route).addRegexpMatcher�8"".(*Route).addRegexpMatcher�*go.string."buildVars"�"go.importpath."".�\type.func(map[string]string) map[string]string�rtype.func(*"".Route, map[string]string) map[string]string�*"".(*Route).buildVars�*"".(*Route).buildVars�4go.string."getNamedRoutes"�"go.importpath."".�@type.func() map[string]*"".Route�Rtype.func(*"".Route) map[string]*"".Route�4"".(*Route).getNamedRoutes�4"".(*Route).getNamedRoutes�4go.string."getRegexpGroup"�"go.importpath."".�@type.func() *"".routeRegexpGroup�Rtype.func(*"".Route) *"".routeRegexpGroup�4"".(*Route).getRegexpGroup�4"".(*Route).getRegexpGroup�.go.string."prepareVars"�"go.importpath."".�^type.func(...string) (map[string]string, error)�ttype.func(*"".Route, ...string) (map[string]string, error)�."".(*Route).prepareVars�."".(*Route).prepareVars�bruntime.gcbits.0x88888888000000000000000000000000 �����2go.string."[8]*mux.Route"@< [8]*mux.Route 2go.string."[8]*mux.Route"�"type.[8]*"".Route��@O�`  runtime.algarray0bruntime.gcbits.0x88888888000000000000000000000000P2go.string."[8]*mux.Route"p4go.weak.type.*[8]*"".Route�"runtime.zerovalue�type.*"".Route� type.[]*"".Route�Lgo.typelink.[8]*mux.Route/[8]*"".Route"type.[8]*"".Route�Rgo.string."*map.bucket[string]*mux.Route"`\*map.bucket[string]*mux.Route Rgo.string."*map.bucket[string]*mux.Route"�Btype.*map.bucket[string]*"".Route���!Om6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PRgo.string."*map.bucket[string]*mux.Route"pTgo.weak.type.**map.bucket[string]*"".Route�"runtime.zerovalue�@type.map.bucket[string]*"".Route�bruntime.gcbits.0x84848484848484848488888888000000 ��������������Pgo.string."map.bucket[string]*mux.Route"`Zmap.bucket[string]*mux.Route Pgo.string."map.bucket[string]*mux.Route"�@type.map.bucket[string]*"".Route�������� � runtime.algarray0bruntime.gcbits.0x84848484848484848488888888000000PPgo.string."map.bucket[string]*mux.Route"pRgo.weak.type.*map.bucket[string]*"".Route�"runtime.zerovalue��@type.map.bucket[string]*"".Route� go.string."keys"�type.[8]string�$go.string."values"�"type.[8]*"".Route�(go.string."overflow"�Btype.*map.bucket[string]*"".Route�Jgo.string."map.hdr[string]*mux.Route"`Tmap.hdr[string]*mux.Route Jgo.string."map.hdr[string]*mux.Route"�:type.map.hdr[string]*"".Route��0 �  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000PJgo.string."map.hdr[string]*mux.Route"pLgo.weak.type.*map.hdr[string]*"".Route�"runtime.zerovalue��:type.map.hdr[string]*"".Route�&go.string."buckets"�Btype.*map.bucket[string]*"".Route�,go.string."oldbuckets"�Btype.*map.bucket[string]*"".Route�Bgo.string."map[string]*mux.Route"PLmap[string]*mux.Route Bgo.string."map[string]*mux.Route"�2type.map[string]*"".Route���U�5� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."map[string]*mux.Route"pDgo.weak.type.*map[string]*"".Route�"runtime.zerovalue�type.string�type.*"".Route�@type.map.bucket[string]*"".Route�:type.map.hdr[string]*"".Route�lgo.typelink.map[string]*mux.Route/map[string]*"".Route2type.map[string]*"".Route�.go.string."**mux.Route"@8 **mux.Route .go.string."**mux.Route"�type.**"".Route��� �6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P.go.string."**mux.Route"p0go.weak.type.***"".Route�"runtime.zerovalue�type.*"".Route�^runtime.gcbits.0x000000000000000000000000000000 �2go.string."[0]*mux.Route"@< [0]*mux.Route 2go.string."[0]*mux.Route"�"type.[0]*"".Route��oM���  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P2go.string."[0]*mux.Route"p4go.weak.type.*[0]*"".Route�"runtime.zerovalue�type.*"".Route� type.[]*"".Route�Lgo.typelink.[0]*mux.Route/[0]*"".Route"type.[0]*"".Route�4go.string."*[0]*mux.Route"@>*[0]*mux.Route 4go.string."*[0]*mux.Route"�$type.*[0]*"".Route�����Q6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."*[0]*mux.Route"p6go.weak.type.**[0]*"".Route�"runtime.zerovalue�"type.[0]*"".Route�6go.string."*mux.contextKey"@@*mux.contextKey 6go.string."*mux.contextKey"�&type.*"".contextKey��*� 6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P6go.string."*mux.contextKey"p8go.weak.type.**"".contextKey�"runtime.zerovalue�$type."".contextKey�4go.string."mux.contextKey"@>mux.contextKey 4go.string."mux.contextKey"�,go.string."contextKey"@6 +contextKey ,go.string."contextKey"�$type."".contextKey��T5M!� � runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P4go.string."mux.contextKey"p&type.*"".contextKey�"runtime.zerovalue`�$type."".contextKey�,go.string."contextKey"�"go.importpath."".��$type."".contextKey�bruntime.gcbits.0xcc000000000000000000000000000000 ��0go.string."interface {}"@: interface {} 0go.string."interface {}"�"type.interface {}���W� � runtime.algarray0bruntime.gcbits.0xcc000000000000000000000000000000P0go.string."interface {}"p4go.weak.type.*interface {}�"runtime.zerovalue��"type.interface {}�4go.string."[]interface {}"@>[]interface {} 4go.string."[]interface {}"�&type.[]interface {}��p��/ � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P4go.string."[]interface {}"p8go.weak.type.*[]interface {}�"runtime.zerovalue�"type.interface {}�Rgo.typelink.[]interface {}/[]interface {}&type.[]interface {}�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�go.weak.type.*[8]*regexp.Regexp�"runtime.zerovalue�&type.*regexp.Regexp�*type.[]*regexp.Regexp�^go.typelink.[8]*regexp.Regexp/[8]*regexp.Regexp,type.[8]*regexp.Regexp�Zgo.string."*map.bucket[string]*regexp.Regexp"pd!*map.bucket[string]*regexp.Regexp Zgo.string."*map.bucket[string]*regexp.Regexp"�Ltype.*map.bucket[string]*regexp.Regexp����e�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PZgo.string."*map.bucket[string]*regexp.Regexp"p^go.weak.type.**map.bucket[string]*regexp.Regexp�"runtime.zerovalue�Jtype.map.bucket[string]*regexp.Regexp�Xgo.string."map.bucket[string]*regexp.Regexp"pb map.bucket[string]*regexp.Regexp Xgo.string."map.bucket[string]*regexp.Regexp"�Jtype.map.bucket[string]*regexp.Regexp���5D�R�� � runtime.algarray0bruntime.gcbits.0x84848484848484848488888888000000PXgo.string."map.bucket[string]*regexp.Regexp"p\go.weak.type.*map.bucket[string]*regexp.Regexp�"runtime.zerovalue��Jtype.map.bucket[string]*regexp.Regexp� go.string."keys"�type.[8]string�$go.string."values"�,type.[8]*regexp.Regexp�(go.string."overflow"�Ltype.*map.bucket[string]*regexp.Regexp�Rgo.string."map.hdr[string]*regexp.Regexp"`\map.hdr[string]*regexp.Regexp Rgo.string."map.hdr[string]*regexp.Regexp"�Dtype.map.hdr[string]*regexp.Regexp��0���  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000PRgo.string."map.hdr[string]*regexp.Regexp"pVgo.weak.type.*map.hdr[string]*regexp.Regexp�"runtime.zerovalue��Dtype.map.hdr[string]*regexp.Regexp�&go.string."buckets"�Ltype.*map.bucket[string]*regexp.Regexp�,go.string."oldbuckets"�Ltype.*map.bucket[string]*regexp.Regexp�Jgo.string."map[string]*regexp.Regexp"`Tmap[string]*regexp.Regexp Jgo.string."map[string]*regexp.Regexp"�type.map.bucket[string][]string�,Ftype..gc.map.bucket[string][]string,�Ntype..gcprog.map.bucket[string][]string*����Y�eY�e �Lgo.string."map.bucket[string][]string"`Vmap.bucket[string][]string Lgo.string."map.bucket[string][]string"�>type.map.bucket[string][]string��P�TJ�Y�H � runtime.algarray0Ftype..gc.map.bucket[string][]string@Ntype..gcprog.map.bucket[string][]stringPLgo.string."map.bucket[string][]string"pPgo.weak.type.*map.bucket[string][]string�"runtime.zerovalue��>type.map.bucket[string][]string� go.string."keys"�type.[8]string�$go.string."values"� type.[8][]string�(go.string."overflow"�@type.*map.bucket[string][]string�Fgo.string."map.hdr[string][]string"PPmap.hdr[string][]string Fgo.string."map.hdr[string][]string"�8type.map.hdr[string][]string��0����  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000PFgo.string."map.hdr[string][]string"pJgo.weak.type.*map.hdr[string][]string�"runtime.zerovalue��8type.map.hdr[string][]string�&go.string."buckets"�@type.*map.bucket[string][]string�,go.string."oldbuckets"�@type.*map.bucket[string][]string�>go.string."map[string][]string"PHmap[string][]string >go.string."map[string][]string"�0type.map[string][]string��'�>@5P � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."map[string][]string"pBgo.weak.type.*map[string][]string�"runtime.zerovalue�type.string�type.[]string�>type.map.bucket[string][]string�8type.map.hdr[string][]string�fgo.typelink.map[string][]string/map[string][]string0type.map[string][]string�Dgo.string."*map.hdr[string]string"PN*map.hdr[string]string Dgo.string."*map.hdr[string]string"�6type.*map.hdr[string]string����Ƽ6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PDgo.string."*map.hdr[string]string"pHgo.weak.type.**map.hdr[string]string�"runtime.zerovalue�4type.map.hdr[string]string�*go.string."[]uintptr"@4 []uintptr *go.string."[]uintptr"�type.[]uintptr���3�] � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P*go.string."[]uintptr"p.go.weak.type.*[]uintptr�"runtime.zerovalue�type.uintptr�>go.typelink.[]uintptr/[]uintptrtype.[]uintptr�,go.string."[4]uintptr"@6 +[4]uintptr ,go.string."[4]uintptr"�type.[4]uintptr�� l<�  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P,go.string."[4]uintptr"p0go.weak.type.*[4]uintptr�"runtime.zerovalue�type.uintptr�type.[]uintptr�Bgo.typelink.[4]uintptr/[4]uintptrtype.[4]uintptr�bruntime.gcbits.0x88888844440000000000000000000000 ���DD�Dgo.string."map.iter[string]string"PNmap.iter[string]string Dgo.string."map.iter[string]string"�go.string."key"0(key go.string."key"�go.string."val"0(val go.string."val"�go.string."t"0$t go.string."t"�go.string."h"0$h go.string."h"� go.string."bptr"0*bptr go.string."bptr"�"go.string."other"0,other "go.string."other"�6type.map.iter[string]string��P��\ (0( � runtime.algarray0bruntime.gcbits.0x88888844440000000000000000000000PDgo.string."map.iter[string]string"pHgo.weak.type.*map.iter[string]string�"runtime.zerovalue��6type.map.iter[string]string�go.string."key"�type.*string�go.string."val"�type.*string�go.string."t"�type.*uint8�go.string."h"�6type.*map.hdr[string]string�&go.string."buckets"�go.weak.type.**"".headerMatcher�"runtime.zerovalue�*type."".headerMatcher`�,type.*"".headerMatcher��,type.*"".headerMatcher�"go.string."Match"�btype.func(*net/http.Request, *"".RouteMatch) bool��type.func(*"".headerMatcher, *net/http.Request, *"".RouteMatch) bool�2"".(*headerMatcher).Match�2"".(*headerMatcher).Match�:go.string."mux.headerMatcher"PDmux.headerMatcher :go.string."mux.headerMatcher"��go.string."func(mux.headerMatcher, *http.Request, *mux.RouteMatch) bool"��<func(mux.headerMatcher, *http.Request, *mux.RouteMatch) bool �go.string."func(mux.headerMatcher, *http.Request, *mux.RouteMatch) bool"��type.func("".headerMatcher, *net/http.Request, *"".RouteMatch) bool�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(mux.headerMatcher, *http.Request, *mux.RouteMatch) bool"p�go.weak.type.*func("".headerMatcher, *net/http.Request, *"".RouteMatch) bool�"runtime.zerovalue���type.func("".headerMatcher, *net/http.Request, *"".RouteMatch) bool���type.func("".headerMatcher, *net/http.Request, *"".RouteMatch) bool�*type."".headerMatcher�,type.*net/http.Request�&type.*"".RouteMatch�type.bool�*type."".headerMatcher���{K5$ � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P:go.string."mux.headerMatcher"p,type.*"".headerMatcher�"runtime.zerovalue�type.string�type.string�:type.map.bucket[string]string�4type.map.hdr[string]string`�*type."".headerMatcher�2go.string."headerMatcher"�"go.importpath."".��*type."".headerMatcher�"go.string."Match"�btype.func(*net/http.Request, *"".RouteMatch) bool��type.func("".headerMatcher, *net/http.Request, *"".RouteMatch) bool�,"".headerMatcher.Match�,"".headerMatcher.Match�Fgo.string."*mux.headerRegexMatcher"PP*mux.headerRegexMatcher Fgo.string."*mux.headerRegexMatcher"�go.weak.type.**"".methodMatcher�"runtime.zerovalue�*type."".methodMatcher`�,type.*"".methodMatcher��,type.*"".methodMatcher�"go.string."Match"�btype.func(*net/http.Request, *"".RouteMatch) bool��type.func(*"".methodMatcher, *net/http.Request, *"".RouteMatch) bool�2"".(*methodMatcher).Match�2"".(*methodMatcher).Match�:go.string."mux.methodMatcher"PDmux.methodMatcher :go.string."mux.methodMatcher"��go.string."func(mux.methodMatcher, *http.Request, *mux.RouteMatch) bool"��<func(mux.methodMatcher, *http.Request, *mux.RouteMatch) bool �go.string."func(mux.methodMatcher, *http.Request, *mux.RouteMatch) bool"��type.func("".methodMatcher, *net/http.Request, *"".RouteMatch) bool�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(mux.methodMatcher, *http.Request, *mux.RouteMatch) bool"p�go.weak.type.*func("".methodMatcher, *net/http.Request, *"".RouteMatch) bool�"runtime.zerovalue���type.func("".methodMatcher, *net/http.Request, *"".RouteMatch) bool���type.func("".methodMatcher, *net/http.Request, *"".RouteMatch) bool�*type."".methodMatcher�,type.*net/http.Request�&type.*"".RouteMatch�type.bool�*type."".methodMatcher��d� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P:go.string."mux.methodMatcher"p,type.*"".methodMatcher�"runtime.zerovalue�type.string`�*type."".methodMatcher�2go.string."methodMatcher"�"go.importpath."".��*type."".methodMatcher�"go.string."Match"�btype.func(*net/http.Request, *"".RouteMatch) bool��type.func("".methodMatcher, *net/http.Request, *"".RouteMatch) bool�2"".(*methodMatcher).Match�,"".methodMatcher.Match�go.weak.type.**"".schemeMatcher�"runtime.zerovalue�*type."".schemeMatcher`�,type.*"".schemeMatcher��,type.*"".schemeMatcher�"go.string."Match"�btype.func(*net/http.Request, *"".RouteMatch) bool��type.func(*"".schemeMatcher, *net/http.Request, *"".RouteMatch) bool�2"".(*schemeMatcher).Match�2"".(*schemeMatcher).Match�:go.string."mux.schemeMatcher"PDmux.schemeMatcher :go.string."mux.schemeMatcher"��go.string."func(mux.schemeMatcher, *http.Request, *mux.RouteMatch) bool"��<func(mux.schemeMatcher, *http.Request, *mux.RouteMatch) bool �go.string."func(mux.schemeMatcher, *http.Request, *mux.RouteMatch) bool"��type.func("".schemeMatcher, *net/http.Request, *"".RouteMatch) bool������3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(mux.schemeMatcher, *http.Request, *mux.RouteMatch) bool"p�go.weak.type.*func("".schemeMatcher, *net/http.Request, *"".RouteMatch) bool�"runtime.zerovalue���type.func("".schemeMatcher, *net/http.Request, *"".RouteMatch) bool���type.func("".schemeMatcher, *net/http.Request, *"".RouteMatch) bool�*type."".schemeMatcher�,type.*net/http.Request�&type.*"".RouteMatch�type.bool�*type."".schemeMatcher����) � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P:go.string."mux.schemeMatcher"p,type.*"".schemeMatcher�"runtime.zerovalue�type.string`�*type."".schemeMatcher�2go.string."schemeMatcher"�"go.importpath."".��*type."".schemeMatcher�"go.string."Match"�btype.func(*net/http.Request, *"".RouteMatch) bool��type.func("".schemeMatcher, *net/http.Request, *"".RouteMatch) bool�2"".(*schemeMatcher).Match�,"".schemeMatcher.Match�,go.string."*[8]string"@6 +*[8]string ,go.string."*[8]string"�type.*[8]string����o6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."*[8]string"p0go.weak.type.**[8]string�"runtime.zerovalue�type.[8]string�&go.string."runtime"00runtime &go.string."runtime"�,go.importpath.runtime.  &go.string."runtime"�&go.string."net/url"00net/url &go.string."net/url"�,go.importpath.net/url.  &go.string."net/url"�$go.string."errors"0.errors $go.string."errors"�*go.importpath.errors.  $go.string."errors"�(go.string."net/http"@2net/http (go.string."net/http"�.go.importpath.net/http.  (go.string."net/http"�&go.string."strconv"00strconv &go.string."strconv"�,go.importpath.strconv.  &go.string."strconv"�&go.string."strings"00strings &go.string."strings"�,go.importpath.strings.  &go.string."strings"�go.string."fmt"0(fmt go.string."fmt"�$go.importpath.fmt.  go.string."fmt"��go.string."github.com/fsouza/go-dockerclient/external/github.com/gorilla/context"��Ejackfan.us.kg/fsouza/go-dockerclient/external/github.com/gorilla/context �go.string."github.com/fsouza/go-dockerclient/external/github.com/gorilla/context"��go.importpath.github.com/fsouza/go-dockerclient/external/github.com/gorilla/context. E �go.string."github.com/fsouza/go-dockerclient/external/github.com/gorilla/context"�*go.importpath.regexp.  $go.string."regexp"�"go.string."bytes"0,bytes "go.string."bytes"�(go.importpath.bytes.  "go.string."bytes"�&go.importpath.path.  go.string."path"�6"".parentRoute.buildVars·f0"".parentRoute.buildVars�@"".parentRoute.getNamedRoutes·f:"".parentRoute.getNamedRoutes�@"".parentRoute.getRegexpGroup·f:"".parentRoute.getRegexpGroup�.type..hash.[8]string·f(type..hash.[8]string�$runtime.strhash·fruntime.strhash�*type..eq.[8]string·f$type..eq.[8]string�&"".matcher.Match·f "".matcher.Match�4"".(*MatcherFunc).Match·f."".(*MatcherFunc).Match�(runtime.panicwrap·f"runtime.panicwrap�:type..hash.[1]interface {}·f4type..hash.[1]interface {}�.runtime.nilinterhash·f(runtime.nilinterhash�6type..eq.[1]interface {}·f0type..eq.[1]interface {}�$runtime.efaceeq·fruntime.efaceeq�:type..hash.[3]interface {}·f4type..hash.[3]interface {}�6type..eq.[3]interface {}·f0type..eq.[3]interface {}�:type..hash.[2]interface {}·f4type..hash.[2]interface {}�6type..eq.[2]interface {}·f0type..eq.[2]interface {}�8"".(*headerMatcher).Match·f2"".(*headerMatcher).Match�B"".(*headerRegexMatcher).Match·f<"".(*headerRegexMatcher).Match�8"".(*methodMatcher).Match·f2"".(*methodMatcher).Match�8"".(*schemeMatcher).Match·f2"".(*schemeMatcher).Match�"runtime.zerovalue0��go13ld \ No newline at end of file diff --git a/Godeps/_workspace/pkg/darwin_amd64/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user.a b/Godeps/_workspace/pkg/darwin_amd64/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user.a new file mode 100644 index 0000000000000..d0d368bcc7eea Binary files /dev/null and b/Godeps/_workspace/pkg/darwin_amd64/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user.a differ diff --git a/Godeps/_workspace/pkg/darwin_amd64/github.com/fsouza/go-dockerclient/testing.a b/Godeps/_workspace/pkg/darwin_amd64/github.com/fsouza/go-dockerclient/testing.a new file mode 100644 index 0000000000000..cc08ea8a564fd --- /dev/null +++ b/Godeps/_workspace/pkg/darwin_amd64/github.com/fsouza/go-dockerclient/testing.a @@ -0,0 +1,1236 @@ +! +__.PKGDEF 0 0 0 644 123084 ` +go object darwin amd64 go1.4.2 X:precisestack + +$$ +package testing + import net "net" + import rand "crypto/rand" + import docker "github.com/fsouza/go-dockerclient" + import sync "sync" + import runtime "runtime" + import time "time" + import mux "github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux" + import errors "errors" + import http "net/http" + import rand "math/rand" + import strconv "strconv" + import strings "strings" + import stdcopy "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy" + import fmt "fmt" + import regexp "regexp" + import tar "archive/tar" + import json "encoding/json" + type @"time".zone struct { @"time".name string; @"time".offset int; @"time".isDST bool } + type @"time".zoneTrans struct { @"time".when int64; @"time".index uint8; @"time".isstd bool; @"time".isutc bool } + type @"time".Location struct { @"time".name string; @"time".zone []@"time".zone; @"time".tx []@"time".zoneTrans; @"time".cacheStart int64; @"time".cacheEnd int64; @"time".cacheZone *@"time".zone } + func (@"time".l·2 *@"time".Location "esc:0x0") String () (? string) + func (@"time".l·2 *@"time".Location "esc:0x0") @"time".firstZoneUsed () (? bool) + func (@"time".l·2 *@"time".Location "esc:0x2") @"time".get () (? *@"time".Location) + func (@"time".l·6 *@"time".Location "esc:0x1") @"time".lookup (@"time".sec·7 int64) (@"time".name·1 string, @"time".offset·2 int, @"time".isDST·3 bool, @"time".start·4 int64, @"time".end·5 int64) + func (@"time".l·2 *@"time".Location "esc:0x0") @"time".lookupFirstZone () (? int) + func (@"time".l·4 *@"time".Location "esc:0x0") @"time".lookupName (@"time".name·5 string "esc:0x0", @"time".unix·6 int64) (@"time".offset·1 int, @"time".isDST·2 bool, @"time".ok·3 bool) + type @"time".Duration int64 + func (@"time".d·2 @"time".Duration) Hours () (? float64) { var @"time".hour·3 @"time".Duration; ; @"time".hour·3 = @"time".d·2 / @"time".Duration(0x34630B8A000); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0x34630B8A000); return float64(@"time".hour·3) + float64(@"time".nsec·4) * 0x9C5FFF26ED75Fp-93 } + func (@"time".d·2 @"time".Duration) Minutes () (? float64) { var @"time".min·3 @"time".Duration; ; @"time".min·3 = @"time".d·2 / @"time".Duration(0xDF8475800); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0xDF8475800); return float64(@"time".min·3) + float64(@"time".nsec·4) * 0x9299FF347E9E9p-87 } + func (@"time".d·2 @"time".Duration) Nanoseconds () (? int64) { return int64(@"time".d·2) } + func (@"time".d·2 @"time".Duration) Seconds () (? float64) { var @"time".sec·3 @"time".Duration; ; @"time".sec·3 = @"time".d·2 / @"time".Duration(0x3B9ACA00); var @"time".nsec·4 @"time".Duration; ; @"time".nsec·4 = @"time".d·2 % @"time".Duration(0x3B9ACA00); return float64(@"time".sec·3) + float64(@"time".nsec·4) * 0x112E0BE826D695p-82 } + func (@"time".d·2 @"time".Duration) String () (? string) + type @"time".Month int + func (@"time".m·2 @"time".Month) String () (? string) { return @"time".months[@"time".m·2 - @"time".Month(0x1)] } + type @"time".Weekday int + func (@"time".d·2 @"time".Weekday) String () (? string) { return @"time".days[@"time".d·2] } + type @"time".Time struct { @"time".sec int64; @"time".nsec int32; @"time".loc *@"time".Location } + func (@"time".t·2 @"time".Time "esc:0x2") Add (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x2") AddDate (@"time".years·3 int, @"time".months·4 int, @"time".days·5 int) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") After (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec > @"time".u·3.@"time".sec || @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec > @"time".u·3.@"time".nsec } + func (@"time".t·2 @"time".Time "esc:0x0") Before (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec < @"time".u·3.@"time".sec || @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec < @"time".u·3.@"time".nsec } + func (@"time".t·4 @"time".Time "esc:0x0") Clock () (@"time".hour·1 int, @"time".min·2 int, @"time".sec·3 int) + func (@"time".t·4 @"time".Time "esc:0x0") Date () (@"time".year·1 int, @"time".month·2 @"time".Month, @"time".day·3 int) + func (@"time".t·2 @"time".Time "esc:0x0") Day () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") Equal (@"time".u·3 @"time".Time "esc:0x0") (? bool) { return @"time".t·2.@"time".sec == @"time".u·3.@"time".sec && @"time".t·2.@"time".nsec == @"time".u·3.@"time".nsec } + func (@"time".t·2 @"time".Time "esc:0x0") Format (@"time".layout·3 string "esc:0x0") (? string) + func (@"time".t·2 *@"time".Time "esc:0x0") GobDecode (@"time".data·3 []byte "esc:0x0") (? error) + func (@"time".t·3 @"time".Time "esc:0x0") GobEncode () (? []byte, ? error) + func (@"time".t·2 @"time".Time "esc:0x0") Hour () (? int) + func (@"time".t·3 @"time".Time "esc:0x0") ISOWeek () (@"time".year·1 int, @"time".week·2 int) + func (@"time".t·2 @"time".Time "esc:0x2") In (@"time".loc·3 *@"time".Location "esc:0x2") (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") IsZero () (? bool) { return @"time".t·2.@"time".sec == 0x0 && @"time".t·2.@"time".nsec == 0x0 } + func (@"time".t·2 @"time".Time "esc:0x2") Local () (? @"time".Time) { @"time".t·2.@"time".loc = @"time".Local; return @"time".t·2 } + func (@"time".t·2 @"time".Time "esc:0x2") Location () (? *@"time".Location) { var @"time".l·3 *@"time".Location; ; @"time".l·3 = @"time".t·2.@"time".loc; if @"time".l·3 == nil { @"time".l·3 = @"time".UTC }; return @"time".l·3 } + func (@"time".t·3 @"time".Time "esc:0x0") MarshalBinary () (? []byte, ? error) + func (@"time".t·3 @"time".Time "esc:0x0") MarshalJSON () (? []byte, ? error) + func (@"time".t·3 @"time".Time "esc:0x0") MarshalText () (? []byte, ? error) + func (@"time".t·2 @"time".Time "esc:0x0") Minute () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") Month () (? @"time".Month) + func (@"time".t·2 @"time".Time "esc:0x0") Nanosecond () (? int) { return int(@"time".t·2.@"time".nsec) } + func (@"time".t·2 @"time".Time "esc:0x2") Round (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x0") Second () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") String () (? string) + func (@"time".t·2 @"time".Time "esc:0x0") Sub (@"time".u·3 @"time".Time "esc:0x0") (? @"time".Duration) + func (@"time".t·2 @"time".Time "esc:0x2") Truncate (@"time".d·3 @"time".Duration) (? @"time".Time) + func (@"time".t·2 @"time".Time "esc:0x2") UTC () (? @"time".Time) { @"time".t·2.@"time".loc = @"time".UTC; return @"time".t·2 } + func (@"time".t·2 @"time".Time "esc:0x0") Unix () (? int64) { return @"time".t·2.@"time".sec + -0xE7791F700 } + func (@"time".t·2 @"time".Time "esc:0x0") UnixNano () (? int64) { return (@"time".t·2.@"time".sec + -0xE7791F700) * 0x3B9ACA00 + int64(@"time".t·2.@"time".nsec) } + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalBinary (@"time".data·3 []byte "esc:0x0") (? error) + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalJSON (@"time".data·3 []byte "esc:0x0") (@"time".err·1 error) + func (@"time".t·2 *@"time".Time "esc:0x0") UnmarshalText (@"time".data·3 []byte "esc:0x0") (@"time".err·1 error) + func (@"time".t·2 @"time".Time "esc:0x0") Weekday () (? @"time".Weekday) + func (@"time".t·2 @"time".Time "esc:0x0") Year () (? int) + func (@"time".t·2 @"time".Time "esc:0x0") YearDay () (? int) + func (@"time".t·3 @"time".Time "esc:0x0") Zone () (@"time".name·1 string, @"time".offset·2 int) + func (@"time".t·2 @"time".Time "esc:0x0") @"time".abs () (? uint64) + func (@"time".t·5 @"time".Time "esc:0x0") @"time".date (@"time".full·6 bool) (@"time".year·1 int, @"time".month·2 @"time".Month, @"time".day·3 int, @"time".yday·4 int) + func (@"time".t·4 @"time".Time "esc:0x1") @"time".locabs () (@"time".name·1 string, @"time".offset·2 int, @"time".abs·3 uint64) + type @"github.com/fsouza/go-dockerclient".Port string + func (@"github.com/fsouza/go-dockerclient".p·2 @"github.com/fsouza/go-dockerclient".Port "esc:0x0") Port () (? string) + func (@"github.com/fsouza/go-dockerclient".p·2 @"github.com/fsouza/go-dockerclient".Port "esc:0x0") Proto () (? string) + type @"github.com/fsouza/go-dockerclient".Config struct { Hostname string "json:\"Hostname,omitempty\" yaml:\"Hostname,omitempty\""; Domainname string "json:\"Domainname,omitempty\" yaml:\"Domainname,omitempty\""; User string "json:\"User,omitempty\" yaml:\"User,omitempty\""; Memory int64 "json:\"Memory,omitempty\" yaml:\"Memory,omitempty\""; MemorySwap int64 "json:\"MemorySwap,omitempty\" yaml:\"MemorySwap,omitempty\""; CPUShares int64 "json:\"CpuShares,omitempty\" yaml:\"CpuShares,omitempty\""; CPUSet string "json:\"Cpuset,omitempty\" yaml:\"Cpuset,omitempty\""; AttachStdin bool "json:\"AttachStdin,omitempty\" yaml:\"AttachStdin,omitempty\""; AttachStdout bool "json:\"AttachStdout,omitempty\" yaml:\"AttachStdout,omitempty\""; AttachStderr bool "json:\"AttachStderr,omitempty\" yaml:\"AttachStderr,omitempty\""; PortSpecs []string "json:\"PortSpecs,omitempty\" yaml:\"PortSpecs,omitempty\""; ExposedPorts map[@"github.com/fsouza/go-dockerclient".Port]struct {} "json:\"ExposedPorts,omitempty\" yaml:\"ExposedPorts,omitempty\""; Tty bool "json:\"Tty,omitempty\" yaml:\"Tty,omitempty\""; OpenStdin bool "json:\"OpenStdin,omitempty\" yaml:\"OpenStdin,omitempty\""; StdinOnce bool "json:\"StdinOnce,omitempty\" yaml:\"StdinOnce,omitempty\""; Env []string "json:\"Env,omitempty\" yaml:\"Env,omitempty\""; Cmd []string "json:\"Cmd\" yaml:\"Cmd\""; DNS []string "json:\"Dns,omitempty\" yaml:\"Dns,omitempty\""; Image string "json:\"Image,omitempty\" yaml:\"Image,omitempty\""; Volumes map[string]struct {} "json:\"Volumes,omitempty\" yaml:\"Volumes,omitempty\""; VolumesFrom string "json:\"VolumesFrom,omitempty\" yaml:\"VolumesFrom,omitempty\""; WorkingDir string "json:\"WorkingDir,omitempty\" yaml:\"WorkingDir,omitempty\""; MacAddress string "json:\"MacAddress,omitempty\" yaml:\"MacAddress,omitempty\""; Entrypoint []string "json:\"Entrypoint\" yaml:\"Entrypoint\""; NetworkDisabled bool "json:\"NetworkDisabled,omitempty\" yaml:\"NetworkDisabled,omitempty\""; SecurityOpts []string "json:\"SecurityOpts,omitempty\" yaml:\"SecurityOpts,omitempty\""; OnBuild []string "json:\"OnBuild,omitempty\" yaml:\"OnBuild,omitempty\""; Labels map[string]string "json:\"Labels,omitempty\" yaml:\"Labels,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".State struct { Running bool "json:\"Running,omitempty\" yaml:\"Running,omitempty\""; Paused bool "json:\"Paused,omitempty\" yaml:\"Paused,omitempty\""; Restarting bool "json:\"Restarting,omitempty\" yaml:\"Restarting,omitempty\""; OOMKilled bool "json:\"OOMKilled,omitempty\" yaml:\"OOMKilled,omitempty\""; Pid int "json:\"Pid,omitempty\" yaml:\"Pid,omitempty\""; ExitCode int "json:\"ExitCode,omitempty\" yaml:\"ExitCode,omitempty\""; Error string "json:\"Error,omitempty\" yaml:\"Error,omitempty\""; StartedAt @"time".Time "json:\"StartedAt,omitempty\" yaml:\"StartedAt,omitempty\""; FinishedAt @"time".Time "json:\"FinishedAt,omitempty\" yaml:\"FinishedAt,omitempty\"" } + func (@"github.com/fsouza/go-dockerclient".s·2 *@"github.com/fsouza/go-dockerclient".State "esc:0x0") String () (? string) + type @"github.com/fsouza/go-dockerclient".SwarmNode struct { ID string "json:\"ID,omitempty\" yaml:\"ID,omitempty\""; IP string "json:\"IP,omitempty\" yaml:\"IP,omitempty\""; Addr string "json:\"Addr,omitempty\" yaml:\"Addr,omitempty\""; Name string "json:\"Name,omitempty\" yaml:\"Name,omitempty\""; CPUs int64 "json:\"CPUs,omitempty\" yaml:\"CPUs,omitempty\""; Memory int64 "json:\"Memory,omitempty\" yaml:\"Memory,omitempty\""; Labels map[string]string "json:\"Labels,omitempty\" yaml:\"Labels,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".PortMapping map[string]string + type @"github.com/fsouza/go-dockerclient".PortBinding struct { HostIP string "json:\"HostIP,omitempty\" yaml:\"HostIP,omitempty\""; HostPort string "json:\"HostPort,omitempty\" yaml:\"HostPort,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".APIPort struct { PrivatePort int64 "json:\"PrivatePort,omitempty\" yaml:\"PrivatePort,omitempty\""; PublicPort int64 "json:\"PublicPort,omitempty\" yaml:\"PublicPort,omitempty\""; Type string "json:\"Type,omitempty\" yaml:\"Type,omitempty\""; IP string "json:\"IP,omitempty\" yaml:\"IP,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".NetworkSettings struct { IPAddress string "json:\"IPAddress,omitempty\" yaml:\"IPAddress,omitempty\""; IPPrefixLen int "json:\"IPPrefixLen,omitempty\" yaml:\"IPPrefixLen,omitempty\""; MacAddress string "json:\"MacAddress,omitempty\" yaml:\"MacAddress,omitempty\""; Gateway string "json:\"Gateway,omitempty\" yaml:\"Gateway,omitempty\""; Bridge string "json:\"Bridge,omitempty\" yaml:\"Bridge,omitempty\""; PortMapping map[string]@"github.com/fsouza/go-dockerclient".PortMapping "json:\"PortMapping,omitempty\" yaml:\"PortMapping,omitempty\""; Ports map[@"github.com/fsouza/go-dockerclient".Port][]@"github.com/fsouza/go-dockerclient".PortBinding "json:\"Ports,omitempty\" yaml:\"Ports,omitempty\""; NetworkID string "json:\"NetworkID,omitempty\" yaml:\"NetworkID,omitempty\""; EndpointID string "json:\"EndpointID,omitempty\" yaml:\"EndpointID,omitempty\""; SandboxKey string "json:\"SandboxKey,omitempty\" yaml:\"SandboxKey,omitempty\""; GlobalIPv6Address string "json:\"GlobalIPv6Address,omitempty\" yaml:\"GlobalIPv6Address,omitempty\""; GlobalIPv6PrefixLen int "json:\"GlobalIPv6PrefixLen,omitempty\" yaml:\"GlobalIPv6PrefixLen,omitempty\""; IPv6Gateway string "json:\"IPv6Gateway,omitempty\" yaml:\"IPv6Gateway,omitempty\""; LinkLocalIPv6Address string "json:\"LinkLocalIPv6Address,omitempty\" yaml:\"LinkLocalIPv6Address,omitempty\""; LinkLocalIPv6PrefixLen int "json:\"LinkLocalIPv6PrefixLen,omitempty\" yaml:\"LinkLocalIPv6PrefixLen,omitempty\""; SecondaryIPAddresses []string "json:\"SecondaryIPAddresses,omitempty\" yaml:\"SecondaryIPAddresses,omitempty\""; SecondaryIPv6Addresses []string "json:\"SecondaryIPv6Addresses,omitempty\" yaml:\"SecondaryIPv6Addresses,omitempty\"" } + func (@"github.com/fsouza/go-dockerclient".settings·2 *@"github.com/fsouza/go-dockerclient".NetworkSettings "esc:0x0") PortMappingAPI () (? []@"github.com/fsouza/go-dockerclient".APIPort) + type @"github.com/fsouza/go-dockerclient".KeyValuePair struct { Key string "json:\"Key,omitempty\" yaml:\"Key,omitempty\""; Value string "json:\"Value,omitempty\" yaml:\"Value,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".RestartPolicy struct { Name string "json:\"Name,omitempty\" yaml:\"Name,omitempty\""; MaximumRetryCount int "json:\"MaximumRetryCount,omitempty\" yaml:\"MaximumRetryCount,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".Device struct { PathOnHost string "json:\"PathOnHost,omitempty\" yaml:\"PathOnHost,omitempty\""; PathInContainer string "json:\"PathInContainer,omitempty\" yaml:\"PathInContainer,omitempty\""; CgroupPermissions string "json:\"CgroupPermissions,omitempty\" yaml:\"CgroupPermissions,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".LogConfig struct { Type string "json:\"Type,omitempty\" yaml:\"Type,omitempty\""; Config map[string]string "json:\"Config,omitempty\" yaml:\"Config,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".ULimit struct { Name string "json:\"Name,omitempty\" yaml:\"Name,omitempty\""; Soft int64 "json:\"Soft,omitempty\" yaml:\"Soft,omitempty\""; Hard int64 "json:\"Hard,omitempty\" yaml:\"Hard,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".HostConfig struct { Binds []string "json:\"Binds,omitempty\" yaml:\"Binds,omitempty\""; CapAdd []string "json:\"CapAdd,omitempty\" yaml:\"CapAdd,omitempty\""; CapDrop []string "json:\"CapDrop,omitempty\" yaml:\"CapDrop,omitempty\""; ContainerIDFile string "json:\"ContainerIDFile,omitempty\" yaml:\"ContainerIDFile,omitempty\""; LxcConf []@"github.com/fsouza/go-dockerclient".KeyValuePair "json:\"LxcConf,omitempty\" yaml:\"LxcConf,omitempty\""; Privileged bool "json:\"Privileged,omitempty\" yaml:\"Privileged,omitempty\""; PortBindings map[@"github.com/fsouza/go-dockerclient".Port][]@"github.com/fsouza/go-dockerclient".PortBinding "json:\"PortBindings,omitempty\" yaml:\"PortBindings,omitempty\""; Links []string "json:\"Links,omitempty\" yaml:\"Links,omitempty\""; PublishAllPorts bool "json:\"PublishAllPorts,omitempty\" yaml:\"PublishAllPorts,omitempty\""; DNS []string "json:\"Dns,omitempty\" yaml:\"Dns,omitempty\""; DNSSearch []string "json:\"DnsSearch,omitempty\" yaml:\"DnsSearch,omitempty\""; ExtraHosts []string "json:\"ExtraHosts,omitempty\" yaml:\"ExtraHosts,omitempty\""; VolumesFrom []string "json:\"VolumesFrom,omitempty\" yaml:\"VolumesFrom,omitempty\""; NetworkMode string "json:\"NetworkMode,omitempty\" yaml:\"NetworkMode,omitempty\""; IpcMode string "json:\"IpcMode,omitempty\" yaml:\"IpcMode,omitempty\""; PidMode string "json:\"PidMode,omitempty\" yaml:\"PidMode,omitempty\""; UTSMode string "json:\"UTSMode,omitempty\" yaml:\"UTSMode,omitempty\""; RestartPolicy @"github.com/fsouza/go-dockerclient".RestartPolicy "json:\"RestartPolicy,omitempty\" yaml:\"RestartPolicy,omitempty\""; Devices []@"github.com/fsouza/go-dockerclient".Device "json:\"Devices,omitempty\" yaml:\"Devices,omitempty\""; LogConfig @"github.com/fsouza/go-dockerclient".LogConfig "json:\"LogConfig,omitempty\" yaml:\"LogConfig,omitempty\""; ReadonlyRootfs bool "json:\"ReadonlyRootfs,omitempty\" yaml:\"ReadonlyRootfs,omitempty\""; SecurityOpt []string "json:\"SecurityOpt,omitempty\" yaml:\"SecurityOpt,omitempty\""; CgroupParent string "json:\"CgroupParent,omitempty\" yaml:\"CgroupParent,omitempty\""; Memory int64 "json:\"Memory,omitempty\" yaml:\"Memory,omitempty\""; MemorySwap int64 "json:\"MemorySwap,omitempty\" yaml:\"MemorySwap,omitempty\""; CPUShares int64 "json:\"CpuShares,omitempty\" yaml:\"CpuShares,omitempty\""; CPUSet string "json:\"Cpuset,omitempty\" yaml:\"Cpuset,omitempty\""; CPUQuota int64 "json:\"CpuQuota,omitempty\" yaml:\"CpuQuota,omitempty\""; CPUPeriod int64 "json:\"CpuPeriod,omitempty\" yaml:\"CpuPeriod,omitempty\""; Ulimits []@"github.com/fsouza/go-dockerclient".ULimit "json:\"Ulimits,omitempty\" yaml:\"Ulimits,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".Container struct { ID string "json:\"Id\" yaml:\"Id\""; Created @"time".Time "json:\"Created,omitempty\" yaml:\"Created,omitempty\""; Path string "json:\"Path,omitempty\" yaml:\"Path,omitempty\""; Args []string "json:\"Args,omitempty\" yaml:\"Args,omitempty\""; Config *@"github.com/fsouza/go-dockerclient".Config "json:\"Config,omitempty\" yaml:\"Config,omitempty\""; State @"github.com/fsouza/go-dockerclient".State "json:\"State,omitempty\" yaml:\"State,omitempty\""; Image string "json:\"Image,omitempty\" yaml:\"Image,omitempty\""; Node *@"github.com/fsouza/go-dockerclient".SwarmNode "json:\"Node,omitempty\" yaml:\"Node,omitempty\""; NetworkSettings *@"github.com/fsouza/go-dockerclient".NetworkSettings "json:\"NetworkSettings,omitempty\" yaml:\"NetworkSettings,omitempty\""; SysInitPath string "json:\"SysInitPath,omitempty\" yaml:\"SysInitPath,omitempty\""; ResolvConfPath string "json:\"ResolvConfPath,omitempty\" yaml:\"ResolvConfPath,omitempty\""; HostnamePath string "json:\"HostnamePath,omitempty\" yaml:\"HostnamePath,omitempty\""; HostsPath string "json:\"HostsPath,omitempty\" yaml:\"HostsPath,omitempty\""; LogPath string "json:\"LogPath,omitempty\" yaml:\"LogPath,omitempty\""; Name string "json:\"Name,omitempty\" yaml:\"Name,omitempty\""; Driver string "json:\"Driver,omitempty\" yaml:\"Driver,omitempty\""; Volumes map[string]string "json:\"Volumes,omitempty\" yaml:\"Volumes,omitempty\""; VolumesRW map[string]bool "json:\"VolumesRW,omitempty\" yaml:\"VolumesRW,omitempty\""; HostConfig *@"github.com/fsouza/go-dockerclient".HostConfig "json:\"HostConfig,omitempty\" yaml:\"HostConfig,omitempty\""; ExecIDs []string "json:\"ExecIDs,omitempty\" yaml:\"ExecIDs,omitempty\""; RestartCount int "json:\"RestartCount,omitempty\" yaml:\"RestartCount,omitempty\""; AppArmorProfile string "json:\"AppArmorProfile,omitempty\" yaml:\"AppArmorProfile,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".ExecProcessConfig struct { Privileged bool "json:\"privileged,omitempty\" yaml:\"privileged,omitempty\""; User string "json:\"user,omitempty\" yaml:\"user,omitempty\""; Tty bool "json:\"tty,omitempty\" yaml:\"tty,omitempty\""; EntryPoint string "json:\"entrypoint,omitempty\" yaml:\"entrypoint,omitempty\""; Arguments []string "json:\"arguments,omitempty\" yaml:\"arguments,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".ExecInspect struct { ID string "json:\"ID,omitempty\" yaml:\"ID,omitempty\""; Running bool "json:\"Running,omitempty\" yaml:\"Running,omitempty\""; ExitCode int "json:\"ExitCode,omitempty\" yaml:\"ExitCode,omitempty\""; OpenStdin bool "json:\"OpenStdin,omitempty\" yaml:\"OpenStdin,omitempty\""; OpenStderr bool "json:\"OpenStderr,omitempty\" yaml:\"OpenStderr,omitempty\""; OpenStdout bool "json:\"OpenStdout,omitempty\" yaml:\"OpenStdout,omitempty\""; ProcessConfig @"github.com/fsouza/go-dockerclient".ExecProcessConfig "json:\"ProcessConfig,omitempty\" yaml:\"ProcessConfig,omitempty\""; Container @"github.com/fsouza/go-dockerclient".Container "json:\"Container,omitempty\" yaml:\"Container,omitempty\"" } + type @"sync".Mutex struct { @"sync".state int32; @"sync".sema uint32 } + func (@"sync".m·1 *@"sync".Mutex) Lock () + func (@"sync".m·1 *@"sync".Mutex) Unlock () + type @"sync".Locker interface { Lock(); Unlock() } + type @"sync".RWMutex struct { @"sync".w @"sync".Mutex; @"sync".writerSem uint32; @"sync".readerSem uint32; @"sync".readerCount int32; @"sync".readerWait int32 } + func (@"sync".rw·1 *@"sync".RWMutex) Lock () + func (@"sync".rw·1 *@"sync".RWMutex) RLock () + func (@"sync".rw·2 *@"sync".RWMutex "esc:0x2") RLocker () (? @"sync".Locker) { return (*@"sync".rlocker)(@"sync".rw·2) } + func (@"sync".rw·1 *@"sync".RWMutex) RUnlock () + func (@"sync".rw·1 *@"sync".RWMutex) Unlock () + type @"github.com/fsouza/go-dockerclient".Image struct { ID string "json:\"Id\" yaml:\"Id\""; Parent string "json:\"Parent,omitempty\" yaml:\"Parent,omitempty\""; Comment string "json:\"Comment,omitempty\" yaml:\"Comment,omitempty\""; Created @"time".Time "json:\"Created,omitempty\" yaml:\"Created,omitempty\""; Container string "json:\"Container,omitempty\" yaml:\"Container,omitempty\""; ContainerConfig @"github.com/fsouza/go-dockerclient".Config "json:\"ContainerConfig,omitempty\" yaml:\"ContainerConfig,omitempty\""; DockerVersion string "json:\"DockerVersion,omitempty\" yaml:\"DockerVersion,omitempty\""; Author string "json:\"Author,omitempty\" yaml:\"Author,omitempty\""; Config *@"github.com/fsouza/go-dockerclient".Config "json:\"Config,omitempty\" yaml:\"Config,omitempty\""; Architecture string "json:\"Architecture,omitempty\" yaml:\"Architecture,omitempty\""; Size int64 "json:\"Size,omitempty\" yaml:\"Size,omitempty\""; VirtualSize int64 "json:\"VirtualSize,omitempty\" yaml:\"VirtualSize,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".Endpoint struct { Name string "json:\"name\""; ID string "json:\"id\""; Network string "json:\"network\"" } + type @"github.com/fsouza/go-dockerclient".Network struct { Name string "json:\"name\""; ID string "json:\"id\""; Type string "json:\"type\""; Endpoints []*@"github.com/fsouza/go-dockerclient".Endpoint "json:\"endpoints\"" } + type @"net".Addr interface { Network() (? string); String() (? string) } + type @"net".Conn interface { Close() (? error); LocalAddr() (? @"net".Addr); Read(@"net".b []byte) (@"net".n int, @"net".err error); RemoteAddr() (? @"net".Addr); SetDeadline(@"net".t @"time".Time) (? error); SetReadDeadline(@"net".t @"time".Time) (? error); SetWriteDeadline(@"net".t @"time".Time) (? error); Write(@"net".b []byte) (@"net".n int, @"net".err error) } + type @"net".Listener interface { Accept() (@"net".c @"net".Conn, @"net".err error); Addr() (? @"net".Addr); Close() (? error) } + import io "io" // indirect + type @"io".Writer interface { Write(@"io".p []byte) (@"io".n int, @"io".err error) } + type @"net/http".keyValues struct { @"net/http".key string; @"net/http".values []string } + type @"net/http".headerSorter struct { @"net/http".kvs []@"net/http".keyValues } + func (@"net/http".s·2 *@"net/http".headerSorter "esc:0x0") Len () (? int) { return len(@"net/http".s·2.@"net/http".kvs) } + func (@"net/http".s·2 *@"net/http".headerSorter "esc:0x0") Less (@"net/http".i·3 int, @"net/http".j·4 int) (? bool) { return @"net/http".s·2.@"net/http".kvs[@"net/http".i·3].@"net/http".key < @"net/http".s·2.@"net/http".kvs[@"net/http".j·4].@"net/http".key } + func (@"net/http".s·1 *@"net/http".headerSorter "esc:0x0") Swap (@"net/http".i·2 int, @"net/http".j·3 int) { @"net/http".s·1.@"net/http".kvs[@"net/http".i·2], @"net/http".s·1.@"net/http".kvs[@"net/http".j·3] = @"net/http".s·1.@"net/http".kvs[@"net/http".j·3], @"net/http".s·1.@"net/http".kvs[@"net/http".i·2] } + type @"net/http".Header map[string][]string + func (@"net/http".h·1 @"net/http".Header "esc:0x0") Add (@"net/http".key·2 string, @"net/http".value·3 string) + func (@"net/http".h·1 @"net/http".Header "esc:0x0") Del (@"net/http".key·2 string "esc:0x0") + func (@"net/http".h·2 @"net/http".Header "esc:0x0") Get (@"net/http".key·3 string "esc:0x0") (? string) + func (@"net/http".h·1 @"net/http".Header "esc:0x0") Set (@"net/http".key·2 string, @"net/http".value·3 string) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") Write (@"net/http".w·3 @"io".Writer) (? error) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") WriteSubset (@"net/http".w·3 @"io".Writer, @"net/http".exclude·4 map[string]bool "esc:0x0") (? error) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") @"net/http".clone () (? @"net/http".Header) + func (@"net/http".h·2 @"net/http".Header "esc:0x0") @"net/http".get (@"net/http".key·3 string "esc:0x0") (? string) { var @"net/http".v·4 []string; ; @"net/http".v·4 = @"net/http".h·2[@"net/http".key·3]; if len(@"net/http".v·4) > 0x0 { return @"net/http".v·4[0x0] }; return "" } + func (@"net/http".h·3 @"net/http".Header "esc:0x0") @"net/http".sortedKeyValues (@"net/http".exclude·4 map[string]bool "esc:0x0") (@"net/http".kvs·1 []@"net/http".keyValues, @"net/http".hs·2 *@"net/http".headerSorter) + type @"net/http".ResponseWriter interface { Header() (? @"net/http".Header); Write(? []byte) (? int, ? error); WriteHeader(? int) } + import url "net/url" // indirect + type @"net/url".Userinfo struct { @"net/url".username string; @"net/url".password string; @"net/url".passwordSet bool } + func (@"net/url".u·3 *@"net/url".Userinfo "esc:0x1") Password () (? string, ? bool) { if @"net/url".u·3.@"net/url".passwordSet { return @"net/url".u·3.@"net/url".password, true }; return "", false } + func (@"net/url".u·2 *@"net/url".Userinfo "esc:0x1") String () (? string) + func (@"net/url".u·2 *@"net/url".Userinfo "esc:0x1") Username () (? string) { return @"net/url".u·2.@"net/url".username } + type @"net/url".Values map[string][]string + func (@"net/url".v·1 @"net/url".Values "esc:0x0") Add (@"net/url".key·2 string, @"net/url".value·3 string) { @"net/url".v·1[@"net/url".key·2] = append(@"net/url".v·1[@"net/url".key·2], @"net/url".value·3) } + func (@"net/url".v·1 @"net/url".Values "esc:0x0") Del (@"net/url".key·2 string "esc:0x0") { delete(@"net/url".v·1, @"net/url".key·2) } + func (@"net/url".v·2 @"net/url".Values "esc:0x0") Encode () (? string) + func (@"net/url".v·2 @"net/url".Values "esc:0x0") Get (@"net/url".key·3 string "esc:0x0") (? string) { if @"net/url".v·2 == nil { return "" }; var @"net/url".vs·4 []string; ; var @"net/url".ok·5 bool; ; @"net/url".vs·4, @"net/url".ok·5 = @"net/url".v·2[@"net/url".key·3]; if !@"net/url".ok·5 || len(@"net/url".vs·4) == 0x0 { return "" }; return @"net/url".vs·4[0x0] } + func (@"net/url".v·1 @"net/url".Values "esc:0x0") Set (@"net/url".key·2 string, @"net/url".value·3 string) { @"net/url".v·1[@"net/url".key·2] = ([]string{ 0x0:@"net/url".value·3 }) } + type @"net/url".URL struct { Scheme string; Opaque string; User *@"net/url".Userinfo; Host string; Path string; RawQuery string; Fragment string } + func (@"net/url".u·2 *@"net/url".URL "esc:0x0") IsAbs () (? bool) { return @"net/url".u·2.Scheme != "" } + func (@"net/url".u·3 *@"net/url".URL "esc:0x2") Parse (@"net/url".ref·4 string) (? *@"net/url".URL, ? error) + func (@"net/url".u·2 *@"net/url".URL) Query () (? @"net/url".Values) + func (@"net/url".u·2 *@"net/url".URL "esc:0x1") RequestURI () (? string) + func (@"net/url".u·2 *@"net/url".URL "esc:0x2") ResolveReference (@"net/url".ref·3 *@"net/url".URL "esc:0x2") (? *@"net/url".URL) + func (@"net/url".u·2 *@"net/url".URL "esc:0x0") String () (? string) + type @"io".ReadCloser interface { Close() (? error); Read(@"io".p []byte) (@"io".n int, @"io".err error) } + import multipart "mime/multipart" // indirect + import textproto "net/textproto" // indirect + type @"net/textproto".MIMEHeader map[string][]string + func (@"net/textproto".h·1 @"net/textproto".MIMEHeader "esc:0x0") Add (@"net/textproto".key·2 string, @"net/textproto".value·3 string) + func (@"net/textproto".h·1 @"net/textproto".MIMEHeader "esc:0x0") Del (@"net/textproto".key·2 string "esc:0x0") + func (@"net/textproto".h·2 @"net/textproto".MIMEHeader "esc:0x0") Get (@"net/textproto".key·3 string "esc:0x0") (? string) + func (@"net/textproto".h·1 @"net/textproto".MIMEHeader "esc:0x0") Set (@"net/textproto".key·2 string, @"net/textproto".value·3 string) + type @"mime/multipart".File interface { Close() (? error); Read(@"io".p []byte) (@"io".n int, @"io".err error); ReadAt(@"io".p []byte, @"io".off int64) (@"io".n int, @"io".err error); Seek(@"io".offset int64, @"io".whence int) (? int64, ? error) } + type @"mime/multipart".FileHeader struct { Filename string; Header @"net/textproto".MIMEHeader; @"mime/multipart".content []byte; @"mime/multipart".tmpfile string } + func (@"mime/multipart".fh·3 *@"mime/multipart".FileHeader) Open () (? @"mime/multipart".File, ? error) + type @"mime/multipart".Form struct { Value map[string][]string; File map[string][]*@"mime/multipart".FileHeader } + func (@"mime/multipart".f·2 *@"mime/multipart".Form "esc:0x0") RemoveAll () (? error) + import tls "crypto/tls" // indirect + import x509 "crypto/x509" // indirect + type @"crypto/x509".SignatureAlgorithm int + type @"crypto/x509".PublicKeyAlgorithm int + import big "math/big" // indirect + type @"math/big".Word uintptr + type @"math/big".divisor struct { @"math/big".bbb @"math/big".nat; @"math/big".nbits int; @"math/big".ndigits int } + type @"math/rand".Source interface { Int63() (? int64); Seed(@"math/rand".seed int64) } + type @"math/rand".Rand struct { @"math/rand".src @"math/rand".Source } + func (@"math/rand".r·2 *@"math/rand".Rand) ExpFloat64 () (? float64) + func (@"math/rand".r·2 *@"math/rand".Rand) Float32 () (? float32) + func (@"math/rand".r·2 *@"math/rand".Rand) Float64 () (? float64) + func (@"math/rand".r·2 *@"math/rand".Rand) Int () (? int) + func (@"math/rand".r·2 *@"math/rand".Rand) Int31 () (? int32) + func (@"math/rand".r·2 *@"math/rand".Rand) Int31n (@"math/rand".n·3 int32) (? int32) + func (@"math/rand".r·2 *@"math/rand".Rand) Int63 () (? int64) + func (@"math/rand".r·2 *@"math/rand".Rand) Int63n (@"math/rand".n·3 int64) (? int64) + func (@"math/rand".r·2 *@"math/rand".Rand) Intn (@"math/rand".n·3 int) (? int) + func (@"math/rand".r·2 *@"math/rand".Rand) NormFloat64 () (? float64) + func (@"math/rand".r·2 *@"math/rand".Rand) Perm (@"math/rand".n·3 int) (? []int) + func (@"math/rand".r·1 *@"math/rand".Rand) Seed (@"math/rand".seed·2 int64) + func (@"math/rand".r·2 *@"math/rand".Rand) Uint32 () (? uint32) + type @"io".RuneScanner interface { ReadRune() (@"io".r rune, @"io".size int, @"io".err error); UnreadRune() (? error) } + type @"math/big".nat []@"math/big".Word + func (@"math/big".z·2 @"math/big".nat) @"math/big".add (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".and (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".andNot (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x0") @"math/big".bit (@"math/big".i·3 uint) (? uint) { var @"math/big".j·4 int; ; @"math/big".j·4 = int(@"math/big".i·3 / 0x40); if @"math/big".j·4 >= len(@"math/big".z·2) { return 0x0 }; return uint(@"math/big".z·2[@"math/big".j·4] >> (@"math/big".i·3 % 0x40) & @"math/big".Word(0x1)) } + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".bitLen () (? int) + func (@"math/big".z·2 @"math/big".nat "esc:0x0") @"math/big".bytes (@"math/big".buf·3 []byte "esc:0x0") (@"math/big".i·1 int) + func (@"math/big".z·1 @"math/big".nat "esc:0x0") @"math/big".clear () + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".cmp (@"math/big".y·3 @"math/big".nat "esc:0x0") (@"math/big".r·1 int) + func (@"math/big".q·1 @"math/big".nat) @"math/big".convertWords (@"math/big".s·2 []byte "esc:0x0", @"math/big".charset·3 string "esc:0x0", @"math/big".b·4 @"math/big".Word, @"math/big".ndigits·5 int, @"math/big".bb·6 @"math/big".Word, @"math/big".table·7 []@"math/big".divisor "esc:0x0") + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".decimalString () (? string) + func (@"math/big".z·3 @"math/big".nat) @"math/big".div (@"math/big".z2·4 @"math/big".nat, @"math/big".u·5 @"math/big".nat, @"math/big".v·6 @"math/big".nat) (@"math/big".q·1 @"math/big".nat, @"math/big".r·2 @"math/big".nat) + func (@"math/big".z·3 @"math/big".nat "esc:0x2") @"math/big".divLarge (@"math/big".u·4 @"math/big".nat, @"math/big".uIn·5 @"math/big".nat, @"math/big".v·6 @"math/big".nat) (@"math/big".q·1 @"math/big".nat, @"math/big".r·2 @"math/big".nat) + func (@"math/big".z·3 @"math/big".nat) @"math/big".divW (@"math/big".x·4 @"math/big".nat, @"math/big".y·5 @"math/big".Word) (@"math/big".q·1 @"math/big".nat, @"math/big".r·2 @"math/big".Word) + func (@"math/big".z·2 @"math/big".nat) @"math/big".expNN (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat "esc:0x0", @"math/big".m·5 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".expNNWindowed (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat "esc:0x0", @"math/big".m·5 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".expWW (@"math/big".x·3 @"math/big".Word, @"math/big".y·4 @"math/big".Word) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".make (@"math/big".n·3 int) (? @"math/big".nat) + func (@"math/big".x·2 @"math/big".nat) @"math/big".modW (@"math/big".d·3 @"math/big".Word) (@"math/big".r·1 @"math/big".Word) + func (@"math/big".z·2 @"math/big".nat) @"math/big".mul (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".mulAddWW (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".Word, @"math/big".r·5 @"math/big".Word) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".mulRange (@"math/big".a·3 uint64, @"math/big".b·4 uint64) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".norm () (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".or (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".n·2 @"math/big".nat) @"math/big".probablyPrime (@"math/big".reps·3 int) (? bool) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".random (@"math/big".rand·3 *@"math/rand".Rand, @"math/big".limit·4 @"math/big".nat "esc:0x0", @"math/big".n·5 int) (? @"math/big".nat) + func (@"math/big".z·4 @"math/big".nat) @"math/big".scan (@"math/big".r·5 @"io".RuneScanner, @"math/big".base·6 int) (? @"math/big".nat, ? int, ? error) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".set (@"math/big".x·3 @"math/big".nat "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setBit (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".i·4 uint, @"math/big".b·5 uint) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setBytes (@"math/big".buf·3 []byte "esc:0x0") (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setUint64 (@"math/big".x·3 uint64) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".setWord (@"math/big".x·3 @"math/big".Word) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".shl (@"math/big".x·3 @"math/big".nat, @"math/big".s·4 uint) (? @"math/big".nat) + func (@"math/big".z·2 @"math/big".nat) @"math/big".shr (@"math/big".x·3 @"math/big".nat, @"math/big".s·4 uint) (? @"math/big".nat) + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".string (@"math/big".charset·3 string "esc:0x0") (? string) + func (@"math/big".z·2 @"math/big".nat) @"math/big".sub (@"math/big".x·3 @"math/big".nat, @"math/big".y·4 @"math/big".nat) (? @"math/big".nat) + func (@"math/big".x·2 @"math/big".nat "esc:0x0") @"math/big".trailingZeroBits () (? uint) + func (@"math/big".z·2 @"math/big".nat "esc:0x2") @"math/big".xor (@"math/big".x·3 @"math/big".nat "esc:0x0", @"math/big".y·4 @"math/big".nat "esc:0x0") (? @"math/big".nat) + type @"fmt".State interface { Flag(@"fmt".c int) (? bool); Precision() (@"fmt".prec int, @"fmt".ok bool); Width() (@"fmt".wid int, @"fmt".ok bool); Write(@"fmt".b []byte) (@"fmt".ret int, @"fmt".err error) } + type @"fmt".ScanState interface { Read(@"fmt".buf []byte) (@"fmt".n int, @"fmt".err error); ReadRune() (@"fmt".r rune, @"fmt".size int, @"fmt".err error); SkipSpace(); Token(@"fmt".skipSpace bool, @"fmt".f func(? rune) (? bool)) (@"fmt".token []byte, @"fmt".err error); UnreadRune() (? error); Width() (@"fmt".wid int, @"fmt".ok bool) } + type @"math/big".Int struct { @"math/big".neg bool; @"math/big".abs @"math/big".nat } + func (@"math/big".z·2 *@"math/big".Int) Abs (@"math/big".x·3 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Add (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) And (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) AndNot (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Binomial (@"math/big".n·3 int64, @"math/big".k·4 int64) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int) Bit (@"math/big".i·3 int) (? uint) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") BitLen () (? int) + func (@"math/big".x·2 *@"math/big".Int "esc:0x1") Bits () (? []@"math/big".Word) { return @"math/big".x·2.@"math/big".abs } + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Bytes () (? []byte) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Cmp (@"math/big".y·3 *@"math/big".Int "esc:0x0") (@"math/big".r·1 int) + func (@"math/big".z·2 *@"math/big".Int) Div (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int) DivMod (@"math/big".x·4 *@"math/big".Int, @"math/big".y·5 *@"math/big".Int, @"math/big".m·6 *@"math/big".Int) (? *@"math/big".Int, ? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Exp (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int "esc:0x0", @"math/big".m·5 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".x·1 *@"math/big".Int "esc:0x0") Format (@"math/big".s·2 @"fmt".State, @"math/big".ch·3 rune) + func (@"math/big".z·2 *@"math/big".Int) GCD (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int, @"math/big".a·5 *@"math/big".Int, @"math/big".b·6 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) GobDecode (@"math/big".buf·3 []byte "esc:0x0") (? error) + func (@"math/big".x·3 *@"math/big".Int "esc:0x0") GobEncode () (? []byte, ? error) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Int64 () (? int64) + func (@"math/big".z·2 *@"math/big".Int) Lsh (@"math/big".x·3 *@"math/big".Int, @"math/big".n·4 uint) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int "esc:0x0") MarshalJSON () (? []byte, ? error) + func (@"math/big".z·3 *@"math/big".Int "esc:0x0") MarshalText () (@"math/big".text·1 []byte, @"math/big".err·2 error) + func (@"math/big".z·2 *@"math/big".Int) Mod (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) ModInverse (@"math/big".g·3 *@"math/big".Int, @"math/big".n·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Mul (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) MulRange (@"math/big".a·3 int64, @"math/big".b·4 int64) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Neg (@"math/big".x·3 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Not (@"math/big".x·3 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Or (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int) ProbablyPrime (@"math/big".n·3 int) (? bool) + func (@"math/big".z·2 *@"math/big".Int) Quo (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int) QuoRem (@"math/big".x·4 *@"math/big".Int, @"math/big".y·5 *@"math/big".Int, @"math/big".r·6 *@"math/big".Int) (? *@"math/big".Int, ? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Rand (@"math/big".rnd·3 *@"math/rand".Rand, @"math/big".n·4 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Rem (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Rsh (@"math/big".x·3 *@"math/big".Int, @"math/big".n·4 uint) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) Scan (@"math/big".s·3 @"fmt".ScanState, @"math/big".ch·4 rune) (? error) + func (@"math/big".z·2 *@"math/big".Int) Set (@"math/big".x·3 *@"math/big".Int "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) SetBit (@"math/big".x·3 *@"math/big".Int, @"math/big".i·4 int, @"math/big".b·5 uint) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int "esc:0x2") SetBits (@"math/big".abs·3 []@"math/big".Word) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) SetBytes (@"math/big".buf·3 []byte "esc:0x0") (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) SetInt64 (@"math/big".x·3 int64) (? *@"math/big".Int) + func (@"math/big".z·3 *@"math/big".Int) SetString (@"math/big".s·4 string, @"math/big".base·5 int) (? *@"math/big".Int, ? bool) + func (@"math/big".z·2 *@"math/big".Int) SetUint64 (@"math/big".x·3 uint64) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Sign () (? int) { if len(@"math/big".x·2.@"math/big".abs) == 0x0 { return 0x0 }; if @"math/big".x·2.@"math/big".neg { return -0x1 }; return 0x1 } + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") String () (? string) + func (@"math/big".z·2 *@"math/big".Int) Sub (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".x·2 *@"math/big".Int "esc:0x0") Uint64 () (? uint64) + func (@"math/big".z·2 *@"math/big".Int) UnmarshalJSON (@"math/big".text·3 []byte) (? error) + func (@"math/big".z·2 *@"math/big".Int) UnmarshalText (@"math/big".text·3 []byte) (? error) + func (@"math/big".z·2 *@"math/big".Int) Xor (@"math/big".x·3 *@"math/big".Int, @"math/big".y·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·2 *@"math/big".Int) @"math/big".binaryGCD (@"math/big".a·3 *@"math/big".Int, @"math/big".b·4 *@"math/big".Int) (? *@"math/big".Int) + func (@"math/big".z·4 *@"math/big".Int) @"math/big".scan (@"math/big".r·5 @"io".RuneScanner, @"math/big".base·6 int) (? *@"math/big".Int, ? int, ? error) + import pkix "crypto/x509/pkix" // indirect + import asn1 "encoding/asn1" // indirect + type @"encoding/asn1".ObjectIdentifier []int + func (@"encoding/asn1".oi·2 @"encoding/asn1".ObjectIdentifier "esc:0x0") Equal (@"encoding/asn1".other·3 @"encoding/asn1".ObjectIdentifier "esc:0x0") (? bool) + func (@"encoding/asn1".oi·2 @"encoding/asn1".ObjectIdentifier "esc:0x0") String () (? string) + type @"crypto/x509/pkix".AttributeTypeAndValue struct { Type @"encoding/asn1".ObjectIdentifier; Value interface {} } + type @"crypto/x509/pkix".RelativeDistinguishedNameSET []@"crypto/x509/pkix".AttributeTypeAndValue + type @"crypto/x509/pkix".RDNSequence []@"crypto/x509/pkix".RelativeDistinguishedNameSET + type @"crypto/x509/pkix".Name struct { Country []string; Organization []string; OrganizationalUnit []string; Locality []string; Province []string; StreetAddress []string; PostalCode []string; SerialNumber string; CommonName string; Names []@"crypto/x509/pkix".AttributeTypeAndValue } + func (@"crypto/x509/pkix".n·1 *@"crypto/x509/pkix".Name) FillFromRDNSequence (@"crypto/x509/pkix".rdns·2 *@"crypto/x509/pkix".RDNSequence "esc:0x0") + func (@"crypto/x509/pkix".n·2 @"crypto/x509/pkix".Name) ToRDNSequence () (@"crypto/x509/pkix".ret·1 @"crypto/x509/pkix".RDNSequence) + type @"crypto/x509".KeyUsage int + type @"crypto/x509/pkix".Extension struct { Id @"encoding/asn1".ObjectIdentifier; Critical bool "asn1:\"optional\""; Value []byte } + type @"crypto/x509".ExtKeyUsage int + type @"net".IPMask []byte + func (@"net".m·3 @"net".IPMask "esc:0x0") Size () (@"net".ones·1 int, @"net".bits·2 int) + func (@"net".m·2 @"net".IPMask "esc:0x0") String () (? string) + type @"net".IP []byte + func (@"net".ip·2 @"net".IP "esc:0x0") DefaultMask () (? @"net".IPMask) + func (@"net".ip·2 @"net".IP "esc:0x0") Equal (@"net".x·3 @"net".IP "esc:0x0") (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsGlobalUnicast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsInterfaceLocalMulticast () (? bool) { return len(@"net".ip·2) == 0x10 && @"net".ip·2[0x0] == byte(0xFF) && @"net".ip·2[0x1] & byte(0xF) == byte(0x1) } + func (@"net".ip·2 @"net".IP "esc:0x0") IsLinkLocalMulticast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsLinkLocalUnicast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsLoopback () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsMulticast () (? bool) + func (@"net".ip·2 @"net".IP "esc:0x0") IsUnspecified () (? bool) + func (@"net".ip·3 @"net".IP "esc:0x0") MarshalText () (? []byte, ? error) + func (@"net".ip·2 @"net".IP "esc:0x0") Mask (@"net".mask·3 @"net".IPMask "esc:0x0") (? @"net".IP) + func (@"net".ip·2 @"net".IP "esc:0x0") String () (? string) + func (@"net".ip·2 @"net".IP "esc:0x2") To16 () (? @"net".IP) + func (@"net".ip·2 @"net".IP "esc:0x2") To4 () (? @"net".IP) + func (@"net".ip·2 *@"net".IP "esc:0x0") UnmarshalText (@"net".text·3 []byte "esc:0x0") (? error) + type @"encoding/asn1".RawContent []byte + type @"encoding/asn1".RawValue struct { Class int; Tag int; IsCompound bool; Bytes []byte; FullBytes []byte } + type @"crypto/x509/pkix".AlgorithmIdentifier struct { Algorithm @"encoding/asn1".ObjectIdentifier; Parameters @"encoding/asn1".RawValue "asn1:\"optional\"" } + type @"crypto/x509/pkix".RevokedCertificate struct { SerialNumber *@"math/big".Int; RevocationTime @"time".Time; Extensions []@"crypto/x509/pkix".Extension "asn1:\"optional\"" } + type @"crypto/x509/pkix".TBSCertificateList struct { Raw @"encoding/asn1".RawContent; Version int "asn1:\"optional,default:2\""; Signature @"crypto/x509/pkix".AlgorithmIdentifier; Issuer @"crypto/x509/pkix".RDNSequence; ThisUpdate @"time".Time; NextUpdate @"time".Time "asn1:\"optional\""; RevokedCertificates []@"crypto/x509/pkix".RevokedCertificate "asn1:\"optional\""; Extensions []@"crypto/x509/pkix".Extension "asn1:\"tag:0,optional,explicit\"" } + type @"encoding/asn1".BitString struct { Bytes []byte; BitLength int } + func (@"encoding/asn1".b·2 @"encoding/asn1".BitString "esc:0x0") At (@"encoding/asn1".i·3 int) (? int) { if @"encoding/asn1".i·3 < 0x0 || @"encoding/asn1".i·3 >= @"encoding/asn1".b·2.BitLength { return 0x0 }; var @"encoding/asn1".x·4 int; ; @"encoding/asn1".x·4 = @"encoding/asn1".i·3 / 0x8; var @"encoding/asn1".y·5 uint; ; @"encoding/asn1".y·5 = 0x7 - uint(@"encoding/asn1".i·3 % 0x8); return int(@"encoding/asn1".b·2.Bytes[@"encoding/asn1".x·4] >> @"encoding/asn1".y·5) & 0x1 } + func (@"encoding/asn1".b·2 @"encoding/asn1".BitString "esc:0x2") RightAlign () (? []byte) + type @"crypto/x509/pkix".CertificateList struct { TBSCertList @"crypto/x509/pkix".TBSCertificateList; SignatureAlgorithm @"crypto/x509/pkix".AlgorithmIdentifier; SignatureValue @"encoding/asn1".BitString } + func (@"crypto/x509/pkix".certList·2 *@"crypto/x509/pkix".CertificateList "esc:0x0") HasExpired (@"crypto/x509/pkix".now·3 @"time".Time "esc:0x0") (? bool) + type @"io".Reader interface { Read(@"io".p []byte) (@"io".n int, @"io".err error) } + type @"crypto/x509".CertPool struct { @"crypto/x509".bySubjectKeyId map[string][]int; @"crypto/x509".byName map[string][]int; @"crypto/x509".certs []*@"crypto/x509".Certificate } + func (@"crypto/x509".s·1 *@"crypto/x509".CertPool) AddCert (@"crypto/x509".cert·2 *@"crypto/x509".Certificate) + func (@"crypto/x509".s·2 *@"crypto/x509".CertPool) AppendCertsFromPEM (@"crypto/x509".pemCerts·3 []byte) (@"crypto/x509".ok·1 bool) + func (@"crypto/x509".s·2 *@"crypto/x509".CertPool "esc:0x0") Subjects () (@"crypto/x509".res·1 [][]byte) + func (@"crypto/x509".s·4 *@"crypto/x509".CertPool "esc:0x0") @"crypto/x509".findVerifiedParents (@"crypto/x509".cert·5 *@"crypto/x509".Certificate) (@"crypto/x509".parents·1 []int, @"crypto/x509".errCert·2 *@"crypto/x509".Certificate, @"crypto/x509".err·3 error) + type @"crypto/x509".VerifyOptions struct { DNSName string; Intermediates *@"crypto/x509".CertPool; Roots *@"crypto/x509".CertPool; CurrentTime @"time".Time; KeyUsages []@"crypto/x509".ExtKeyUsage } + type @"crypto/x509".Certificate struct { Raw []byte; RawTBSCertificate []byte; RawSubjectPublicKeyInfo []byte; RawSubject []byte; RawIssuer []byte; Signature []byte; SignatureAlgorithm @"crypto/x509".SignatureAlgorithm; PublicKeyAlgorithm @"crypto/x509".PublicKeyAlgorithm; PublicKey interface {}; Version int; SerialNumber *@"math/big".Int; Issuer @"crypto/x509/pkix".Name; Subject @"crypto/x509/pkix".Name; NotBefore @"time".Time; NotAfter @"time".Time; KeyUsage @"crypto/x509".KeyUsage; Extensions []@"crypto/x509/pkix".Extension; ExtraExtensions []@"crypto/x509/pkix".Extension; ExtKeyUsage []@"crypto/x509".ExtKeyUsage; UnknownExtKeyUsage []@"encoding/asn1".ObjectIdentifier; BasicConstraintsValid bool; IsCA bool; MaxPathLen int; MaxPathLenZero bool; SubjectKeyId []byte; AuthorityKeyId []byte; OCSPServer []string; IssuingCertificateURL []string; DNSNames []string; EmailAddresses []string; IPAddresses []@"net".IP; PermittedDNSDomainsCritical bool; PermittedDNSDomains []string; CRLDistributionPoints []string; PolicyIdentifiers []@"encoding/asn1".ObjectIdentifier } + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate) CheckCRLSignature (@"crypto/x509".crl·3 *@"crypto/x509/pkix".CertificateList) (@"crypto/x509".err·1 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate) CheckSignature (@"crypto/x509".algo·3 @"crypto/x509".SignatureAlgorithm, @"crypto/x509".signed·4 []byte, @"crypto/x509".signature·5 []byte) (@"crypto/x509".err·1 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate) CheckSignatureFrom (@"crypto/x509".parent·3 *@"crypto/x509".Certificate) (@"crypto/x509".err·1 error) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate) CreateCRL (@"crypto/x509".rand·4 @"io".Reader, @"crypto/x509".priv·5 interface {}, @"crypto/x509".revokedCerts·6 []@"crypto/x509/pkix".RevokedCertificate, @"crypto/x509".now·7 @"time".Time, @"crypto/x509".expiry·8 @"time".Time) (@"crypto/x509".crlBytes·1 []byte, @"crypto/x509".err·2 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate "esc:0x0") Equal (@"crypto/x509".other·3 *@"crypto/x509".Certificate "esc:0x0") (? bool) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate) Verify (@"crypto/x509".opts·4 @"crypto/x509".VerifyOptions "esc:0x4") (@"crypto/x509".chains·1 [][]*@"crypto/x509".Certificate, @"crypto/x509".err·2 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate "esc:0x2") VerifyHostname (@"crypto/x509".h·3 string "esc:0x2") (? error) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate) @"crypto/x509".buildChains (@"crypto/x509".cache·4 map[int][][]*@"crypto/x509".Certificate "esc:0x0", @"crypto/x509".currentChain·5 []*@"crypto/x509".Certificate "esc:0x0", @"crypto/x509".opts·6 *@"crypto/x509".VerifyOptions "esc:0x0") (@"crypto/x509".chains·1 [][]*@"crypto/x509".Certificate, @"crypto/x509".err·2 error) + func (@"crypto/x509".c·2 *@"crypto/x509".Certificate "esc:0x2") @"crypto/x509".isValid (@"crypto/x509".certType·3 int, @"crypto/x509".currentChain·4 []*@"crypto/x509".Certificate "esc:0x0", @"crypto/x509".opts·5 *@"crypto/x509".VerifyOptions "esc:0x0") (? error) + func (@"crypto/x509".c·3 *@"crypto/x509".Certificate "esc:0x0") @"crypto/x509".systemVerify (@"crypto/x509".opts·4 *@"crypto/x509".VerifyOptions "esc:0x0") (@"crypto/x509".chains·1 [][]*@"crypto/x509".Certificate, @"crypto/x509".err·2 error) { return nil, nil } + type @"crypto/tls".ConnectionState struct { Version uint16; HandshakeComplete bool; DidResume bool; CipherSuite uint16; NegotiatedProtocol string; NegotiatedProtocolIsMutual bool; ServerName string; PeerCertificates []*@"crypto/x509".Certificate; VerifiedChains [][]*@"crypto/x509".Certificate; TLSUnique []byte } + type @"net/http".Cookie struct { Name string; Value string; Path string; Domain string; Expires @"time".Time; RawExpires string; MaxAge int; Secure bool; HttpOnly bool; Raw string; Unparsed []string } + func (@"net/http".c·2 *@"net/http".Cookie) String () (? string) + import bufio "bufio" // indirect + type @"bufio".Reader struct { @"bufio".buf []byte; @"bufio".rd @"io".Reader; @"bufio".r int; @"bufio".w int; @"bufio".err error; @"bufio".lastByte int; @"bufio".lastRuneSize int } + func (@"bufio".b·2 *@"bufio".Reader "esc:0x0") Buffered () (? int) { return @"bufio".b·2.@"bufio".w - @"bufio".b·2.@"bufio".r } + func (@"bufio".b·3 *@"bufio".Reader) Peek (@"bufio".n·4 int) (? []byte, ? error) + func (@"bufio".b·3 *@"bufio".Reader) Read (@"bufio".p·4 []byte) (@"bufio".n·1 int, @"bufio".err·2 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadByte () (@"bufio".c·1 byte, @"bufio".err·2 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadBytes (@"bufio".delim·4 byte) (@"bufio".line·1 []byte, @"bufio".err·2 error) + func (@"bufio".b·4 *@"bufio".Reader) ReadLine () (@"bufio".line·1 []byte, @"bufio".isPrefix·2 bool, @"bufio".err·3 error) + func (@"bufio".b·4 *@"bufio".Reader) ReadRune () (@"bufio".r·1 rune, @"bufio".size·2 int, @"bufio".err·3 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadSlice (@"bufio".delim·4 byte) (@"bufio".line·1 []byte, @"bufio".err·2 error) + func (@"bufio".b·3 *@"bufio".Reader) ReadString (@"bufio".delim·4 byte) (@"bufio".line·1 string, @"bufio".err·2 error) + func (@"bufio".b·1 *@"bufio".Reader) Reset (@"bufio".r·2 @"io".Reader) + func (@"bufio".b·2 *@"bufio".Reader "esc:0x0") UnreadByte () (? error) + func (@"bufio".b·2 *@"bufio".Reader "esc:0x0") UnreadRune () (? error) { if @"bufio".b·2.@"bufio".lastRuneSize < 0x0 || @"bufio".b·2.@"bufio".r < @"bufio".b·2.@"bufio".lastRuneSize { return @"bufio".ErrInvalidUnreadRune }; @"bufio".b·2.@"bufio".r -= @"bufio".b·2.@"bufio".lastRuneSize; @"bufio".b·2.@"bufio".lastByte = -0x1; @"bufio".b·2.@"bufio".lastRuneSize = -0x1; return nil } + func (@"bufio".b·3 *@"bufio".Reader) WriteTo (@"bufio".w·4 @"io".Writer) (@"bufio".n·1 int64, @"bufio".err·2 error) + func (@"bufio".b·1 *@"bufio".Reader) @"bufio".fill () + func (@"bufio".b·2 *@"bufio".Reader "esc:0x1") @"bufio".readErr () (? error) { var @"bufio".err·3 error; ; @"bufio".err·3 = @"bufio".b·2.@"bufio".err; @"bufio".b·2.@"bufio".err = nil; return @"bufio".err·3 } + func (@"bufio".b·1 *@"bufio".Reader "esc:0x0") @"bufio".reset (@"bufio".buf·2 []byte, @"bufio".r·3 @"io".Reader) { *@"bufio".b·1 = (@"bufio".Reader{ @"bufio".buf:@"bufio".buf·2, @"bufio".rd:@"bufio".r·3, @"bufio".lastByte:-0x1, @"bufio".lastRuneSize:-0x1 }) } + func (@"bufio".b·3 *@"bufio".Reader) @"bufio".writeBuf (@"bufio".w·4 @"io".Writer) (? int64, ? error) + import bytes "bytes" // indirect + type @"bytes".readOp int + type @"bytes".Buffer struct { @"bytes".buf []byte; @"bytes".off int; @"bytes".runeBytes [4]byte; @"bytes".bootstrap [64]byte; @"bytes".lastRead @"bytes".readOp } + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x1") Bytes () (? []byte) { return @"bytes".b·2.@"bytes".buf[@"bytes".b·2.@"bytes".off:] } + func (@"bytes".b·1 *@"bytes".Buffer) Grow (@"bytes".n·2 int) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") Len () (? int) { return len(@"bytes".b·2.@"bytes".buf) - @"bytes".b·2.@"bytes".off } + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x1") Next (@"bytes".n·3 int) (? []byte) + func (@"bytes".b·3 *@"bytes".Buffer) Read (@"bytes".p·4 []byte "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) ReadByte () (@"bytes".c·1 byte, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x0") ReadBytes (@"bytes".delim·4 byte) (@"bytes".line·1 []byte, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) ReadFrom (@"bytes".r·4 @"io".Reader) (@"bytes".n·1 int64, @"bytes".err·2 error) + func (@"bytes".b·4 *@"bytes".Buffer) ReadRune () (@"bytes".r·1 rune, @"bytes".size·2 int, @"bytes".err·3 error) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x0") ReadString (@"bytes".delim·4 byte) (@"bytes".line·1 string, @"bytes".err·2 error) + func (@"bytes".b·1 *@"bytes".Buffer) Reset () + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") String () (? string) { if @"bytes".b·2 == nil { return "" }; return string(@"bytes".b·2.@"bytes".buf[@"bytes".b·2.@"bytes".off:]) } + func (@"bytes".b·1 *@"bytes".Buffer) Truncate (@"bytes".n·2 int) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") UnreadByte () (? error) + func (@"bytes".b·2 *@"bytes".Buffer "esc:0x0") UnreadRune () (? error) + func (@"bytes".b·3 *@"bytes".Buffer) Write (@"bytes".p·4 []byte "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·2 *@"bytes".Buffer) WriteByte (@"bytes".c·3 byte) (? error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteRune (@"bytes".r·4 rune) (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteString (@"bytes".s·4 string "esc:0x0") (@"bytes".n·1 int, @"bytes".err·2 error) + func (@"bytes".b·3 *@"bytes".Buffer) WriteTo (@"bytes".w·4 @"io".Writer) (@"bytes".n·1 int64, @"bytes".err·2 error) + func (@"bytes".b·2 *@"bytes".Buffer) @"bytes".grow (@"bytes".n·3 int) (? int) + func (@"bytes".b·3 *@"bytes".Buffer "esc:0x1") @"bytes".readSlice (@"bytes".delim·4 byte) (@"bytes".line·1 []byte, @"bytes".err·2 error) + type @"mime/multipart".Part struct { Header @"net/textproto".MIMEHeader; @"mime/multipart".buffer *@"bytes".Buffer; @"mime/multipart".mr *@"mime/multipart".Reader; @"mime/multipart".bytesRead int; @"mime/multipart".disposition string; @"mime/multipart".dispositionParams map[string]string; @"mime/multipart".r @"io".Reader } + func (@"mime/multipart".p·2 *@"mime/multipart".Part) Close () (? error) + func (@"mime/multipart".p·2 *@"mime/multipart".Part "esc:0x0") FileName () (? string) + func (@"mime/multipart".p·2 *@"mime/multipart".Part "esc:0x0") FormName () (? string) + func (@"mime/multipart".p·3 *@"mime/multipart".Part) Read (@"mime/multipart".d·4 []byte) (@"mime/multipart".n·1 int, @"mime/multipart".err·2 error) + func (@"mime/multipart".p·1 *@"mime/multipart".Part "esc:0x0") @"mime/multipart".parseContentDisposition () + func (@"mime/multipart".bp·2 *@"mime/multipart".Part) @"mime/multipart".populateHeaders () (? error) + type @"mime/multipart".Reader struct { @"mime/multipart".bufReader *@"bufio".Reader; @"mime/multipart".currentPart *@"mime/multipart".Part; @"mime/multipart".partsRead int; @"mime/multipart".nl []byte; @"mime/multipart".nlDashBoundary []byte; @"mime/multipart".dashBoundaryDash []byte; @"mime/multipart".dashBoundary []byte } + func (@"mime/multipart".r·3 *@"mime/multipart".Reader) NextPart () (? *@"mime/multipart".Part, ? error) + func (@"mime/multipart".r·3 *@"mime/multipart".Reader) ReadForm (@"mime/multipart".maxMemory·4 int64) (@"mime/multipart".f·1 *@"mime/multipart".Form, @"mime/multipart".err·2 error) + func (@"mime/multipart".mr·2 *@"mime/multipart".Reader) @"mime/multipart".isBoundaryDelimiterLine (@"mime/multipart".line·3 []byte "esc:0x0") (@"mime/multipart".ret·1 bool) + func (@"mime/multipart".mr·2 *@"mime/multipart".Reader "esc:0x0") @"mime/multipart".isFinalBoundary (@"mime/multipart".line·3 []byte "esc:0x0") (? bool) + func (@"mime/multipart".mr·2 *@"mime/multipart".Reader "esc:0x0") @"mime/multipart".peekBufferIsEmptyPart (@"mime/multipart".peek·3 []byte "esc:0x0") (? bool) + type @"net/http".Request struct { Method string; URL *@"net/url".URL; Proto string; ProtoMajor int; ProtoMinor int; Header @"net/http".Header; Body @"io".ReadCloser; ContentLength int64; TransferEncoding []string; Close bool; Host string; Form @"net/url".Values; PostForm @"net/url".Values; MultipartForm *@"mime/multipart".Form; Trailer @"net/http".Header; RemoteAddr string; RequestURI string; TLS *@"crypto/tls".ConnectionState } + func (@"net/http".r·1 *@"net/http".Request "esc:0x0") AddCookie (@"net/http".c·2 *@"net/http".Cookie) + func (@"net/http".r·4 *@"net/http".Request "esc:0x0") BasicAuth () (@"net/http".username·1 string, @"net/http".password·2 string, @"net/http".ok·3 bool) + func (@"net/http".r·3 *@"net/http".Request "esc:0x0") Cookie (@"net/http".name·4 string "esc:0x0") (? *@"net/http".Cookie, ? error) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") Cookies () (? []*@"net/http".Cookie) + func (@"net/http".r·4 *@"net/http".Request) FormFile (@"net/http".key·5 string "esc:0x0") (? @"mime/multipart".File, ? *@"mime/multipart".FileHeader, ? error) + func (@"net/http".r·2 *@"net/http".Request) FormValue (@"net/http".key·3 string "esc:0x0") (? string) + func (@"net/http".r·3 *@"net/http".Request) MultipartReader () (? *@"mime/multipart".Reader, ? error) + func (@"net/http".r·2 *@"net/http".Request) ParseForm () (? error) + func (@"net/http".r·2 *@"net/http".Request) ParseMultipartForm (@"net/http".maxMemory·3 int64) (? error) + func (@"net/http".r·2 *@"net/http".Request) PostFormValue (@"net/http".key·3 string "esc:0x0") (? string) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") ProtoAtLeast (@"net/http".major·3 int, @"net/http".minor·4 int) (? bool) { return @"net/http".r·2.ProtoMajor > @"net/http".major·3 || @"net/http".r·2.ProtoMajor == @"net/http".major·3 && @"net/http".r·2.ProtoMinor >= @"net/http".minor·4 } + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") Referer () (? string) + func (@"net/http".r·1 *@"net/http".Request "esc:0x0") SetBasicAuth (@"net/http".username·2 string "esc:0x0", @"net/http".password·3 string "esc:0x0") + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") UserAgent () (? string) + func (@"net/http".r·2 *@"net/http".Request) Write (@"net/http".w·3 @"io".Writer) (? error) + func (@"net/http".r·2 *@"net/http".Request) WriteProxy (@"net/http".w·3 @"io".Writer) (? error) + func (@"net/http".r·1 *@"net/http".Request) @"net/http".closeBody () + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") @"net/http".expectsContinue () (? bool) + func (@"net/http".r·3 *@"net/http".Request) @"net/http".multipartReader () (? *@"mime/multipart".Reader, ? error) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") @"net/http".wantsClose () (? bool) + func (@"net/http".r·2 *@"net/http".Request "esc:0x0") @"net/http".wantsHttp10KeepAlive () (? bool) + func (@"net/http".req·2 *@"net/http".Request) @"net/http".write (@"net/http".w·3 @"io".Writer, @"net/http".usingProxy·4 bool, @"net/http".extraHeaders·5 @"net/http".Header "esc:0x0") (? error) + type @"net/http".Handler interface { ServeHTTP(? @"net/http".ResponseWriter, ? *@"net/http".Request) } + type @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".RouteMatch struct { Route *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route; Handler @"net/http".Handler; Vars map[string]string } + type @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".matcher interface { Match(? *@"net/http".Request, ? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".RouteMatch) (? bool) } + import syntax "regexp/syntax" // indirect + type @"regexp/syntax".InstOp uint8 + func (@"regexp/syntax".i·2 @"regexp/syntax".InstOp) String () (? string) { if uint(@"regexp/syntax".i·2) >= uint(len(@"regexp/syntax".instOpNames)) { return "" }; return @"regexp/syntax".instOpNames[@"regexp/syntax".i·2] } + type @"regexp/syntax".Inst struct { Op @"regexp/syntax".InstOp; Out uint32; Arg uint32; Rune []rune } + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") MatchEmptyWidth (@"regexp/syntax".before·3 rune, @"regexp/syntax".after·4 rune) (? bool) + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") MatchRune (@"regexp/syntax".r·3 rune) (? bool) + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") MatchRunePos (@"regexp/syntax".r·3 rune) (? int) + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") String () (? string) + func (@"regexp/syntax".i·2 *@"regexp/syntax".Inst "esc:0x0") @"regexp/syntax".op () (? @"regexp/syntax".InstOp) + type @"regexp/syntax".EmptyOp uint8 + type @"regexp/syntax".Prog struct { Inst []@"regexp/syntax".Inst; Start int; NumCap int } + func (@"regexp/syntax".p·3 *@"regexp/syntax".Prog "esc:0x0") Prefix () (@"regexp/syntax".prefix·1 string, @"regexp/syntax".complete·2 bool) + func (@"regexp/syntax".p·2 *@"regexp/syntax".Prog "esc:0x0") StartCond () (? @"regexp/syntax".EmptyOp) + func (@"regexp/syntax".p·2 *@"regexp/syntax".Prog "esc:0x0") String () (? string) + func (@"regexp/syntax".p·3 *@"regexp/syntax".Prog "esc:0x1") @"regexp/syntax".skipNop (@"regexp/syntax".pc·4 uint32) (? *@"regexp/syntax".Inst, ? uint32) + type @"regexp".onePassInst struct { ? @"regexp/syntax".Inst; Next []uint32 } + type @"regexp".onePassProg struct { Inst []@"regexp".onePassInst; Start int; NumCap int } + type @"regexp".thread struct { @"regexp".inst *@"regexp/syntax".Inst; @"regexp".cap []int } + type @"regexp".entry struct { @"regexp".pc uint32; @"regexp".t *@"regexp".thread } + type @"regexp".queue struct { @"regexp".sparse []uint32; @"regexp".dense []@"regexp".entry } + type @"regexp".inputBytes struct { @"regexp".str []byte } + func (@"regexp".i·2 *@"regexp".inputBytes "esc:0x0") @"regexp".canCheckPrefix () (? bool) { return true } + func (@"regexp".i·2 *@"regexp".inputBytes "esc:0x0") @"regexp".context (@"regexp".pos·3 int) (? @"regexp/syntax".EmptyOp) + func (@"regexp".i·2 *@"regexp".inputBytes "esc:0x0") @"regexp".hasPrefix (@"regexp".re·3 *@"regexp".Regexp "esc:0x0") (? bool) + func (@"regexp".i·2 *@"regexp".inputBytes "esc:0x0") @"regexp".index (@"regexp".re·3 *@"regexp".Regexp "esc:0x0", @"regexp".pos·4 int) (? int) + func (@"regexp".i·3 *@"regexp".inputBytes "esc:0x0") @"regexp".step (@"regexp".pos·4 int) (? rune, ? int) + type @"regexp".inputString struct { @"regexp".str string } + func (@"regexp".i·2 *@"regexp".inputString "esc:0x0") @"regexp".canCheckPrefix () (? bool) { return true } + func (@"regexp".i·2 *@"regexp".inputString "esc:0x0") @"regexp".context (@"regexp".pos·3 int) (? @"regexp/syntax".EmptyOp) + func (@"regexp".i·2 *@"regexp".inputString "esc:0x0") @"regexp".hasPrefix (@"regexp".re·3 *@"regexp".Regexp "esc:0x0") (? bool) + func (@"regexp".i·2 *@"regexp".inputString "esc:0x0") @"regexp".index (@"regexp".re·3 *@"regexp".Regexp "esc:0x0", @"regexp".pos·4 int) (? int) + func (@"regexp".i·3 *@"regexp".inputString "esc:0x0") @"regexp".step (@"regexp".pos·4 int) (? rune, ? int) + type @"io".RuneReader interface { ReadRune() (@"io".r rune, @"io".size int, @"io".err error) } + type @"regexp".inputReader struct { @"regexp".r @"io".RuneReader; @"regexp".atEOT bool; @"regexp".pos int } + func (@"regexp".i·2 *@"regexp".inputReader "esc:0x0") @"regexp".canCheckPrefix () (? bool) { return false } + func (@"regexp".i·2 *@"regexp".inputReader "esc:0x0") @"regexp".context (@"regexp".pos·3 int) (? @"regexp/syntax".EmptyOp) { return @"regexp/syntax".EmptyOp(0x0) } + func (@"regexp".i·2 *@"regexp".inputReader "esc:0x0") @"regexp".hasPrefix (@"regexp".re·3 *@"regexp".Regexp "esc:0x0") (? bool) { return false } + func (@"regexp".i·2 *@"regexp".inputReader "esc:0x0") @"regexp".index (@"regexp".re·3 *@"regexp".Regexp "esc:0x0", @"regexp".pos·4 int) (? int) { return -0x1 } + func (@"regexp".i·3 *@"regexp".inputReader) @"regexp".step (@"regexp".pos·4 int) (? rune, ? int) + type @"regexp".input interface { @"regexp".canCheckPrefix() (? bool); @"regexp".context(@"regexp".pos int) (? @"regexp/syntax".EmptyOp); @"regexp".hasPrefix(@"regexp".re *@"regexp".Regexp) (? bool); @"regexp".index(@"regexp".re *@"regexp".Regexp, @"regexp".pos int) (? int); @"regexp".step(@"regexp".pos int) (@"regexp".r rune, @"regexp".width int) } + type @"regexp".machine struct { @"regexp".re *@"regexp".Regexp; @"regexp".p *@"regexp/syntax".Prog; @"regexp".op *@"regexp".onePassProg; @"regexp".q0 @"regexp".queue; @"regexp".q1 @"regexp".queue; @"regexp".pool []*@"regexp".thread; @"regexp".matched bool; @"regexp".matchcap []int; @"regexp".inputBytes @"regexp".inputBytes; @"regexp".inputString @"regexp".inputString; @"regexp".inputReader @"regexp".inputReader } + func (@"regexp".m·2 *@"regexp".machine) @"regexp".add (@"regexp".q·3 *@"regexp".queue, @"regexp".pc·4 uint32, @"regexp".pos·5 int, @"regexp".cap·6 []int "esc:0x0", @"regexp".cond·7 @"regexp/syntax".EmptyOp, @"regexp".t·8 *@"regexp".thread) (? *@"regexp".thread) + func (@"regexp".m·2 *@"regexp".machine) @"regexp".alloc (@"regexp".i·3 *@"regexp/syntax".Inst) (? *@"regexp".thread) + func (@"regexp".m·1 *@"regexp".machine) @"regexp".clear (@"regexp".q·2 *@"regexp".queue) + func (@"regexp".m·1 *@"regexp".machine) @"regexp".free (@"regexp".t·2 *@"regexp".thread) { @"regexp".m·1.@"regexp".inputBytes.@"regexp".str = nil; @"regexp".m·1.@"regexp".inputString.@"regexp".str = ""; @"regexp".m·1.@"regexp".inputReader.@"regexp".r = nil; @"regexp".m·1.@"regexp".pool = append(@"regexp".m·1.@"regexp".pool, @"regexp".t·2) } + func (@"regexp".m·1 *@"regexp".machine) @"regexp".init (@"regexp".ncap·2 int) + func (@"regexp".m·2 *@"regexp".machine) @"regexp".match (@"regexp".i·3 @"regexp".input, @"regexp".pos·4 int) (? bool) + func (@"regexp".m·2 *@"regexp".machine "esc:0x2") @"regexp".newInputBytes (@"regexp".b·3 []byte) (? @"regexp".input) { @"regexp".m·2.@"regexp".inputBytes.@"regexp".str = @"regexp".b·3; return &@"regexp".m·2.@"regexp".inputBytes } + func (@"regexp".m·2 *@"regexp".machine "esc:0x2") @"regexp".newInputReader (@"regexp".r·3 @"io".RuneReader) (? @"regexp".input) { @"regexp".m·2.@"regexp".inputReader.@"regexp".r = @"regexp".r·3; @"regexp".m·2.@"regexp".inputReader.@"regexp".atEOT = false; @"regexp".m·2.@"regexp".inputReader.@"regexp".pos = 0x0; return &@"regexp".m·2.@"regexp".inputReader } + func (@"regexp".m·2 *@"regexp".machine "esc:0x2") @"regexp".newInputString (@"regexp".s·3 string) (? @"regexp".input) { @"regexp".m·2.@"regexp".inputString.@"regexp".str = @"regexp".s·3; return &@"regexp".m·2.@"regexp".inputString } + func (@"regexp".m·2 *@"regexp".machine) @"regexp".onepass (@"regexp".i·3 @"regexp".input, @"regexp".pos·4 int) (? bool) + func (@"regexp".m·1 *@"regexp".machine) @"regexp".step (@"regexp".runq·2 *@"regexp".queue, @"regexp".nextq·3 *@"regexp".queue, @"regexp".pos·4 int, @"regexp".nextPos·5 int, @"regexp".c·6 rune, @"regexp".nextCond·7 @"regexp/syntax".EmptyOp) + type @"regexp".Regexp struct { @"regexp".expr string; @"regexp".prog *@"regexp/syntax".Prog; @"regexp".onepass *@"regexp".onePassProg; @"regexp".prefix string; @"regexp".prefixBytes []byte; @"regexp".prefixComplete bool; @"regexp".prefixRune rune; @"regexp".prefixEnd uint32; @"regexp".cond @"regexp/syntax".EmptyOp; @"regexp".numSubexp int; @"regexp".subexpNames []string; @"regexp".longest bool; @"regexp".mu @"sync".Mutex; @"regexp".machine []*@"regexp".machine } + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") Expand (@"regexp".dst·3 []byte "esc:0x2", @"regexp".template·4 []byte "esc:0x0", @"regexp".src·5 []byte "esc:0x0", @"regexp".match·6 []int "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") ExpandString (@"regexp".dst·3 []byte "esc:0x2", @"regexp".template·4 string, @"regexp".src·5 string "esc:0x0", @"regexp".match·6 []int "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) Find (@"regexp".b·3 []byte) (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) FindAll (@"regexp".b·3 []byte, @"regexp".n·4 int) (? [][]byte) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllIndex (@"regexp".b·3 []byte, @"regexp".n·4 int) (? [][]int) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllString (@"regexp".s·3 string, @"regexp".n·4 int) (? []string) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllStringIndex (@"regexp".s·3 string, @"regexp".n·4 int) (? [][]int) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllStringSubmatch (@"regexp".s·3 string, @"regexp".n·4 int) (? [][]string) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllStringSubmatchIndex (@"regexp".s·3 string, @"regexp".n·4 int) (? [][]int) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllSubmatch (@"regexp".b·3 []byte, @"regexp".n·4 int) (? [][][]byte) + func (@"regexp".re·2 *@"regexp".Regexp) FindAllSubmatchIndex (@"regexp".b·3 []byte, @"regexp".n·4 int) (? [][]int) + func (@"regexp".re·2 *@"regexp".Regexp) FindIndex (@"regexp".b·3 []byte) (@"regexp".loc·1 []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindReaderIndex (@"regexp".r·3 @"io".RuneReader) (@"regexp".loc·1 []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindReaderSubmatchIndex (@"regexp".r·3 @"io".RuneReader) (? []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindString (@"regexp".s·3 string) (? string) + func (@"regexp".re·2 *@"regexp".Regexp) FindStringIndex (@"regexp".s·3 string) (@"regexp".loc·1 []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindStringSubmatch (@"regexp".s·3 string) (? []string) + func (@"regexp".re·2 *@"regexp".Regexp) FindStringSubmatchIndex (@"regexp".s·3 string) (? []int) + func (@"regexp".re·2 *@"regexp".Regexp) FindSubmatch (@"regexp".b·3 []byte) (? [][]byte) + func (@"regexp".re·2 *@"regexp".Regexp) FindSubmatchIndex (@"regexp".b·3 []byte) (? []int) + func (@"regexp".re·3 *@"regexp".Regexp "esc:0x1") LiteralPrefix () (@"regexp".prefix·1 string, @"regexp".complete·2 bool) { return @"regexp".re·3.@"regexp".prefix, @"regexp".re·3.@"regexp".prefixComplete } + func (@"regexp".re·1 *@"regexp".Regexp "esc:0x0") Longest () { @"regexp".re·1.@"regexp".longest = true } + func (@"regexp".re·2 *@"regexp".Regexp) Match (@"regexp".b·3 []byte) (? bool) + func (@"regexp".re·2 *@"regexp".Regexp) MatchReader (@"regexp".r·3 @"io".RuneReader) (? bool) + func (@"regexp".re·2 *@"regexp".Regexp) MatchString (@"regexp".s·3 string) (? bool) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") NumSubexp () (? int) { return @"regexp".re·2.@"regexp".numSubexp } + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAll (@"regexp".src·3 []byte, @"regexp".repl·4 []byte "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllFunc (@"regexp".src·3 []byte, @"regexp".repl·4 func(? []byte) (? []byte) "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllLiteral (@"regexp".src·3 []byte, @"regexp".repl·4 []byte "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllLiteralString (@"regexp".src·3 string, @"regexp".repl·4 string "esc:0x0") (? string) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllString (@"regexp".src·3 string, @"regexp".repl·4 string) (? string) + func (@"regexp".re·2 *@"regexp".Regexp) ReplaceAllStringFunc (@"regexp".src·3 string, @"regexp".repl·4 func(? string) (? string) "esc:0x0") (? string) + func (@"regexp".re·2 *@"regexp".Regexp) Split (@"regexp".s·3 string, @"regexp".n·4 int) (? []string) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x1") String () (? string) { return @"regexp".re·2.@"regexp".expr } + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x1") SubexpNames () (? []string) { return @"regexp".re·2.@"regexp".subexpNames } + func (@"regexp".re·1 *@"regexp".Regexp) @"regexp".allMatches (@"regexp".s·2 string, @"regexp".b·3 []byte, @"regexp".n·4 int, @"regexp".deliver·5 func(? []int) "esc:0x0") + func (@"regexp".re·2 *@"regexp".Regexp) @"regexp".doExecute (@"regexp".r·3 @"io".RuneReader, @"regexp".b·4 []byte, @"regexp".s·5 string, @"regexp".pos·6 int, @"regexp".ncap·7 int) (? []int) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") @"regexp".expand (@"regexp".dst·3 []byte "esc:0x2", @"regexp".template·4 string, @"regexp".bsrc·5 []byte "esc:0x0", @"regexp".src·6 string "esc:0x0", @"regexp".match·7 []int "esc:0x0") (? []byte) + func (@"regexp".re·2 *@"regexp".Regexp) @"regexp".get () (? *@"regexp".machine) + func (@"regexp".re·2 *@"regexp".Regexp "esc:0x0") @"regexp".pad (@"regexp".a·3 []int "esc:0x2") (? []int) + func (@"regexp".re·1 *@"regexp".Regexp) @"regexp".put (@"regexp".z·2 *@"regexp".machine) + func (@"regexp".re·2 *@"regexp".Regexp) @"regexp".replaceAll (@"regexp".bsrc·3 []byte, @"regexp".src·4 string, @"regexp".nmatch·5 int, @"regexp".repl·6 func(@"regexp".dst []byte, @"regexp".m []int) (? []byte) "esc:0x0") (? []byte) + type @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routeRegexp struct { @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".template string; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".matchHost bool; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".matchQuery bool; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".strictSlash bool; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".regexp *@"regexp".Regexp; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".reverse string; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".varsN []string; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".varsR []*@"regexp".Regexp } + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routeRegexp) Match (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".req·3 *@"net/http".Request, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".match·4 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".RouteMatch "esc:0x0") (? bool) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routeRegexp "esc:0x0") @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".getUrlQuery (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".req·3 *@"net/http".Request) (? string) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routeRegexp) @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".matchQueryString (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".req·3 *@"net/http".Request) (? bool) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·3 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routeRegexp) @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".url (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".values·4 map[string]string "esc:0x0") (? string, ? error) + type @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routeRegexpGroup struct { @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".host *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routeRegexp; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".path *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routeRegexp; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".queries []*@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routeRegexp } + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".v·1 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routeRegexpGroup) @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".setMatch (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".req·2 *@"net/http".Request, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".m·3 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".RouteMatch "esc:0x0", @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·4 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route "esc:0x0") + type @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".BuildVarsFunc func(? map[string]string) (? map[string]string) + type @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".MatcherFunc func(? *@"net/http".Request, ? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".RouteMatch) (? bool) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".m·2 @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".MatcherFunc "esc:0x0") Match (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·3 *@"net/http".Request, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".match·4 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".RouteMatch) (? bool) + type @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route struct { @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".parent @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".parentRoute; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".handler @"net/http".Handler; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".matchers []@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".matcher; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".regexp *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routeRegexpGroup; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".strictSlash bool; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".buildOnly bool; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".name string; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".err error; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".buildVarsFunc @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".BuildVarsFunc } + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route "esc:0x2") BuildOnly () (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) { @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2.@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".buildOnly = true; return @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 } + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route "esc:0x2") BuildVarsFunc (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".f·3 @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".BuildVarsFunc) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) { @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2.@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".buildVarsFunc = @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".f·3; return @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 } + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route "esc:0x1") GetError () (? error) { return @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2.@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".err } + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route "esc:0x1") GetHandler () (? @"net/http".Handler) { return @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2.@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".handler } + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route "esc:0x1") GetName () (? string) { return @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2.@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".name } + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route "esc:0x2") Handler (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".handler·3 @"net/http".Handler) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) { if @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2.@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".err == nil { @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2.@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".handler = @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".handler·3 }; return @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 } + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route "esc:0x2") HandlerFunc (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".f·3 func(? @"net/http".ResponseWriter, ? *@"net/http".Request)) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) Headers (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".pairs·3 ...string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) HeadersRegexp (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".pairs·3 ...string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) Host (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".tpl·3 string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) Match (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".req·3 *@"net/http".Request, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".match·4 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".RouteMatch) (? bool) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) MatcherFunc (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".f·3 @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".MatcherFunc) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) Methods (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".methods·3 ...string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) Name (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".name·3 string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) Path (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".tpl·3 string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) PathPrefix (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".tpl·3 string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) Queries (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".pairs·3 ...string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) Schemes (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".schemes·3 ...string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) Subrouter () (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·3 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) URL (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".pairs·4 ...string) (? *@"net/url".URL, ? error) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·3 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) URLHost (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".pairs·4 ...string) (? *@"net/url".URL, ? error) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·3 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) URLPath (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".pairs·4 ...string) (? *@"net/url".URL, ? error) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".addMatcher (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".m·3 @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".matcher) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) { if @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2.@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".err == nil { @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2.@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".matchers = append(@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2.@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".matchers, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".m·3) }; return @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 } + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".addRegexpMatcher (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".tpl·3 string, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".matchHost·4 bool, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".matchPrefix·5 bool, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".matchQuery·6 bool) (? error) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".buildVars (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".m·3 map[string]string) (? map[string]string) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".getNamedRoutes () (? map[string]*@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".getRegexpGroup () (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routeRegexpGroup) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·3 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".prepareVars (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".pairs·4 ...string) (? map[string]string, ? error) + type @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".parentRoute interface { @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".buildVars(? map[string]string) (? map[string]string); @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".getNamedRoutes() (? map[string]*@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route); @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".getRegexpGroup() (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routeRegexpGroup) } + type @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".WalkFunc func(@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".route *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".router *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".ancestors []*@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) (? error) + type @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router struct { NotFoundHandler @"net/http".Handler; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".parent @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".parentRoute; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routes []*@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".namedRoutes map[string]*@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".strictSlash bool; KeepContext bool } + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) BuildVarsFunc (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".f·3 @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".BuildVarsFunc) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) Get (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".name·3 string "esc:0x0") (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) GetRoute (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".name·3 string "esc:0x0") (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) Handle (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".path·3 string, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".handler·4 @"net/http".Handler) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) HandleFunc (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".path·3 string, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".f·4 func(? @"net/http".ResponseWriter, ? *@"net/http".Request)) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) Headers (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".pairs·3 ...string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) Host (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".tpl·3 string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router "esc:0x0") Match (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".req·3 *@"net/http".Request, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".match·4 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".RouteMatch) (? bool) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) MatcherFunc (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".f·3 @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".MatcherFunc) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) Methods (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".methods·3 ...string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) NewRoute () (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) { var @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".route·3 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route; ; @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".route·3 = (&@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route{ @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".parent:@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".strictSlash:@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2.@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".strictSlash }); @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2.@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routes = append(@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2.@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routes, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".route·3); return @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".route·3 } + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) Path (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".tpl·3 string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) PathPrefix (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".tpl·3 string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) Queries (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".pairs·3 ...string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) Schemes (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".schemes·3 ...string) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·1 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) ServeHTTP (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".w·2 @"net/http".ResponseWriter, @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".req·3 *@"net/http".Request) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router "esc:0x2") StrictSlash (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".value·3 bool) (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) { @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2.@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".strictSlash = @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".value·3; return @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 } + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) Walk (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".walkFn·3 @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".WalkFunc "esc:0x0") (? error) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".buildVars (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".m·3 map[string]string) (? map[string]string) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".getNamedRoutes () (? map[string]*@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".getRegexpGroup () (? *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".routeRegexpGroup) + func (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".r·2 *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router) @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".walk (@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".walkFn·3 @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".WalkFunc "esc:0x0", @"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".ancestors·4 []*@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Route) (? error) + type @"github.com/fsouza/go-dockerclient".BlkioStatsEntry struct { Major uint64 "json:\"major,omitempty\" yaml:\"major,omitempty\""; Minor uint64 "json:\"minor,omitempty\" yaml:\"minor,omitempty\""; Op string "json:\"op,omitempty\" yaml:\"op,omitempty\""; Value uint64 "json:\"value,omitempty\" yaml:\"value,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".CPUStats struct { CPUUsage struct { PercpuUsage []uint64 "json:\"percpu_usage,omitempty\" yaml:\"percpu_usage,omitempty\""; UsageInUsermode uint64 "json:\"usage_in_usermode,omitempty\" yaml:\"usage_in_usermode,omitempty\""; TotalUsage uint64 "json:\"total_usage,omitempty\" yaml:\"total_usage,omitempty\""; UsageInKernelmode uint64 "json:\"usage_in_kernelmode,omitempty\" yaml:\"usage_in_kernelmode,omitempty\"" } "json:\"cpu_usage,omitempty\" yaml:\"cpu_usage,omitempty\""; SystemCPUUsage uint64 "json:\"system_cpu_usage,omitempty\" yaml:\"system_cpu_usage,omitempty\""; ThrottlingData struct { Periods uint64 "json:\"periods,omitempty\""; ThrottledPeriods uint64 "json:\"throttled_periods,omitempty\""; ThrottledTime uint64 "json:\"throttled_time,omitempty\"" } "json:\"throttling_data,omitempty\" yaml:\"throttling_data,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".Stats struct { Read @"time".Time "json:\"read,omitempty\" yaml:\"read,omitempty\""; Network struct { RxDropped uint64 "json:\"rx_dropped,omitempty\" yaml:\"rx_dropped,omitempty\""; RxBytes uint64 "json:\"rx_bytes,omitempty\" yaml:\"rx_bytes,omitempty\""; RxErrors uint64 "json:\"rx_errors,omitempty\" yaml:\"rx_errors,omitempty\""; TxPackets uint64 "json:\"tx_packets,omitempty\" yaml:\"tx_packets,omitempty\""; TxDropped uint64 "json:\"tx_dropped,omitempty\" yaml:\"tx_dropped,omitempty\""; RxPackets uint64 "json:\"rx_packets,omitempty\" yaml:\"rx_packets,omitempty\""; TxErrors uint64 "json:\"tx_errors,omitempty\" yaml:\"tx_errors,omitempty\""; TxBytes uint64 "json:\"tx_bytes,omitempty\" yaml:\"tx_bytes,omitempty\"" } "json:\"network,omitempty\" yaml:\"network,omitempty\""; MemoryStats struct { Stats struct { TotalPgmafault uint64 "json:\"total_pgmafault,omitempty\" yaml:\"total_pgmafault,omitempty\""; Cache uint64 "json:\"cache,omitempty\" yaml:\"cache,omitempty\""; MappedFile uint64 "json:\"mapped_file,omitempty\" yaml:\"mapped_file,omitempty\""; TotalInactiveFile uint64 "json:\"total_inactive_file,omitempty\" yaml:\"total_inactive_file,omitempty\""; Pgpgout uint64 "json:\"pgpgout,omitempty\" yaml:\"pgpgout,omitempty\""; Rss uint64 "json:\"rss,omitempty\" yaml:\"rss,omitempty\""; TotalMappedFile uint64 "json:\"total_mapped_file,omitempty\" yaml:\"total_mapped_file,omitempty\""; Writeback uint64 "json:\"writeback,omitempty\" yaml:\"writeback,omitempty\""; Unevictable uint64 "json:\"unevictable,omitempty\" yaml:\"unevictable,omitempty\""; Pgpgin uint64 "json:\"pgpgin,omitempty\" yaml:\"pgpgin,omitempty\""; TotalUnevictable uint64 "json:\"total_unevictable,omitempty\" yaml:\"total_unevictable,omitempty\""; Pgmajfault uint64 "json:\"pgmajfault,omitempty\" yaml:\"pgmajfault,omitempty\""; TotalRss uint64 "json:\"total_rss,omitempty\" yaml:\"total_rss,omitempty\""; TotalRssHuge uint64 "json:\"total_rss_huge,omitempty\" yaml:\"total_rss_huge,omitempty\""; TotalWriteback uint64 "json:\"total_writeback,omitempty\" yaml:\"total_writeback,omitempty\""; TotalInactiveAnon uint64 "json:\"total_inactive_anon,omitempty\" yaml:\"total_inactive_anon,omitempty\""; RssHuge uint64 "json:\"rss_huge,omitempty\" yaml:\"rss_huge,omitempty\""; HierarchicalMemoryLimit uint64 "json:\"hierarchical_memory_limit,omitempty\" yaml:\"hierarchical_memory_limit,omitempty\""; TotalPgfault uint64 "json:\"total_pgfault,omitempty\" yaml:\"total_pgfault,omitempty\""; TotalActiveFile uint64 "json:\"total_active_file,omitempty\" yaml:\"total_active_file,omitempty\""; ActiveAnon uint64 "json:\"active_anon,omitempty\" yaml:\"active_anon,omitempty\""; TotalActiveAnon uint64 "json:\"total_active_anon,omitempty\" yaml:\"total_active_anon,omitempty\""; TotalPgpgout uint64 "json:\"total_pgpgout,omitempty\" yaml:\"total_pgpgout,omitempty\""; TotalCache uint64 "json:\"total_cache,omitempty\" yaml:\"total_cache,omitempty\""; InactiveAnon uint64 "json:\"inactive_anon,omitempty\" yaml:\"inactive_anon,omitempty\""; ActiveFile uint64 "json:\"active_file,omitempty\" yaml:\"active_file,omitempty\""; Pgfault uint64 "json:\"pgfault,omitempty\" yaml:\"pgfault,omitempty\""; InactiveFile uint64 "json:\"inactive_file,omitempty\" yaml:\"inactive_file,omitempty\""; TotalPgpgin uint64 "json:\"total_pgpgin,omitempty\" yaml:\"total_pgpgin,omitempty\"" } "json:\"stats,omitempty\" yaml:\"stats,omitempty\""; MaxUsage uint64 "json:\"max_usage,omitempty\" yaml:\"max_usage,omitempty\""; Usage uint64 "json:\"usage,omitempty\" yaml:\"usage,omitempty\""; Failcnt uint64 "json:\"failcnt,omitempty\" yaml:\"failcnt,omitempty\""; Limit uint64 "json:\"limit,omitempty\" yaml:\"limit,omitempty\"" } "json:\"memory_stats,omitempty\" yaml:\"memory_stats,omitempty\""; BlkioStats struct { IOServiceBytesRecursive []@"github.com/fsouza/go-dockerclient".BlkioStatsEntry "json:\"io_service_bytes_recursive,omitempty\" yaml:\"io_service_bytes_recursive,omitempty\""; IOServicedRecursive []@"github.com/fsouza/go-dockerclient".BlkioStatsEntry "json:\"io_serviced_recursive,omitempty\" yaml:\"io_serviced_recursive,omitempty\""; IOQueueRecursive []@"github.com/fsouza/go-dockerclient".BlkioStatsEntry "json:\"io_queue_recursive,omitempty\" yaml:\"io_queue_recursive,omitempty\""; IOServiceTimeRecursive []@"github.com/fsouza/go-dockerclient".BlkioStatsEntry "json:\"io_service_time_recursive,omitempty\" yaml:\"io_service_time_recursive,omitempty\""; IOWaitTimeRecursive []@"github.com/fsouza/go-dockerclient".BlkioStatsEntry "json:\"io_wait_time_recursive,omitempty\" yaml:\"io_wait_time_recursive,omitempty\""; IOMergedRecursive []@"github.com/fsouza/go-dockerclient".BlkioStatsEntry "json:\"io_merged_recursive,omitempty\" yaml:\"io_merged_recursive,omitempty\""; IOTimeRecursive []@"github.com/fsouza/go-dockerclient".BlkioStatsEntry "json:\"io_time_recursive,omitempty\" yaml:\"io_time_recursive,omitempty\""; SectorsRecursive []@"github.com/fsouza/go-dockerclient".BlkioStatsEntry "json:\"sectors_recursive,omitempty\" yaml:\"sectors_recursive,omitempty\"" } "json:\"blkio_stats,omitempty\" yaml:\"blkio_stats,omitempty\""; CPUStats @"github.com/fsouza/go-dockerclient".CPUStats "json:\"cpu_stats,omitempty\" yaml:\"cpu_stats,omitempty\""; PreCPUStats @"github.com/fsouza/go-dockerclient".CPUStats "json:\"precpu_stats,omitempty\"" } + type @"github.com/fsouza/go-dockerclient".APIEvents struct { Status string "json:\"Status,omitempty\" yaml:\"Status,omitempty\""; ID string "json:\"ID,omitempty\" yaml:\"ID,omitempty\""; From string "json:\"From,omitempty\" yaml:\"From,omitempty\""; Time int64 "json:\"Time,omitempty\" yaml:\"Time,omitempty\"" } + type @"".DockerServer struct { @"".containers []*@"github.com/fsouza/go-dockerclient".Container; @"".execs []*@"github.com/fsouza/go-dockerclient".ExecInspect; @"".execMut @"sync".RWMutex; @"".cMut @"sync".RWMutex; @"".images []@"github.com/fsouza/go-dockerclient".Image; @"".iMut @"sync".RWMutex; @"".imgIDs map[string]string; @"".networks []*@"github.com/fsouza/go-dockerclient".Network; @"".netMut @"sync".RWMutex; @"".listener @"net".Listener; @"".mux *@"github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux".Router; @"".hook func(? *@"net/http".Request); @"".failures map[string]string; @"".multiFailures []map[string]string; @"".execCallbacks map[string]func(); @"".statsCallbacks map[string]func(? string) (? @"github.com/fsouza/go-dockerclient".Stats); @"".customHandlers map[string]@"net/http".Handler; @"".handlerMutex @"sync".RWMutex; @"".cChan chan<- *@"github.com/fsouza/go-dockerclient".Container } + func (@"".s·1 *@"".DockerServer) CustomHandler (@"".path·2 string, @"".handler·3 @"net/http".Handler) + func (@"".s·2 *@"".DockerServer "esc:0x1") DefaultHandler () (? @"net/http".Handler) { return @"".s·2.@"".mux } + func (@"".s·2 *@"".DockerServer "esc:0x0") MutateContainer (@"".id·3 string "esc:0x0", @"".state·4 @"github.com/fsouza/go-dockerclient".State) (? error) + func (@"".s·1 *@"".DockerServer "esc:0x0") PrepareExec (@"".id·2 string, @"".callback·3 func()) { @"".s·1.@"".execCallbacks[@"".id·2] = @"".callback·3 } + func (@"".s·1 *@"".DockerServer "esc:0x0") PrepareFailure (@"".id·2 string, @"".urlRegexp·3 string) { @"".s·1.@"".failures[@"".id·2] = @"".urlRegexp·3 } + func (@"".s·1 *@"".DockerServer) PrepareMultiFailures (@"".id·2 string, @"".urlRegexp·3 string) { @"".s·1.@"".multiFailures = append(@"".s·1.@"".multiFailures, (map[string]string{ "error":@"".id·2, "url":@"".urlRegexp·3 })) } + func (@"".s·1 *@"".DockerServer "esc:0x0") PrepareStats (@"".id·2 string, @"".callback·3 func(? string) (? @"github.com/fsouza/go-dockerclient".Stats)) { @"".s·1.@"".statsCallbacks[@"".id·2] = @"".callback·3 } + func (@"".s·1 *@"".DockerServer "esc:0x0") ResetFailure (@"".id·2 string "esc:0x0") { delete(@"".s·1.@"".failures, @"".id·2) } + func (@"".s·1 *@"".DockerServer "esc:0x0") ResetMultiFailures () { @"".s·1.@"".multiFailures = ([]map[string]string{ }) } + func (@"".s·1 *@"".DockerServer) ServeHTTP (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request) + func (@"".s·1 *@"".DockerServer "esc:0x0") SetHook (@"".hook·2 func(? *@"net/http".Request)) { @"".s·1.@"".hook = @"".hook·2 } + func (@"".s·1 *@"".DockerServer) Stop () + func (@"".s·2 *@"".DockerServer) URL () (? string) + func (@"".s·1 *@"".DockerServer) @"".attachContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer) @"".buildImage (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request) + func (@"".s·1 *@"".DockerServer) @"".buildMuxer () + func (@"".s·1 *@"".DockerServer) @"".commitContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request) + func (@"".s·1 *@"".DockerServer) @"".createContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request) + func (@"".s·1 *@"".DockerServer) @"".createExecContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request) + func (@"".s·1 *@"".DockerServer) @"".createNetwork (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request) + func (@"".s·4 *@"".DockerServer) @"".findContainer (@"".idOrName·5 string "esc:0x0") (? *@"github.com/fsouza/go-dockerclient".Container, ? int, ? error) + func (@"".s·3 *@"".DockerServer) @"".findImage (@"".id·4 string "esc:0x0") (? string, ? error) + func (@"".s·4 *@"".DockerServer) @"".findImageByID (@"".id·5 string "esc:0x0") (? string, ? int, ? error) + func (@"".s·4 *@"".DockerServer) @"".findNetwork (@"".idOrName·5 string "esc:0x0") (? *@"github.com/fsouza/go-dockerclient".Network, ? int, ? error) + func (@"".s·2 *@"".DockerServer "esc:0x0") @"".generateEvent () (? *@"github.com/fsouza/go-dockerclient".APIEvents) + func (@"".s·2 *@"".DockerServer "esc:0x0") @"".generateID () (? string) + func (@"".s·3 *@"".DockerServer) @"".getExec (@"".id·4 string "esc:0x0") (? *@"github.com/fsouza/go-dockerclient".ExecInspect, ? error) + func (@"".s·1 *@"".DockerServer "esc:0x0") @"".getImage (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·2 *@"".DockerServer) @"".handlerWrapper (@"".f·3 func(? @"net/http".ResponseWriter, ? *@"net/http".Request)) (? func(? @"net/http".ResponseWriter, ? *@"net/http".Request)) + func (@"".s·1 *@"".DockerServer) @"".inspectContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer) @"".inspectExecContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer) @"".inspectImage (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer) @"".listContainers (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request) + func (@"".s·1 *@"".DockerServer "esc:0x0") @"".listEvents (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer) @"".listImages (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer) @"".listNetworks (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer "esc:0x0") @"".loadImage (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer) @"".networkInfo (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer "esc:0x0") @"".notify (@"".container·2 *@"github.com/fsouza/go-dockerclient".Container) { if @"".s·1.@"".cChan != nil { @"".s·1.@"".cChan <- @"".container·2 } } + func (@"".s·1 *@"".DockerServer) @"".pauseContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer "esc:0x0") @"".pingDocker (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer) @"".pullImage (@"".w·2 @"net/http".ResponseWriter "esc:0x0", @"".r·3 *@"net/http".Request) + func (@"".s·1 *@"".DockerServer) @"".pushImage (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request) + func (@"".s·1 *@"".DockerServer) @"".removeContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request) + func (@"".s·1 *@"".DockerServer) @"".removeImage (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer) @"".renameContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request) + func (@"".s·1 *@"".DockerServer) @"".resizeExecContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer) @"".startContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request) + func (@"".s·1 *@"".DockerServer) @"".startExecContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer) @"".statsContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request) + func (@"".s·1 *@"".DockerServer) @"".stopContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer) @"".tagImage (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request) + func (@"".s·1 *@"".DockerServer) @"".topContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer) @"".unpauseContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func (@"".s·1 *@"".DockerServer) @"".waitContainer (@"".w·2 @"net/http".ResponseWriter, @"".r·3 *@"net/http".Request "esc:0x0") + func @"".NewServer (@"".bind·3 string, @"".containerChan·4 chan<- *@"github.com/fsouza/go-dockerclient".Container, @"".hook·5 func(? *@"net/http".Request)) (? *@"".DockerServer, ? error) + func @"".init () + var @"time".months [12]string + var @"time".days [7]string + var @"time".Local *@"time".Location + var @"time".UTC *@"time".Location + type @"sync".rlocker struct { @"sync".w @"sync".Mutex; @"sync".writerSem uint32; @"sync".readerSem uint32; @"sync".readerCount int32; @"sync".readerWait int32 } + func (@"sync".r·1 *@"sync".rlocker) Lock () + func (@"sync".r·1 *@"sync".rlocker) Unlock () + var @"bufio".ErrInvalidUnreadRune error + var @"regexp/syntax".instOpNames []string + +$$ +_go_.6 0 0 0 644 429450 ` +go object darwin amd64 go1.4.2 X:precisestack + +! +go13ldarchive/tar.acrypto/rand.aencoding/json.aerrors.a +fmt.amath/rand.a +net.anet/http.aregexp.astrconv.astrings.a sync.a time.aFgithub.com/fsouza/go-dockerclient.a�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.a�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.a�"".NewServer� � eH� %H��$����H;Aw���H���HDŽ$HDŽ$H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$�H�\$ H�\$pH�\$(H�\$xH�D$0H�L$8H��$�H��H��$�t$HDŽ$H��$H��$H����H�H�$�H�\$H�\$hH�H�$H�D$�H�\$H�\$`H�H�$H�D$�H�\$H�\$XH�H�$H�D$�H�\$H�\$PH�H�$H�D$�H�\$H�\$HH�H�$H�D$�H�D$H�H��$�H��H���H�\$pH��$hH�\$xH��$pH�\$`H��$0H��$H��$�H�\$XH��$�H�\$PH��$�H�\$HH��$�H��$�H��$�H��$�H�H�$H�\$hH�\$H��$�H�\$�H�\$hH�$�H�\$hH�\$@H�1�H9�tpH�\$pH�$H�\$xH�\$H�L$@H��$�H�D$H��$�H�L$H� Qj0�YYH�\$hH��$HDŽ$HDŽ$H����H�H�$H�H�\$H�H�\$�H�D$�^���8 +00runtime.morestack_noctxt�go.string."tcp"�net.Listen�(type."".DockerServer�"runtime.newobject�,type.map[string]string�runtime.makemap�,type.map[string]string�runtime.makemap�,type.map[string]func()�runtime.makemap��type.map[string]func(string) github.com/fsouza/go-dockerclient.Stats�runtime.makemap�@type.map[string]net/http.Handler�runtime.makemap�""".statictmp_0019�� runtime.duffcopy�(type."".DockerServer� .runtime.writebarrierfat� :"".(*DockerServer).buildMuxer� Rgo.itab.*"".DockerServer.net/http.Handler� +"net/http.Serve·f� +runtime.newproc� *type.*"".DockerServer� *type.net/http.Handler� Rgo.itab.*"".DockerServer.net/http.Handler�  runtime.typ2Itabp�"".autotmp_0018�*type.*"".DockerServer"".autotmp_0017�(type."".DockerServer"".autotmp_0015��type.map[string]func(string) github.com/fsouza/go-dockerclient.Stats"".autotmp_0014�,type.map[string]func()"".autotmp_0013�,type.map[string]string"".autotmp_0012�,type.map[string]string"".&server�*type.*"".DockerServer "".err�type.error"".listener�"type.net.Listener "".~r4Ptype.error "".~r3@*type.*"".DockerServer"".hook08type.func(*net/http.Request) "".containerChan ptype.chan<- *github.com/fsouza/go-dockerclient.Container"".bindtype.string8%�����,��6�B�=Z$ #### �[-64lh#####� YWTgclocals·23366670ca46633c69280dc1e0c74865Tgclocals·84a0367b0d2b39a68501b8591f74d154�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�2"".(*DockerServer).notify��eH� %H;aw���H�� H�D$(H��81�H9�t0H�\$0H�\$H�H�$H��8H�l$H�\$H�\$�H�� � + 0runtime.morestack_noctxttptype.chan<- *github.com/fsouza/go-dockerclient.Container�"runtime.chansend1 @"".autotmp_0023btype.*github.com/fsouza/go-dockerclient.Container"".containerbtype.*github.com/fsouza/go-dockerclient.Container"".s*type.*"".DockerServer@G?p�0 +XTgclocals·9d97800b9eac7aaad25644c1094f6baaTgclocals·e1ae6533a9e39048ba0735a2264ce16a�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�:"".(*DockerServer).buildMuxer����eH� %H;aw���H��hH�H�$H�D$�H�\$H�\$HH�H�$�H�|$H��H���\)1��H�L$0H� $H�<$�5)H�$8H�\$HH�\$�H�D$01�@�hAH�\$pH�$H�<$��(H�$�H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H����(H�-H��H��H�H�H���m(H�\$@H��H��H�$H�D$PH�D$H�T$XH�T$H�L$`H�L$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��'H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���Q'H�-H��H��H�H�H���/'H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��&H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���&H�-H��H��H�H�H����%H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�n%H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H����$H�-H��H��H�H�H����$H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�0$H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H����#H�-H��H��H�H�H���u#H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��"H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���Y"H�-H��H��H�H�H���7"H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��!H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���!H�-H��H��H�H�H���� H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�v H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H����H�-H��H��H�H�H����H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�8H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H����H�-H��H��H�H�H���}H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���aH�-H��H��H�H�H���?H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���#H�-H��H��H�H�H���H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�~H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H����H�-H��H��H�H�H����H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�@H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H����H�-H��H��H�H�H����H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���iH�-H��H��H�H�H���GH�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���+H�-H��H��H�H�H��� H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H����H�-H��H��H�H�H����H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�HH�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H����H�-H��H��H�H�H����H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$� +H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���qH�-H��H��H�H�H���OH�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���3H�-H��H��H�H�H���H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H����H�-H��H��H�H�H����H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�PH�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H����H�-H��H��H�H�H����H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���yH�-H��H��H�H�H���WH�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�� H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���; H�-H��H��H�H�H��� H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�� H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���� H�-H��H��H�H�H���� H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�X H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���� +H�-H��H��H�H�H���� +H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$� +H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���� H�-H��H��H�H�H���_ H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���CH�-H��H��H�H�H���!H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��H�$H�\$pH�\$�H�\$8H�$H�\$(H�\$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���H�-H��H��H�H�H����H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�xH�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H����H�-H��H��H�H�H����H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$�:H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H����H�-H��H��H�H�H���H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���cH�-H��H��H�H�H���AH�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H���%H�-H��H��H�H�H���H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$��H�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H�t$pH���H�<$H�5H�|$H�H��H�\$H�\$@H�H�$�H�|$H����H�-H��H��H�H�H����H�\$@H��H��H�$H�D$PH�D$H�L$XH�L$H�T$`H�T$�H�\$ H�\$8H�H�$�H�D$H�-H�(H�D$(H�$H�<$tIH�$H�\$pH�\$�H�\$pH�$H�\$(H�\$�H�D$H�\$8H�$H�D$�H��hÉ%뮉�4���������%�t����������������%�6��������������%�������z�����X����%������<���������%�|����������������%�V���������������%�����������x����%�������\�����:����%����������������%�^���������������%� ��������������%�������d�����B����%������&���������%�f����������������%�(��������������%�������l�����J����%������.����� ����%�n����������������%�0��������������%�������t�����R����%������6���������%�v����������������%�8��������������%�������|�����Z����%������>���������%�~���������������%�@���������������%�����������b����%�������F�����$����%����������������%�H���������������%� +����������j����%������%���������� + 0runtime.morestack_noctxt:�type.map[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route^runtime.makemap��type.github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Router�"runtime.newobject�� runtime.duffzero�.runtime.writebarrierptr�.runtime.writebarrierptr�&go.string."/commit"��github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).Path�type.[1]string�"runtime.newobject�""".statictmp_0126��github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).Methods�Ztype.struct { F uintptr; R *"".DockerServer }�"runtime.newobject�R"".*DockerServer.("".commitContainer)·fm�.runtime.writebarrierptr�B"".(*DockerServer).handlerWrapper��github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).HandlerFunc�8go.string."/containers/json"��github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).Path�type.[1]string�"runtime.newobject�""".statictmp_0130� +�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).Methods� +Ztype.struct { F uintptr; R *"".DockerServer }� +"runtime.newobject� +P"".*DockerServer.("".listContainers)·fm� .runtime.writebarrierptr� B"".(*DockerServer).handlerWrapper� �github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).HandlerFunc� �github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).HandlerFunc�>>go.string."/containers/{id:.*}"�>�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).Path�?type.[1]string�?"runtime.newobject�?""".statictmp_0178�@�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).Methods�AZtype.struct { F uintptr; R *"".DockerServer }�A"runtime.newobject�AR"".*DockerServer.("".removeContainer)·fm�B.runtime.writebarrierptr�BB"".(*DockerServer).handlerWrapper�B�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).HandlerFunc�CHgo.string."/containers/{id:.*}/exec"�C�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).Path�Ctype.[1]string�C"runtime.newobject�D""".statictmp_0182�E�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).Methods�EZtype.struct { F uintptr; R *"".DockerServer }�E"runtime.newobject�EZ"".*DockerServer.("".createExecContainer)·fm�F.runtime.writebarrierptr�FB"".(*DockerServer).handlerWrapper�G�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).HandlerFunc�GJgo.string."/containers/{id:.*}/stats"�G�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).Path�Htype.[1]string�H"runtime.newobject�H""".statictmp_0186�I�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).Methods�JZtype.struct { F uintptr; R *"".DockerServer }�J"runtime.newobject�JP"".*DockerServer.("".statsContainer)·fm�K.runtime.writebarrierptr�KB"".(*DockerServer).handlerWrapper�K�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).HandlerFunc�L@go.string."/exec/{id:.*}/resize"�L�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).Path�Ltype.[1]string�L"runtime.newobject�M""".statictmp_0190�N�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).Methods�NZtype.struct { F uintptr; R *"".DockerServer }�N"runtime.newobject�OZ"".*DockerServer.("".resizeExecContainer)·fm�O.runtime.writebarrierptr�PB"".(*DockerServer).handlerWrapper�P�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).HandlerFunc�P>go.string."/exec/{id:.*}/start"�Q�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).Path�Qtype.[1]string�Q"runtime.newobject�Q""".statictmp_0194�S�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).Methods�SZtype.struct { F uintptr; R *"".DockerServer }�S"runtime.newobject�SX"".*DockerServer.("".startExecContainer)·fm�T.runtime.writebarrierptr�TB"".(*DockerServer).handlerWrapper�U�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).HandlerFunc�Ugo.string."/images/{id:.*}/get"���github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).Pathڇtype.[1]string�"runtime.newobject��""".statictmp_0242���github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).Methods։Ztype.struct { F uintptr; R *"".DockerServer }�"runtime.newobject��D"".*DockerServer.("".getImage)·fm֊.runtime.writebarrierptr��B"".(*DockerServer).handlerWrapper���github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).HandlerFunc�*go.string."/networks"���github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).Path��type.[1]string��"runtime.newobject��""".statictmp_0246���github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).Methods��Ztype.struct { F uintptr; R *"".DockerServer }��"runtime.newobjectȎL"".*DockerServer.("".listNetworks)·fm��.runtime.writebarrierptrΏB"".(*DockerServer).handlerWrapper���github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).HandlerFunc��:go.string."/networks/{id:.*}"Ȑ�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).Path�type.[1]string��"runtime.newobject��""".statictmp_0250Ē�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).Methods�Ztype.struct { F uintptr; R *"".DockerServer }��"runtime.newobject��J"".*DockerServer.("".networkInfo)·fm�.runtime.writebarrierptr��B"".(*DockerServer).handlerWrapperƔ�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).HandlerFunc��*go.string."/networks"���github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).Path��type.[1]stringĕ"runtime.newobject�""".statictmp_0254���github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).Methods��Ztype.struct { F uintptr; R *"".DockerServer }��"runtime.newobjectؗN"".*DockerServer.("".createNetwork)·fm��.runtime.writebarrierptr֘B"".(*DockerServer).handlerWrapper���github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).HandlerFunc��"".autotmp_0256\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0255type.*[1]string"".autotmp_0253type.[]string"".autotmp_0252\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0251type.*[1]string"".autotmp_0249type.[]string"".autotmp_0248\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0247type.*[1]string"".autotmp_0245type.[]string"".autotmp_0244\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0243type.*[1]string"".autotmp_0241type.[]string"".autotmp_0240\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0239type.*[1]string"".autotmp_0237type.[]string"".autotmp_0236\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0235type.*[1]string"".autotmp_0233type.[]string"".autotmp_0232\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0231type.*[1]string"".autotmp_0229type.[]string"".autotmp_0228\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0227type.*[1]string"".autotmp_0225type.[]string"".autotmp_0224\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0223type.*[1]string"".autotmp_0221type.[]string"".autotmp_0220\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0219type.*[1]string"".autotmp_0217type.[]string"".autotmp_0216\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0215type.*[1]string"".autotmp_0213type.[]string"".autotmp_0212\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0211type.*[1]string"".autotmp_0209type.[]string"".autotmp_0208\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0207type.*[1]string"".autotmp_0205type.[]string"".autotmp_0204\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0203type.*[1]string"".autotmp_0201type.[]string"".autotmp_0200\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0199type.*[1]string"".autotmp_0197type.[]string"".autotmp_0196\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0195type.*[1]string"".autotmp_0193type.[]string"".autotmp_0192\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0191type.*[1]string"".autotmp_0189type.[]string"".autotmp_0188\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0187type.*[1]string"".autotmp_0185type.[]string"".autotmp_0184\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0183type.*[1]string"".autotmp_0181type.[]string"".autotmp_0180\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0179type.*[1]string"".autotmp_0177type.[]string"".autotmp_0176\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0175type.*[1]string"".autotmp_0173type.[]string"".autotmp_0172\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0171type.*[1]string"".autotmp_0169type.[]string"".autotmp_0168\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0167type.*[1]string"".autotmp_0165type.[]string"".autotmp_0164\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0163type.*[1]string"".autotmp_0161type.[]string"".autotmp_0160\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0159type.*[1]string"".autotmp_0157type.[]string"".autotmp_0156\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0155type.*[1]string"".autotmp_0153type.[]string"".autotmp_0152\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0151type.*[1]string"".autotmp_0149type.[]string"".autotmp_0148\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0147type.*[1]string"".autotmp_0145type.[]string"".autotmp_0144\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0143type.*[1]string"".autotmp_0141type.[]string"".autotmp_0140\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0139type.*[1]string"".autotmp_0137type.[]string"".autotmp_0136\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0135type.*[1]string"".autotmp_0133type.[]string"".autotmp_0132\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0131type.*[1]string"".autotmp_0129type.[]string"".autotmp_0128\type.*struct { F uintptr; R *"".DockerServer }"".autotmp_0125/type.[]string"".autotmp_0124o�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Router"".autotmp_0123�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Router"".autotmp_0122jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0121�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0120�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0119jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0118�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0117�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0116jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0115�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0114�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0113jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0112�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0111�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0110jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0109�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0108�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0107jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0106�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0105�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0104�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0103�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0102jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0101�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0100�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0099jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0098�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0097�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0096jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0095�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0094�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0093jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0092�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0091�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0090jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0089�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0088�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0087jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0086�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0085�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0084jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0083�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0082�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0081jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0080�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0079�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0078jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0077�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0076�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0075jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0074�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0073�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0072jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0071�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0070�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0069jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0068�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0067�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0066jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0065�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0064�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0063jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0062�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0061�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0060jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0059�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0058�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0057jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0056�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0055�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0054jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0053�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0052�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0051jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0050�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0049�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0048jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0047�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0046�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0045jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0044�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0043�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0042jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0041�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0040�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0039jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0038�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0037�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0036jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0035�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0034�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0033jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0032�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0031�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0030jtype.func(net/http.ResponseWriter, *net/http.Request)"".autotmp_0029�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0028�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0026_�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0025O�type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".autotmp_0024?�type.map[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route"".s*type.*"".DockerServer��L����S������������������������������������-�.A1?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7?d7 ?d7?d7?d7?d7?d7?d3�Tgclocals·2c09ec81c5cb12328d7183f25bc48833Tgclocals·76225bbef6ae6e9e5960f6f7925b8185�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�4"".(*DockerServer).SetHook��eH� %H;aw���H��H�\$H�$H�<$tH�$�H�\$ H�\$�H��É%�� + 0runtime.morestack_noctxtz.runtime.writebarrierptr "".hook8type.func(*net/http.Request)"".s*type.*"".DockerServer + +P�' + +<Tgclocals·e8c55b930b09fa5028b5e4b78b8932dcTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�<"".(*DockerServer).PrepareExec��eH� %H;aw���H��8H�\$HH�\$(H�\$PH�\$0H�\$XH�\$ H�H�$H�\$@H��H�l$H�\$(H�\$H�\$ H�\$�H��8� + 0runtime.morestack_noctxtv,type.map[string]func()�$runtime.mapassign1@p +"".autotmp_0357/type.func()"".autotmp_0356type.string"".callback0type.func() +"".idtype.string"".s*type.*"".DockerServerpWo��S +hTgclocals·5197b04b6fafdc0c7d1822cc34066683Tgclocals·31214a5fe2ac06a8b2e85038c37289d6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�>"".(*DockerServer).PrepareStats��eH� %H;aw���H��8H�\$HH�\$(H�\$PH�\$0H�\$XH�\$ H�H�$H�\$@H��H�l$H�\$(H�\$H�\$ H�\$�H��8� + 0runtime.morestack_noctxtv�type.map[string]func(string) github.com/fsouza/go-dockerclient.Stats�$runtime.mapassign1@p +"".autotmp_0359/rtype.func(string) github.com/fsouza/go-dockerclient.Stats"".autotmp_0358type.string"".callback0rtype.func(string) github.com/fsouza/go-dockerclient.Stats +"".idtype.string"".s*type.*"".DockerServerpWo��S +hTgclocals·5197b04b6fafdc0c7d1822cc34066683Tgclocals·31214a5fe2ac06a8b2e85038c37289d6�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�B"".(*DockerServer).PrepareFailure��eH� %H;aw���H��@H�\$PH�\$0H�\$XH�\$8H�\$`H�\$ H�\$hH�\$(H�H�$H�\$HH���H�l$H�\$0H�\$H�\$ H�\$�H��@� + 0runtime.morestack_noctxt�,type.map[string]string�$runtime.mapassign1P� +"".autotmp_0361?type.string"".autotmp_0360type.string"".urlRegexp0type.string +"".idtype.string"".s*type.*"".DockerServer�a��]  +rTgclocals·1765c43755fbf91dfae87195c1ec24fbTgclocals·f29b89ce4cd57d8100665fbda8fdf405�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�N"".(*DockerServer).PrepareMultiFailures��eH� %H�D$�H;Aw���H��H�H�$H�D$�H�D$H�H�+H�l$XH�kH�l$`H��$�H�\$HH��$�H�\$PH�H�$H�D$@H�D$H�\$XH�\$H�\$HH�\$�H�H�+H�l$XH�kH�l$`H��$�H�\$HH��$�H�\$PH�H�$H�\$@H�\$H�\$XH�\$H�\$HH�\$�H��$�H���H���H���H��H��$�H��$�H��$�H��H)�H��}FH�H�$H�T$hH�T$H�L$pH�L$H�D$xH�D$H�D$ �H�T$(H�L$0H�D$8H��H��H�t$pH�D$xH�T$hH��H�$H�\$@H�\$�H�T$hH�L$pH�D$xH��$�H�$H�<$t"".(*DockerServer).ResetFailure��eH� %H;aw���H��(H�\$8H�\$H�\$@H�\$ H�H�$H�\$0H���H�l$H�\$H�\$�H��(� + 0runtime.morestack_noctxtb,type.map[string]string�"runtime.mapdelete0P"".autotmp_0375type.string +"".idtype.string"".s*type.*"".DockerServerPCO`�? +T Tgclocals·bd51743682bd6c0f7b9f2e8e6dffed99Tgclocals·8d600a433c6aaa81a4fe446d95c5546b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�J"".(*DockerServer).ResetMultiFailures��eH� %H;aw���H��8H�H�$�H�l$H��tMH�\$@1�1�H�$H�<$t0H�$�H�l$ H�l$H�T$(H�T$H�L$0H�L$�H��8É%�ljE� + + 0runtime.morestack_noctxt:2type.[0]map[string]stringL"runtime.newobject�2runtime.writebarrierslicep"".autotmp_0376/0type.[]map[string]string"".s*type.*"".DockerServerp^op��Z +%kTgclocals·519efd86263089ddb84df3cfe7fd2992Tgclocals·0528ab8f76149a707fd2f0025c2178a3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�@"".(*DockerServer).CustomHandler��eH� %H;aw���H��@H�\$HH�$H�<$��H�$ �H�\$PH�\$0H�\$XH�\$8H�\$`H�\$ H�\$hH�\$(H�H�$H�\$HH��H�l$H�\$0H�\$H�\$ H�\$�H�\$HH�$H�<$tH�$ �H��@É%��%�_��� + 0runtime.morestack_noctxtn(sync.(*RWMutex).Lock�@type.map[string]net/http.Handler�$runtime.mapassign1�,sync.(*RWMutex).UnlockP� +"".autotmp_0382?*type.net/http.Handler"".autotmp_0381type.string"".handler0*type.net/http.Handler"".pathtype.string"".s*type.*"".DockerServer���� �!]  6] Tgclocals·925be0824eaf197a56a5d7050bf29309Tgclocals·85223f890d4c8f80203775beed82eadd�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�D"".(*DockerServer).MutateContainer� +� eH� %H�D$�H;Aw���H���L��$HDŽ$hHDŽ$pH��$�H���H�H�CH�kH��$�1�H��$�H�D$(H��$�H�l$(H9��H�T$HH�H�t$0H�\$8H����H�;H�|$pH�CH�D$xL9���H�<$H�D$H��$H�l$L�D$�L��$H�t$0H�T$H�\$ ��tyH��$H��$�H��H���H�H�$H�\$8H�\$H�|$t8H�D$XH��$�H�\$�HDŽ$hHDŽ$pH���É%�H��H��H�l$(H9�����H�H�+H�l$`H�kH�l$hH�D$PH�D$XH�H�$�H�D$H�D$@H�$H�<$t}H�\$`H�\$H�\$hH�\$�H�\$@H�\$@H�1�H9�tH�L$@H��$hH��$pH����H�H�$H�H�\$H�H�\$�H�D$봉%�w�����=��������� +*0runtime.morestack_noctxt� runtime.eqstring�� runtime.duffcopy�Xtype.github.com/fsouza/go-dockerclient.State�.runtime.writebarrierfat�>go.string."container not found"�.type.errors.errorString�"runtime.newobject�4runtime.writebarrierstring�Bgo.itab.*errors.errorString.error�0type.*errors.errorString�type.error� Bgo.itab.*errors.errorString.error�  runtime.typ2Itab��"".autotmp_0391�0type.*errors.errorString"".autotmp_0390�type.string"".autotmp_0388�dtype.**github.com/fsouza/go-dockerclient.Container"".autotmp_0387�type.int"".autotmp_0386�type.int"".autotmp_03850type.*errors.errorString"".autotmp_0384�Xtype.github.com/fsouza/go-dockerclient.State"".autotmp_0383�ftype.[]*github.com/fsouza/go-dockerclient.Container "".~r0�type.errorerrors.text·2�type.string"".container�btype.*github.com/fsouza/go-dockerclient.Container "".~r2�type.error"".state0Xtype.github.com/fsouza/go-dockerclient.State +"".idtype.string"".s*type.*"".DockerServer("�������W�.�BWeP   � �lx.�Tgclocals·1e2d550ac4f017d716d87ff44946577fTgclocals·0e8ff9f111235a6bccca3fa33f624774�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�."".(*DockerServer).Stop��eH� %H;aw���H��(H�D$0H���t"H���H���H�l$ H�,$H�L$H�Y0��H��(� + 0runtime.morestack_noctxt� +P"".s*type.*"".DockerServerP5OP� +" +ITgclocals·519efd86263089ddb84df3cfe7fd2992Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�,"".(*DockerServer).URL��eH� %H;aw���H��pH�D$xHDŽ$�HDŽ$�H���uHDŽ$�HDŽ$�H��p�H���H���H�l$HH�,$H�L$@H�Y(��H�L$H�D$H�D$hH�$H�L$`H�Y(��H�L$H�D$H�H�,$H��H��H�H�H�L$PH�L$H�D$XH�D$H�H�l$ H��H��H�H��H�\$0H��$�H�\$8H��$�H��p� + 0runtime.morestack_noctxt� +� +�&go.string."http://"�go.string."/"�*runtime.concatstring30�"".autotmp_0397?type.string"".autotmp_0396type.net.Addr "".~r0type.string"".s*type.*"".DockerServer �C������7 +� ~�Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·f883d3996c76325fd1714d4e3de9fa33�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�8"".(*DockerServer).ServeHTTP� � eH� %H�D$�H;Aw���H��H��$�H�$H�<$�&H�$ �H��$�H�$H�<$��H�$ H� Qj�YYH����H��$�H��H�|$h1��H�H�$H�l$H�\$hH�\$�H�\$h1�H9���H�\$pH���fH�;H�sH�\$hH���IH�H�kH�|$XH�|$HH�t$`H�t$PH�T$8H�$H�l$@H�l$H��$�H�~H����H�w8H�|$H�H���\$ ��tIH��$�H�\$H��$�H�\$H��$�H�\$H�\$PH�$H�\$HH�[ �Ӑ�H�ĸ�H�\$hH�$�H�\$h1�H9�����H��$�H���H�,$H��$�H�\$H��$�H�\$H��$�H�\$�H��$�H���1�H9�tH��$�H�$H���H��Ӑ�H�ĸÉ�����������������H�ĸÉ%������%����� +*0runtime.morestack_noctxt�*sync.(*RWMutex).RLock�4sync.(*RWMutex).RUnlock·f�"runtime.deferproc�� runtime.duffzero�@type.map[string]net/http.Handler�&runtime.mapiterinit�$regexp.MatchString� +�&runtime.deferreturn�&runtime.mapiternext��github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).ServeHTTP� +�&runtime.deferreturn� &runtime.deferreturn@�"".autotmp_0401�*type.net/http.Handler"".autotmp_0400�Jtype.map.iter[string]net/http.Handler"".handler�*type.net/http.Handler +"".re�type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServerD"�K������"���F�"$9�E; GAs�Hh�Tgclocals·bc335ce91c3a8b5f426dd201465802bdTgclocals·3901c619f635162fa423fe138099ace5�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�B"".(*DockerServer).DefaultHandler��eH� %H;aw���H�� H�D$0H�D$8H�1�H9�tH�\$(H���H�l$8H�D$0H�� �H�H�$H�H�\$H�H�\$�H�D$� + 0runtime.morestack_noctxt^�go.itab.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Router.net/http.Handler��type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Router�*type.net/http.Handler��go.itab.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Router.net/http.Handler� runtime.typ2Itab0@ "".~r0*type.net/http.Handler"".s*type.*"".DockerServer@:?@;� �,d +xTgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�B"".(*DockerServer).handlerWrapper��eH� %H;aw���H��(H�H�$�H�D$H�D$H�l$0H�(H�H�$�H�D$H�D$ H�l$8H�(H�H�$�H�D$H�-H�(H�D$H�$H�<$tPH�$H�\$H�\$�H�\$H�$H�<$t#H�$H�\$ H�\$�H�\$H�\$@H��(É%�ԉ%� + 0runtime.morestack_noctxt:*type.*"".DockerServerL"runtime.newobject~jtype.func(net/http.ResponseWriter, *net/http.Request)�"runtime.newobject��type.struct { F uintptr; A0 **"".DockerServer; A1 *func(net/http.ResponseWriter, *net/http.Request) }�"runtime.newobject�"".func·001�.runtime.writebarrierptr�.runtime.writebarrierptr0P"".autotmp_0404/�type.*struct { F uintptr; A0 **"".DockerServer; A1 *func(net/http.ResponseWriter, *net/http.Request) } +"".&fltype.*func(net/http.ResponseWriter, *net/http.Request) +"".&s,type.**"".DockerServer "".~r1 jtype.func(net/http.ResponseWriter, *net/http.Request)P�OP��^8�%""3$0Tgclocals·ab01a2d55089ff50c402006df1039c39Tgclocals·be18fcff1e4d1cf801d0b47f660b9806�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�B"".(*DockerServer).listContainers�+�+eH� %H��$����H;Aw���H��1�H��$ �H��$�H�kH�,$�H�D$H�H�H��$�H�KH��$�H�D$xHDŽ$�1�H9��� 1�1�H�L$xH��$H��$�H��$ H��$�H�$H�<$�J H�$H�H��$�H�kH�H�$H�D$H�l$�H�T$H�L$ H�D$(H��$XH��$`H��$hH��$�H����H�3H�CH�kH��$�1�H��$�H�D$HH��$�H�l$HH9��YH�t$pH�H�|$PH�D$`H��$ H���hH��$H�,$H��$ H�t$H�5L�D$L��H�H��H�|$PH�t$pH�D$`�\$ ���H��� H�h8H�$H��H��H�H�H�H�H�l$H��H��H�H��H�\$(H��$�H�\$0H��$H��$1��H��$H����H��H��H��$�H��$�H��$�H�H�$H�\$`H�\$H�|$�EH�D$(�H�T$H�D$H��$�H�$H��$�H�T$H��$�H�D$�H�H�$H��$�H�\$�H�T$H�D$H��$�H��H�$H��$�H�T$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�T$`H�\$(H��$H�\$0H��$H���>H�Z�j��$HH�j H��$PH��$@H� n�����H�H�\$@H�$H�<$��H�$X�H�\$H��$�H�\$H��$�H�\$`H���H�,$�H�\$H��$�H�\$H��$�H�\$H��$�H��$�H�H�CH��$�H���_H��H��H��$�H��$�H��$�H�H�$H�\$`H�\$H�|$� H�D$ �H�T$H�D$H��$�H�$H��$�H�T$H��$�H�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�D$`H�\$(H��$�H�t$0H��$�H�5H��$ H���H���AH�(H��$ H��H��H�H�H���H��$0H��H��H�H�H��$H��$@H��$H��$HH�\$@H��$PH��$�H��$XH��$�H��$`H��$�H��$hH��$�H��$pH��$�H��$xH�H�$�H�T$H���pH��H��H��$�H��$�H��$�H��$�H��$�H��$xH��$�H��$�H��$pH�$H��$�H�\$H��$�H�\$�H��$XH��$`H��$hH��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H�H�$H��H��$�H��Hi�H�H�\$H��$ H�\$�H�|$PH�t$pH��$�H��$�H��$�H��$XH��$`H��$hH��H��H�l$HH9������H��$�H�$H�<$��H�$H�H��$�H�$H��$�H�[ ��H�t$H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H��H�D$�H��$�H�$H��$�H�[0��H�H�$H��$�H�\$H��$�H�\$�H�L$H�D$ H��$�H��$�H��$�H��$�H�H�$�H�|$H��H����1��H�L$hH� $H�<$��H��$�H�\$H��$�H�\$�H�\$hH�\$XH��$XH��$�H��$`H��$�H��$hH��$�H�H�$H��$�H�\$�H�\$H�l$H��H��H�H�H�\$XH�$�H�ĨÉ%�T�����2����%�2��������������%�����������%����������%������a����������XX��������������#����%����HDŽ$(HDŽ$0HDŽ$8H�H�$H�D$H��$H�T$H��$H�L$�H�D$ �\$(H��tPH�0H��$(H�PH��$0H�hH��$8��tH��tH��v H�H�F������ 1�1��������n +00runtime.morestack_noctxt`� runtime.duffzero�(net/url.(*URL).Query�go.string."all"�*sync.(*RWMutex).RLock�ltype.[]github.com/fsouza/go-dockerclient.APIContainers�"runtime.makeslice�go.string."1"� runtime.eqstring�go.string." "�strings.Join�� runtime.duffzero� type.string� +runtime.convT2E� 2runtime.writebarrieriface� type.string� runtime.convT2E� 2runtime.writebarrieriface� "go.string."%s %s"� fmt.Sprintf�bgithub.com/fsouza/go-dockerclient.(*State).String��github.com/fsouza/go-dockerclient.(*NetworkSettings).PortMappingAPI�type.string�runtime.convT2E�2runtime.writebarrieriface�go.string."/%s"�fmt.Sprintf�""".statictmp_0435�� runtime.duffcopy�type.[1]string�"runtime.newobject�4runtime.writebarrierstring�ltype.[]github.com/fsouza/go-dockerclient.APIContainers�"runtime.growslice�htype.github.com/fsouza/go-dockerclient.APIContainers�.runtime.writebarrierfat�.sync.(*RWMutex).RUnlock� +� 0go.string."Content-Type"� 8go.string."application/json"� &net/http.Header.Set�! +�!type.io.Writer�!runtime.convI2I�"4type.encoding/json.Encoder�""runtime.newobject�#� runtime.duffzero�$2runtime.writebarrieriface�%ltype.[]github.com/fsouza/go-dockerclient.APIContainers�%runtime.convT2E�%>encoding/json.(*Encoder).Encode�)&type.net/url.Values�)4runtime.mapaccess2_faststr�*$runtime.panicindex@� +R"".autotmp_0444�6type.*encoding/json.Encoder"".autotmp_04436type.*encoding/json.Encoder"".autotmp_0442�type.io.Writer"".autotmp_0439type.int"".autotmp_0438type.int"".autotmp_0437ltype.[]github.com/fsouza/go-dockerclient.APIContainers"".autotmp_0434�htype.github.com/fsouza/go-dockerclient.APIContainers"".autotmp_0433"type.interface {}"".autotmp_0431&type.[]interface {}"".autotmp_0429"type.interface {}"".autotmp_0428�"type.interface {}"".autotmp_0426�&type.[]interface {}"".autotmp_0424�dtype.**github.com/fsouza/go-dockerclient.Container"".autotmp_0423� type.int"".autotmp_0422type.int"".autotmp_0420�ltype.[]github.com/fsouza/go-dockerclient.APIContainers"".autotmp_0417�type.string"".autotmp_0416�(type.[1]interface {}"".autotmp_0415�`type.[]github.com/fsouza/go-dockerclient.APIPort"".autotmp_0414�type.string"".autotmp_0413type.string"".autotmp_0412�type.string"".autotmp_0411�(type.[2]interface {}"".autotmp_0410�ftype.[]*github.com/fsouza/go-dockerclient.Container"".autotmp_0409�ltype.[]github.com/fsouza/go-dockerclient.APIContainers"".autotmp_0408type.int"".autotmp_0407� type.int"".autotmp_0406�type.string "".~r0� 6type.*encoding/json.Encoder$encoding/json.w·2�type.io.Writer "".~r0� type.int64time.t·2�type.time.Time "".~r0�type.stringnet/url.vs·4�type.[]stringnet/url.key·3�type.string"".container� btype.*github.com/fsouza/go-dockerclient.Container"".result�ltype.[]github.com/fsouza/go-dockerclient.APIContainers "".all�type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer%� +�� +� +����4y!Q\`�>3<��!Q#�     �dD����v/�v��d_p�LW"�!xTgclocals·02bfe185cbfa386cc6696a665007ff28Tgclocals·a74ca190396b92ed76efe93c61653942�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�:"".(*DockerServer).listImages��eH� %H��$P���H;Aw���H��0H��$8H�$H�<$�EH�$H�H��$8H�khH�H�$H�l$H�l$�H�T$H�L$ H�D$(H��$�H��$�H��$�H��$8H����H�S`H�KhH�kpH��$H1�H��$@H�L$HH��$8H�l$HH9���H�T$hH����H��$ H��H���H�D$PH��H��$ H��$H��H���H��$@��$H��$�H��$PH��$�H��$�H� n�����H�H��$�1��H��$H��$�H��H��H�H�H��$�H�H�$H��$�H��H�T$@L��$�L9���Hk�pH�H�\$H��$�H�\$�H��$8H���H��$P1��H�H�$H�l$H��$PH�\$�H��$P1�H9��)H��$XH���'H�H�CH��$PH���H�+H��$�H�kH��$�H��$�H��$H��$�H��$H��$�H��$�H9���H�$H�D$H�t$H�L$��\$ ���dH��$�H�l$@L��$�L9��lHk�pH�H�SH�KH�[ H��$H��$H��$H��H)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$H��H��$�H��Hk�H�H�$H��$�H�\$H��$�H�\$�H��$�H��$�H��$H��$�H�l$@L��$�L9��[Hk�pH�H�$H�$H��$H�T$H��$H�L$H��$H�D$�H��$PH�$�H��$P1�H9������H�T$hH�D$PH��H��H�l$HH9��g���H��$8H�$H�<$��H�$H�H��$HH�$H��$@H�[ ��H�t$H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H��H�D$�H��$HH�$H��$@H�[0��H�H�$H��$@H�\$H��$HH�\$�H�L$H�D$ H��$�H�L$pH��$�H�D$xH�H�$�H�|$H��H����1��H�L$`H� $H�<$��H�\$pH�\$H�\$xH�\$�H�\$`H�\$XH��$�H��$ H��$�H��$(H��$�H��$0H�H�$H��$ H�\$�H�\$H�l$H��H��H�H�H�\$XH�$�H��0É%�Z�����8����%�>���� � ������������� ��s�����$����%����J +00runtime.morestack_noctxt�*sync.(*RWMutex).RLock�dtype.[]github.com/fsouza/go-dockerclient.APIImages�"runtime.makeslice�� runtime.duffcopy�� runtime.duffcopy�� runtime.duffzero�`type.github.com/fsouza/go-dockerclient.APIImages�.runtime.writebarrierfat�� runtime.duffzero�,type.map[string]string�&runtime.mapiterinit� + runtime.eqstring� type.[]string� "runtime.growslice�4runtime.writebarrierstring�2runtime.writebarrierslice�&runtime.mapiternext�.sync.(*RWMutex).RUnlock� +�0go.string."Content-Type"�8go.string."application/json"�&net/http.Header.Set� +�type.io.Writer�runtime.convI2I�4type.encoding/json.Encoder�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface�dtype.[]github.com/fsouza/go-dockerclient.APIImages�runtime.convT2E�>encoding/json.(*Encoder).Encode�$runtime.panicindex�$runtime.panicindex�$runtime.panicindex@�8"".autotmp_0483�6type.*encoding/json.Encoder"".autotmp_04826type.*encoding/json.Encoder"".autotmp_0481�type.io.Writer"".autotmp_0476�type.[]string"".autotmp_0475�type.[]string"".autotmp_0474type.string"".autotmp_0473�type.string"".autotmp_0471�Xtype.github.com/fsouza/go-dockerclient.Image"".autotmp_0470�Ztype.*github.com/fsouza/go-dockerclient.Image"".autotmp_0469�type.int"".autotmp_0468type.int"".autotmp_0467�dtype.[]github.com/fsouza/go-dockerclient.APIImages"".autotmp_0465type.[]string"".autotmp_0464�6type.map.iter[string]string"".autotmp_0462�`type.github.com/fsouza/go-dockerclient.APIImages"".autotmp_0461�\type.[]github.com/fsouza/go-dockerclient.Image"".autotmp_0459�type.int "".~r0�6type.*encoding/json.Encoder$encoding/json.w·2�type.io.Writertime.t·2�type.time.Time +"".id�type.string "".tag�type.string"".image�Xtype.github.com/fsouza/go-dockerclient.Image"".i�type.int"".result�dtype.[]github.com/fsouza/go-dockerclient.APIImages"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer%�� ��_� h�%!M�;{�W�$ "!Q#� BA�@��rg�FW", $Tgclocals·f7ba1512b6938de3ab7810c798567682Tgclocals·e0b091cc964057ade987c1196ae02e2e�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�8"".(*DockerServer).findImage��eH� %H;aw���H��pHDŽ$�HDŽ$�HDŽ$�HDŽ$�H�\$xH�$H�<$�xH�$x�H�\$xH�$H�<$�NH�$xH� Qj�YYH���$H��$�H��$�H�H�$H�\$xH���H�l$H�T$`H�T$H�L$hH�L$�H�T$ �\$(H����H�2H�t$@H�jH�l$H��t3H��$�H��$�HDŽ$�HDŽ$���H��p�H�\$xH�$H��$�H�\$H��$�H�\$�H�t$H�l$ H�T$0H�L$8H�t$@H��$�H�l$HH��$�H�T$PH��$�H�L$XH��$���H��pÉ�5�����H��pÉ%�����%�|��� + 0runtime.morestack_noctxt�*sync.(*RWMutex).RLock�4sync.(*RWMutex).RUnlock·f�"runtime.deferproc�,type.map[string]string�4runtime.mapaccess2_faststr�&runtime.deferreturn�@"".(*DockerServer).findImageByID�&runtime.deferreturn�&runtime.deferreturnp�"".autotmp_0487type.string "".err?type.error"".image_type.string "".~r2Ptype.error "".~r10type.string +"".idtype.string"".s*type.*"".DockerServerB�o���z����&�0�J3j3<? cxW.MCTgclocals·f7cb58e18cf0f9d3ee7dc7385e94aef7Tgclocals·660c52760819425e2fa6ae9a8a8ae931�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�@"".(*DockerServer).findImageByID�� eH� %H��$����H;Aw���H���HDŽ$�HDŽ$�HDŽ$�HDŽ$HDŽ$H��$�H�$H�<$��H�$x�H��$�H�$H�<$��H�$xH� Qj�L��$�YYH����H��$�H���xL�C`H�ChH�kpH��$�1�H��$�H�D$0L��$�L��H�l$0H9��L�D$HI���%H��$�H��L���H�T$8H�T$(H��$�H��$�H��H���H��$�H��$�H��$�H��$�L9���H�4$H�L$H��$�H�l$L�L$�L��$�L�D$HH�T$8�\$ ��tSH��$�H��$�H��$�H��$�H�\$(H��$�HDŽ$HDŽ$��H����I��H��H�l$0H9������H�H�+H�l$`H�kH�l$hH�D$PH�D$XH�H�$�H�L$H�L$@H� $H�<$��H�\$`H�\$H�\$hH�\$�H�\$@H�\$@H� 1�H9�t[H�T$@H�L$pH�T$xHDŽ$�HDŽ$�HDŽ$�����H�L$PH��$H�T$XH��$��H����H�H�$H�H�\$H�H�\$�H�L$�s����%�6���A�������������H���É%�.����%����( +00runtime.morestack_noctxt�*sync.(*RWMutex).RLock�4sync.(*RWMutex).RUnlock·f�"runtime.deferproc�� runtime.duffcopy�� runtime.duffcopy� runtime.eqstring�&runtime.deferreturn�2go.string."No such image"� .type.errors.errorString� "runtime.newobject� +4runtime.writebarrierstring� +Bgo.itab.*errors.errorString.error� &runtime.deferreturn� 0type.*errors.errorString� type.error� Bgo.itab.*errors.errorString.error�  runtime.typ2Itab� &runtime.deferreturn��$"".autotmp_0497�type.error"".autotmp_0496�0type.*errors.errorString"".autotmp_0495�type.string"".autotmp_0494�Xtype.github.com/fsouza/go-dockerclient.Image"".autotmp_0493�Ztype.*github.com/fsouza/go-dockerclient.Image"".autotmp_0492�type.int"".autotmp_0491�type.int"".autotmp_04900type.*errors.errorString"".autotmp_0489�\type.[]github.com/fsouza/go-dockerclient.Image "".~r0�type.errorerrors.text·2�type.string"".image�Xtype.github.com/fsouza/go-dockerclient.Image"".i�type.int "".~r3`type.error "".~r2Ptype.int "".~r10type.string +"".idtype.string"".s*type.*"".DockerServerF%��������Z��!�4�a!>�eS +� (}�f[2j1*/Tgclocals·c958acb0df1ea67178a15bee7623bbbdTgclocals·52c6e5e411ef106b9194437a527f5a0f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�D"".(*DockerServer).createContainer�M�MeH� %H��$����H;Aw���H��H�H�$�H�\$H��$�H��$�H����H�S@H�kHH��$�H�,$H��$�H����H�Z Sj�YYH����H�H�$H��$�H���rH�o@H�|$H��H�H��H�L$H�D$ H��$�H��$�H��$�H��$�H�H�$�H�|$H��H��� +1��H��$�H� $H�<$��H��$�H�\$H��$�H�\$�H��$�H��$�H�$H��H� H��$pH�L$H��$xH�T$�H�L$H�T$ H��$HH��H��$@tcH�$H�Y ��H�T$H�L$H��$�H�$H��$�H�\$H��$�H�T$H��$�H�L$H�D$ ����H�İ�H��$�H�kH�,$�H�D$H�H�H��$ H�KH��$(HDŽ$�HDŽ$�1�H9���1�1�H��$�H��$H��$�H��$H��tlH�H�$H�T$H�L$��\$��uHH��$�H�$H��$�H�t$H�5H�l$H��H�H�H�D$ ����H�İ�H��$�H�$H��$�H�>H���1H���H�|$H�H��H�L$(H�\$0H��$XH��H��$PtcH�$H�Y ��H�T$H�L$H��$�H�$H��$�H�\$H��$�H�T$H��$�H�L$H�D$ ����H�İ�H�H�$H�D$�H�\$H�\$hH��$�H�H�kxH��$�1��H�H�$H�l$H��$�H�\$�H��$�H��$�1�H9��pH��$�H���H�H�kH��$�H��$�H��$�H��$��H�$H��H��?H%��H�H����H)�H�$�H�\$H��$�H�\$H��$�H�H�$�H�|$H���o H�-H��H���H���L H��H��H��$�H��$�H��$�H��$�H�$H�$H��$�H�\$H��$�H�\$�H�H�$H�\$hH�\$H��$�H�\$H��$�H�\$�H��$�H�$�H��$�H��$�1�H9������HDŽ$HDŽ$HDŽ$�HDŽ$�HDŽ$�H�.H���H���n H�.H���Y H���H���H���H��$(H��$H��H��$ � H�)H��$H�iH��$H��$�H�$�H�L$H�D$H��$8H�� �� +H��$�H�+H�,$H�<$�� +H��$0H�� H��$�H�L$H��$�H�D$�H�H�$�H�\$H��$��H�$H��$��\$��$�H�\$H��$��H�$H�\$@�H�$H��$x�\$��$�H�\$H��$��H�4$H��I���S㥛� H��I��H��H��H��?H)�H��Hi��H��H)�H��H��H�\$HH��$�H�H�CH��$�H���� H��H��H��$HH��$PH��$XH�H�$H�\$HH�\$�H�D$H�l$H��$HH�$H��$pH�D$H��$xH�l$�H�H�,$H��H��H�H�H��$HH�\$H��$PH�\$H��$XH�\$ �L�T$@L��$�H�\$(H��$�H�t$0H��$�H�5H��$(H���H��$H��$HH��$H��$PH��$0H��$(H��$8H��$0H��$�H��$8��$���$@H��$�H��$HH��$H��$PH��$H��$XH��$�H��$`H��$�H��$hH��$�H��$pI�H��$xI�YH��$xL��I� ���k�)L��I��H��H�� H��?H)�H��Hi�P�L��H)�H��$�H��$xH��$���$���$�H��$�H��$�I�9H���EH���H��$�H�H�H�H�$�H�|$H��H��� 1��H��$�H� $H�<$��H��$�H�\$H��$�H�\$�H��$�H�@H�h(H�H��H��H�H�H�h8H�H��H��H�H�H�$H�<$�kH�$PH�\$hH�\$�H��$�H��$�H�H�$H��$�H�\$H��$(H�\$�H��$�H�$H�<$��H�$H�H��$�H��(H����H��$�H����H� H�CH�kH��$p1�H��$hH�D$PH��$`H��$�H�l$PH9��vH��$�H�+H��$�H�T$@H��$�H�\$xH�\$xH���CH�� H��$�H��(H�� H��$�H��(H��$�H��$�H��$�H9���H�4$H�L$H��$�H�l$H��$�H�l$�H��$�H�T$@�\$ ����H��$�H�$H�<$tqH�$HH� Qj�YYH��uHH��$�H�$H��$�H�t$H�5H�l$H��H�H�H�D$ ����H�İÐ�H�İÉ%�H��$�H��H��$�H��H�l$PH9������H��$�H��$�H����H�H�KH�kH��$`H��$0H��$hH��$8H��$pH��$@H��$@H��$8H)�H��}gH�H�$H��$0H�\$H��$8H�\$H��$@H�\$H�D$ �H�\$(H��$0H�\$0H��$8H�\$8H��$@H��$8H��H��H��$@H��$0H��$0H��$8H��$@H��$0H��H�$H��$�H�\$�H��$0H��$8H��$@H��$�H�$H�<$��H��$`H�T$H��$hH�L$H��$pH�D$�H��$�H�$H�<$�CH�$H�H�D$�H��$�H�$H��$�H�[0��H��$�H�\$`H��$�H��$�H��$�H�\$pH�l$`H��81�H9�t;H�\$pH��$�H�H�$H�\$`H��8H�l$H��$�H�\$�H��$`H�H�CH��$�H�/H��$`H��H�H�H�H�$H��$�H�\$H��$�H�\$�H�\$H��$�H�\$ H��$�H��$�H��$�H��$�H��$�H�D$XH�H�$�H�\$H��$�H��$�H����1��H��$�H�$H�<$��H��$�H�\$H��$�H�\$�H��$�H��$�H��$�H�t$XH��$`H��$�H��H�H�H�H�$H��$�H�\$�H�\$H�l$H��H��H�H�H�\$XH�$���H�İÉ%�T�����/����%�����%�b���������������;����%������%�����%�����������������p����%�J���� � �E����H�.H���H�������H�.H����H���H���H���H��$(H��$H��H��$ vuH�)H��$H�iH��$H�.H���H�.H���H��r=H�H���H��H��H��H��H��tH��H��$�H��$�H��$��+���� � �E�I�������������������������HDŽ$HDŽ$HDŽ$H�H�$H�D$H��$�H�T$H��$�H�L$�H�D$ �\$(H��tPH�0H��$H�PH��$H�hH��$��tH��tH��v H�H�N�r���� 1�1��b����묉%�����������������H�İÉ�>��������� +00runtime.morestack_noctxtP�type.struct { *github.com/fsouza/go-dockerclient.Config; HostConfig *github.com/fsouza/go-dockerclient.HostConfig }b"runtime.newobject�"runtime.deferproc�type.io.Reader�runtime.convI2I�4type.encoding/json.Decoder�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface��type.*struct { *github.com/fsouza/go-dockerclient.Config; HostConfig *github.com/fsouza/go-dockerclient.HostConfig }�>encoding/json.(*Decoder).Decode� +�net/http.Error�&runtime.deferreturn�(net/url.(*URL).Query� go.string."name"� +"".nameRegexp� +8regexp.(*Regexp).MatchString� Dgo.string."Invalid container name"� net/http.Error� &runtime.deferreturn� 8"".(*DockerServer).findImage� +�net/http.Error�&runtime.deferreturn��type.map[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding�runtime.makemap�� runtime.duffzero�rtype.map[github.com/fsouza/go-dockerclient.Port]struct {}�&runtime.mapiterinit�math/rand.Int�strconv.Itoa�jtype.[1]github.com/fsouza/go-dockerclient.PortBinding�"runtime.newobject�""".statictmp_0534�� runtime.duffcopy�4runtime.writebarrierstring��type.map[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding�$runtime.mapassign1�&runtime.mapiternext�:"".(*DockerServer).generateID�4runtime.writebarrierstring�`type.github.com/fsouza/go-dockerclient.Container�"runtime.newobject�time.Now�math/rand.Int�time.Now�math/rand.Int�type.int�runtime.convT2E� 2runtime.writebarrieriface� 0go.string."172.16.42.%d"�!fmt.Sprintf�"""".statictmp_0542�"� runtime.duffcopy�(ltype.github.com/fsouza/go-dockerclient.NetworkSettings�("runtime.newobject�(� runtime.duffzero�)4runtime.writebarrierstring�).go.string."172.16.42.1"�*&go.string."docker0"�*.runtime.writebarrierptr�+`type.github.com/fsouza/go-dockerclient.Container�+.runtime.writebarrierfat�,(sync.(*RWMutex).Lock�0 runtime.eqstring�12sync.(*RWMutex).Unlock·f�1"runtime.deferproc�2lgo.string."there's already a container with this name"�2net/http.Error�2&runtime.deferreturn�2&runtime.deferreturn�5ftype.[]*github.com/fsouza/go-dockerclient.Container�6"runtime.growslice�8.runtime.writebarrierptr�92runtime.writebarrierslice�:,sync.(*RWMutex).Unlock�; +�<ptype.chan<- *github.com/fsouza/go-dockerclient.Container�<"runtime.chansend1�=type.io.Writer�>runtime.convI2I�?4type.encoding/json.Encoder�?"runtime.newobject�@� runtime.duffzero�A2runtime.writebarrieriface�A2type.struct { ID string }�Bruntime.convT2E�B>encoding/json.(*Encoder).Encode�B&runtime.deferreturn�E$runtime.panicslice�E$runtime.panicindex�H$runtime.panicslice�H$runtime.panicindex�I&type.net/url.Values�J4runtime.mapaccess2_faststr�K$runtime.panicindex�L&runtime.deferreturn@��"".autotmp_0560�6type.*encoding/json.Encoder"".autotmp_05596type.*encoding/json.Encoder"".autotmp_0558� type.io.Writer"".autotmp_0557btype.*github.com/fsouza/go-dockerclient.Container"".autotmp_0556type.uint64"".autotmp_0555type.uint64"".autotmp_0554type.int"".autotmp_0553type.int"".autotmp_0552� ftype.[]*github.com/fsouza/go-dockerclient.Container"".autotmp_0551ftype.[]*github.com/fsouza/go-dockerclient.Container"".autotmp_0550btype.*github.com/fsouza/go-dockerclient.Container"".autotmp_0549� type.string"".autotmp_0548type.string"".autotmp_0547�btype.*github.com/fsouza/go-dockerclient.Container"".autotmp_0546�dtype.**github.com/fsouza/go-dockerclient.Container"".autotmp_0545type.int"".autotmp_0544type.int"".autotmp_0543�ntype.*github.com/fsouza/go-dockerclient.NetworkSettings"".autotmp_0541"type.interface {}"".autotmp_0539� &type.[]interface {}"".autotmp_0538type.uint64"".autotmp_0531�6type.*encoding/json.Decoder"".autotmp_05306type.*encoding/json.Decoder"".autotmp_0529� type.io.Reader"".autotmp_0528� 2type.struct { ID string }"".autotmp_0527�btype.*github.com/fsouza/go-dockerclient.Container"".autotmp_0526ftype.[]*github.com/fsouza/go-dockerclient.Container"".autotmp_0525� ftype.[]*github.com/fsouza/go-dockerclient.Container"".autotmp_0524�`type.github.com/fsouza/go-dockerclient.Container"".autotmp_0523type.string"".autotmp_0522�type.int"".autotmp_0521type.int"".autotmp_0520� (type.[1]interface {}"".autotmp_0519�type.time.Time"".autotmp_0518type.int"".autotmp_0517�type.time.Time"".autotmp_0516type.string"".autotmp_0515type.int"".autotmp_0514type.int"".autotmp_0513�htype.[]github.com/fsouza/go-dockerclient.PortBinding"".autotmp_0512type.string"".autotmp_0511type.int"".autotmp_0510� Vtype.github.com/fsouza/go-dockerclient.Port"".autotmp_0509�|type.map.iter[github.com/fsouza/go-dockerclient.Port]struct {}"".autotmp_0507type.string"".autotmp_0505�type.int"".autotmp_0504type.string"".autotmp_0502� +type.string"".&container�btype.*github.com/fsouza/go-dockerclient.Container"".&config��type.*struct { *github.com/fsouza/go-dockerclient.Config; HostConfig *github.com/fsouza/go-dockerclient.HostConfig } "".~r0�6type.*encoding/json.Encoder$encoding/json.w·2�type.io.Writer"".container�btype.*github.com/fsouza/go-dockerclient.Container"".s�*type.*"".DockerServer "".~r0�type.stringnet/url.vs·4� +type.[]stringnet/url.key·3�type.string$encoding/json.r·2�type.io.Reader"".c� 2type.struct { ID string }"".c�btype.*github.com/fsouza/go-dockerclient.Container"".generatedID� type.string"".args�type.[]string"".path�type.string"".port�Vtype.github.com/fsouza/go-dockerclient.Port"".ports��type.map[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding "".err� type.error"".name�type.string "".err� type.error"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer�%�Z����������L�� ���������&��%O�Uw2:VU#t F�,$[ Z.%))��(�':.)w�.:%�!#x1�   ' ! WR  � )�0QwO=-k�DD-oJYQ�E�d")7�v�O]:!�B)�,[�[S\Q"9�UWTgclocals·d1e6514bc516778716e9d38209cf4ab8Tgclocals·37832594314d999d6d2b5ee19691d36f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�:"".(*DockerServer).generateID��eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$�H�H�$�H�l$H�l$HH���@H��H��H�l$pH�,$H�T$xH�T$H��$�H�L$�H�t$HH�l$8H��H�H�H�\$`H�H�CH�\$`H����H��H��H��$�H��$�H��$�H�H�$H�\$8H�\$�H�L$H�D$H��$�H�$H�L$PH�L$H�D$XH�D$�H�H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�L$(H�D$0H��$�H��$�H�ĠÉ�'����E���� +*0runtime.morestack_noctxtztype.[16]uint8�"runtime.newobject� crypto/rand.Read�type.[16]uint8�runtime.convT2E�2runtime.writebarrieriface�go.string."%x"�fmt.Sprintf0�"".autotmp_0587�"type.interface {}"".autotmp_0585/&type.[]interface {}"".autotmp_0582�type.[16]uint8"".autotmp_0581(type.[1]interface {}"".&buf�type.*[16]uint8 "".~r0type.string"".s*type.*"".DockerServer"������:B�EGtp@Tgclocals·1ee14e32cec51f1cde6c2b0577d81887Tgclocals·80320eec1018401d2b0daec3b250b99e�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�D"".(*DockerServer).renameContainer��eH� %H�D$�H;Aw���H���H��$H�$�H�L$H�H�3H�kH�H�$H�L$H��$�H�t$H��$�H�l$�H�\$ H���H�H�kH��$�H�$H�T$pH�T$H�l$xH�l$�H�\$H�\$@H�\$ H�\$8H�L$(H�\$0H��$�H��H��$�tcH�$H�Y ��H�T$H�L$H��$�H�$H��$�H�\$H��$�H�T$H��$�H�L$H�D$ ����H����H�H�$�H�D$H�H�$H�D$HH�D$H�\$@H�\$H�|$� �H��$H�kH�,$�H�D$H�H�H�T$`H�KH�L$hH�D$PH�D$X1�H9���1�1�H�\$HH�$H�$ H�T$PH�T$H�L$XH�L$�H��$�H�$H�<$��H�$H�H��$�H�$H�<$��H�$HH� Qj�YYH���SH��$�H���:H� H�CH�kH��$�H��$�H�l$8H��$�H9��H��H�+H����H�uH��$�H�MH�\$HH�H��$�H�CH��$�H��$�H9�utH�4$H�L$H�T$H�D$��\$ ��tRH��$�H��t|H� H�CH�kH��$�H��$�H�l$8H��$�H9�sHH��H�$H�\$HH�\$�H�D$�H��$�H�$H��$�H�[0�Ӑ�H����� �뀉E����� �������H���É%�t����%�G���HDŽ$�HDŽ$�HDŽ$�H�H�$H�D$H��$�H�T$H��$�H�L$�H�D$ �\$(H��tPH�0H��$�H�PH��$�H�hH��$���tH��tH��v H�H�N�c���� 1�1��S����묉%�����������< +*0runtime.morestack_noctxt^�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Varsvgo.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�@"".(*DockerServer).findContainer� +�net/http.Error�&runtime.deferreturn�`type.github.com/fsouza/go-dockerclient.Container�"runtime.newobject�`type.github.com/fsouza/go-dockerclient.Container�.runtime.writebarrierfat�(net/url.(*URL).Query� go.string."name"�4runtime.writebarrierstring�(sync.(*RWMutex).Lock� 2sync.(*RWMutex).Unlock·f� "runtime.deferproc�  runtime.eqstring� .runtime.writebarrierptr� +�&runtime.deferreturn�$runtime.panicindex�$runtime.panicindex�&runtime.deferreturn�&type.net/url.Values�4runtime.mapaccess2_faststr�$runtime.panicindex@�""".autotmp_0604type.string"".autotmp_0603type.string"".autotmp_0600type.string"".autotmp_0598type.string"".autotmp_0597�type.string"".autotmp_0596type.string"".©�btype.*github.com/fsouza/go-dockerclient.Container "".~r0�type.stringnet/url.vs·4_type.[]stringnet/url.key·3�type.string "".err�type.error"".index�type.int"".container�btype.*github.com/fsouza/go-dockerclient.Container +"".id�type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServerH"��������.���� \�"iKUE}!6�R#  � $.�5��U4Tgclocals·bc335ce91c3a8b5f426dd201465802bdTgclocals·5699c890da9a4c1a61d89d978591d077�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�F"".(*DockerServer).inspectContainer� +� +eH� %H�D$�H;Aw���H��H��$�H�$�H�D$H�H�H�kH�H�$H�D$H��$�H�T$H��$�H�l$�H�\$ H���H� H�kH��$�H�$H�L$XH�L$H�l$`H�l$�H�\$H�\$8H�D$(H�\$0H�\$pH��H�D$ht]H�$H�X ��H�L$H�D$H��$�H�$H��$�H�\$H��$�H�L$H��$�H�D$H�D$ ��H�Ĩ�H��$�H�$H��$�H�[ ��H�t$H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H��H�D$�H��$�H�$H��$�H�[0��H�H�$H��$�H�\$H��$�H�\$�H�L$H�D$ H��$�H�L$HH��$�H�D$PH�H�$�H�L$H��H��tu1��H�L$@H� $H�<$tUH�\$HH�\$H�\$PH�\$�H�L$@H�D$8H� $H��H�H�D$xH�D$H��$�H�L$�H�ĨÉ%뢉뇉�����, +*0runtime.morestack_noctxt^�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Varsvgo.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�@"".(*DockerServer).findContainer� +�net/http.Error� +�0go.string."Content-Type"�8go.string."application/json"�&net/http.Header.Set� +�type.io.Writer�runtime.convI2I�4type.encoding/json.Encoder�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface� btype.*github.com/fsouza/go-dockerclient.Container� >encoding/json.(*Encoder).Encode@�"".autotmp_0615�6type.*encoding/json.Encoder"".autotmp_06146type.*encoding/json.Encoder"".autotmp_0613?type.io.Writer"".autotmp_0610type.string"".autotmp_0608type.string$encoding/json.w·2�type.io.Writer "".errtype.error"".container�btype.*github.com/fsouza/go-dockerclient.Container +"".id�type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer("�������#�6�"i> UQ#� .��>40Tgclocals·cec9627e2837f98af62e9c7580b3baccTgclocals·04f43ee17c64d5db43a23c286d1bf236�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�B"".(*DockerServer).statsContainer��eH� %H��$����H;Aw���H�� H��$� H�$�H�D$H�H�H�kH�H�$H�D$H��$PH�T$H��$XH�l$�H�\$ H����H� H�kH��$� H�$H��$ H�L$H��$(H�l$�H�D$(H�\$0H��$8H��H��$0t]H�$H�X ��H�L$H�D$H��$� H�$H��$� H�\$H��$PH�L$H��$XH�D$H�D$ ��H�ĸ �H��$� H�kH�,$�H�D$H�H�H��$H�KH��$HDŽ$�HDŽ$�1�H9���1�1�H��$�H�$H��$�H�D$��\$��$�H��$ H��$(H�H�$H��$� H��H�l$H��$PH�T$H��$XH�D$�H�\$ H�+H��$�H��$� H�$H��$� H�[ ��H�t$H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H��H�D$�H��$� H�$H��$� H�[0��H�H�$H��$� H�\$H��$� H�\$�H�T$H�D$ H��$@H��$H��$HH��$H�H�$�H�|$H��H���C1��H��$�H�$H�<$�H��$H�\$H��$H�\$�H��$�H��$�H��$�H��$x1��1�H9�tQH��$ H�$H��$(H�\$H���H�\$H��$�H��H���H��$�H��$xH��H���H��$xH��$8H��H���H�H�$H��$8H�\$�H�\$H�l$H��H��H�H�H��$�H�$�H��$���$��&���H�ĸ É%����������HDŽ$`HDŽ$hHDŽ$pH�H�$H�D$H��$PH�T$H��$XH�L$�H�D$ �\$(H��tPH�0H��$`H�PH��$hH�hH��$p��tH��tH��v H�H�F����� 1�1������묉�x���H +00runtime.morestack_noctxtd�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Vars|go.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�@"".(*DockerServer).findContainer� +�net/http.Error�(net/url.(*URL).Query�$go.string."stream"�"strconv.ParseBool��type.map[string]func(string) github.com/fsouza/go-dockerclient.Stats�4runtime.mapaccess1_faststr� +� 0go.string."Content-Type"� 8go.string."application/json"� &net/http.Header.Set� + +� +type.io.Writer� +runtime.convI2I� 4type.encoding/json.Encoder� "runtime.newobject� � runtime.duffzero� 2runtime.writebarrieriface� � runtime.duffzero� +�� runtime.duffcopy�� runtime.duffcopy�� runtime.duffcopy�Xtype.github.com/fsouza/go-dockerclient.Stats�runtime.convT2E�>encoding/json.(*Encoder).Encode�&type.net/url.Values�4runtime.mapaccess2_faststr�$runtime.panicindex@�.,"".autotmp_0633�#6type.*encoding/json.Encoder"".autotmp_06326type.*encoding/json.Encoder"".autotmp_0631�!type.io.Writer"".autotmp_0629�Xtype.github.com/fsouza/go-dockerclient.Stats"".autotmp_0628� +Xtype.github.com/fsouza/go-dockerclient.Stats"".autotmp_0625type.string"".autotmp_0623type.string"".autotmp_0621type.string"".autotmp_0619�!type.string$encoding/json.w·2�"type.io.Writer "".~r0�#type.stringnet/url.vs·4�!type.[]stringnet/url.key·3�"type.string"".stats� Xtype.github.com/fsouza/go-dockerclient.Stats"".encoder�#6type.*encoding/json.Encoder"".callback�#rtype.func(string) github.com/fsouza/go-dockerclient.Stats"".stream�#type.bool "".err�"type.error +"".id�"type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer*%�.��.�.��.�.�� +Z�%i=U�^Q#�Qe� :1�-e��OOh� UTgclocals·7a383875e23784cb158d762414ce6278Tgclocals·bdc1cfaf863af97c7b8d007001384e8a�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�>"".(*DockerServer).topContainer��eH� %H��$����H;Aw���H��H��$�H�$�H�D$H�H�H�kH�H�$H�D$H��$�H�T$H��$�H�l$�H�\$ H���KH� H�kH��$�H�$H�L$xH�L$H��$�H�l$�H��$�H��$�H�|$H�D$(H�\$0H��$�H��H��$�t]H�$H�X ��H�L$H�D$H��$�H�$H��$�H�\$H��$�H�L$H��$�H�D$H�D$ ��H�Đ�H�|$X�_X���_H�D$�H�$H�^0��H�\$xH��$�H��$�H��$�H��$�H�H�CH��$�H����H��H��H��$H��$ H��$(H�H�$H��$�H�\$�H�L$H�D$H��$H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$H��$�H�\$�H�\$H�,$H��H��H�H�H�H�l$H��H��H�H�H��$H�\$ H��$ H�\$(H��$(H�\$0�H�ĐÉ�����H�$H�^ ��H�t$H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H��H�D$�H��$�H�$H��$�H�[0��H�|$XH����H�o8H�<$H��H�H�H�H�H�l$H��H��H�H��H�\$(H��$�H�\$0H��$�H��$01��H�H�$�H�|$H���$H�-H��H���H���H��H��H��$H��$0H��$H��$8H��$H��$@H�H�$�H�\$H����H��H��H��$�H��$�H��$�H�H�$�H�|$H���QH�-H��H���H���.H��H��H��$H��$H��$H�|$XH����H�w(H�<$H�H�H�H�l$H��H��H�H�H��$�H�\$ H��$�H�\$(�H�\$0H�l$H��H��H�H�H��$H��pH�$�H��$�H�$H��$H�\$H��$H�\$H��$H�\$�H��$�H��$HH��$�H��$PH��$�H��$XH�H�$H��$�H�\$H��$�H�\$�H�T$H�D$ H��$�H�T$hH��$�H�D$pH�H�$�H�|$H��H����1��H�T$`H�$H�<$��H�\$hH�\$H�\$pH�\$�H�t$`H�t$PH��$0H��$`H���H�H�$H��$`H�\$�H�\$H�l$H��H��H�H�H�\$PH�$�H�ĐÉ%�r�����P���������������������\�����������������Z��������b +00runtime.morestack_noctxtd�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Vars|go.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�@"".(*DockerServer).findContainer� +�net/http.Error� +�type.string�runtime.convT2E�2runtime.writebarrieriface�type.io.Writer� runtime.convI2I� Ngo.string."Container %s is not running"� +fmt.Fprintf� +� 0go.string."Content-Type"� 8go.string."application/json"� &net/http.Header.Set� +� go.string." "� strings.Join� � runtime.duffzero�type.[8]string�"runtime.newobject�""".statictmp_0647�� runtime.duffcopy� type.[1][]string�"runtime.newobject�type.[8]string�"runtime.newobject�""".statictmp_0652�� runtime.duffcopy�go.string." "�*runtime.concatstring3�4runtime.writebarrierstring�2runtime.writebarrierslice�type.io.Writer�runtime.convI2I�4type.encoding/json.Encoder�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface�� runtime.duffcopy�`type.github.com/fsouza/go-dockerclient.TopResult�runtime.convT2E�>encoding/json.(*Encoder).Encode@�0"".autotmp_0656�6type.*encoding/json.Encoder"".autotmp_06556type.*encoding/json.Encoder"".autotmp_0654�type.io.Writer"".autotmp_0653type.*[8]string"".autotmp_0651type.[]string"".autotmp_0649�type.[][]string"".autotmp_0646�type.[]string"".autotmp_0645�"type.interface {}"".autotmp_0643�&type.[]interface {}"".autotmp_0642_`type.github.com/fsouza/go-dockerclient.TopResult"".autotmp_0641type.string"".autotmp_0639�type.string"".autotmp_0638�(type.[1]interface {}"".autotmp_0637type.string"".autotmp_0635�type.string "".~r0�6type.*encoding/json.Encoder$encoding/json.w·2�type.io.Writer"".result�`type.github.com/fsouza/go-dockerclient.TopResult "".err�type.error"".container�btype.*github.com/fsouza/go-dockerclient.Container +"".id�type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer6%����������R� ~�%iOU�A#UI0E�80 �R1�Bx�����aZ4F?"_Tgclocals·a484a676faa0084ad5f98b43c17e101cTgclocals·950db48493931155c4d72d2be7776567�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�B"".(*DockerServer).startContainer��eH� %H�D$�H;Aw���H���H��$�H�$�H�L$H�H�3H�kH�H�$H�L$H��$�H�t$H��$�H�l$�H�\$ H���H�H�kH��$�H�$H�T$hH�T$H�l$pH�l$�H�\$H�\$8H�L$(H�\$0H��$�H��H�L$xtcH�$H�Y ��H�T$H�L$H��$�H�$H��$�H�\$H��$�H�T$H��$�H�L$H�D$ ����H����H��$�H�$H�<$�CH�$H�H��$�H�$H�<$�H�$HH� Qj�YYH����H��$�H����H�S@H�kHH��$�H�,$H��$�H����H�Z Sj�YYH���~H�H�$�H�\$H�\$PH�H�$H��$�H���@H�o@H�|$H��H�H��H�L$H�D$ H��$�H�L$XH��$�H�D$`H�H�$�H�L$H��H����1��H�L$@H� $H�<$��H�\$XH�\$H�\$`H�\$�H�T$@H�L$PH�$H��H� H��$�H�L$H��$�H�T$�H�T$H�L$ H��$�H��H�T$xtcH� $H�Z ��H�T$H�L$H��$�H�$H��$�H�\$H��$�H�T$H��$�H�L$H�D$ ����H����H�\$8H�$H�<$��H�$PH�\$PH�\$�H�T$8�ZX��tHH��$�H�$H��$�H�t$H�5H�l$H��H�H�H�D$ ����H����H��@�jXH��$�H��81�H9�t+H�T$HH�H�$H��8H�l$H�\$HH�\$���H���É%�-����%�=���������������H���É�V�����)�����H���É%������%����������H +*0runtime.morestack_noctxt^�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Varsvgo.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�@"".(*DockerServer).findContainer� +�net/http.Error�&runtime.deferreturn�(sync.(*RWMutex).Lock�2sync.(*RWMutex).Unlock·f�"runtime.deferproc�"runtime.deferproc�btype.github.com/fsouza/go-dockerclient.HostConfig�"runtime.newobject�type.io.Reader�runtime.convI2I� 4type.encoding/json.Decoder� "runtime.newobject� � runtime.duffzero� +2runtime.writebarrieriface� +dtype.*github.com/fsouza/go-dockerclient.HostConfig� >encoding/json.(*Decoder).Decode� +� net/http.Error� &runtime.deferreturn�.runtime.writebarrierptr�Jgo.string."Container already running"�net/http.Error�&runtime.deferreturn�ptype.chan<- *github.com/fsouza/go-dockerclient.Container�"runtime.chansend1�&runtime.deferreturn�&runtime.deferreturn�&runtime.deferreturn@�"".autotmp_0679�6type.*encoding/json.Decoder"".autotmp_06786type.*encoding/json.Decoder"".autotmp_0677?type.io.Reader"".autotmp_0676�btype.*github.com/fsouza/go-dockerclient.Container"".autotmp_0675type.string"".autotmp_0672type.string"".autotmp_0670type.string"".&hostConfig�dtype.*github.com/fsouza/go-dockerclient.HostConfig$encoding/json.r·2�type.io.Reader "".err�type.error"".container�btype.*github.com/fsouza/go-dockerclient.Container +"".id�type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServerz"����EF������Y��3����(� r�"iA U!6O� U0 : A  :.��4F7*�HZ + �Tgclocals·f3828558443ce662a87feff12c09632bTgclocals·9cd0f1c7734d56b3c926d71ae19f8ec3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�@"".(*DockerServer).stopContainer� � eH� %H;aw���H��xH��$�H�$�H�L$H�H�3H�kH�H�$H�L$H�t$hH�t$H�l$pH�l$�H�\$ H����H�H�kH��$�H�$H�T$HH�T$H�l$PH�l$�H�\$H�\$8H�L$(H�\$0H�\$`H��H�L$XtZH�$H�Y ��H�T$H�L$H��$�H�$H��$�H�\$H�T$hH�T$H�L$pH�L$H�D$ ����H��x�H��$�H�$H�<$�H�$H�H��$�H�$H�<$��H�$HH� Qj�H��$�H��$�YYH����H�l$8�]X��u8H�4$H�T$H�H�l$H��H��H�H�H�D$ ����H��x�H�D$�H�$H�^0��H�T$81�@�jXH��$�H��81�H9�t+H�T$@H�H�$H��8H�l$H�\$@H�\$���H��xÐ�H��xÉ%� +����%����������* + 0runtime.morestack_noctxtN�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Varsfgo.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�@"".(*DockerServer).findContainer� +�net/http.Error�&runtime.deferreturn�(sync.(*RWMutex).Lock�2sync.(*RWMutex).Unlock·f�"runtime.deferproc�Bgo.string."Container not running"�net/http.Error�&runtime.deferreturn� +�ptype.chan<- *github.com/fsouza/go-dockerclient.Container�"runtime.chansend1�&runtime.deferreturn�&runtime.deferreturn@�"".autotmp_0687obtype.*github.com/fsouza/go-dockerclient.Container"".autotmp_0686type.string"".autotmp_0684type.string "".err?type.error"".containerbtype.*github.com/fsouza/go-dockerclient.Container +"".id_type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServerN����EN��i�� +��.�L�c> O !F- A  "&��!IDTgclocals·bc335ce91c3a8b5f426dd201465802bdTgclocals·97d2741936c7bda613787afceb8adff3�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�B"".(*DockerServer).pauseContainer��eH� %H;aw���H��pH��$�H�$�H�L$H�H�3H�kH�H�$H�L$H�t$`H�t$H�l$hH�l$�H�\$ H����H�H�kH�\$xH�$H�T$@H�T$H�l$HH�l$�H�\$H�\$8H�L$(H�\$0H�\$XH��H�L$PtZH�$H�Y ��H�T$H�L$H��$�H�$H��$�H�\$H�T$`H�T$H�L$hH�L$H�D$ ����H��p�H�\$xH�$H�<$��H�$H�H�\$xH�$H�<$��H�$HH� Qj�H��$�H��$�YYH��utH�l$8�]Y��t8H�4$H�T$H�H�l$H��H��H�H�H�D$ ����H��p�H�D$�H�$H�^0��H�\$8H��@�kY��H��pÐ�H��pÉ%�J����%� �����^���& + 0runtime.morestack_noctxtN�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Varsfgo.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�@"".(*DockerServer).findContainer� +�net/http.Error�&runtime.deferreturn�(sync.(*RWMutex).Lock�2sync.(*RWMutex).Unlock·f�"runtime.deferproc�Hgo.string."Container already paused"�net/http.Error�&runtime.deferreturn� +�&runtime.deferreturn�&runtime.deferreturn@�"".autotmp_0691type.string"".autotmp_0689type.string "".err?type.error"".containerobtype.*github.com/fsouza/go-dockerclient.Container +"".id_type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServerN����?J��-�� +��'�H�c; O ?-   &�z!=Tgclocals·0b0af158856f2ab75a5e0667d877f9ebTgclocals·0a4b95df80c389fe7e338059324575e1�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�F"".(*DockerServer).unpauseContainer��eH� %H;aw���H��pH��$�H�$�H�L$H�H�3H�kH�H�$H�L$H�t$`H�t$H�l$hH�l$�H�\$ H����H�H�kH�\$xH�$H�T$@H�T$H�l$HH�l$�H�\$H�\$8H�L$(H�\$0H�\$XH��H�L$PtZH�$H�Y ��H�T$H�L$H��$�H�$H��$�H�\$H�T$`H�T$H�L$hH�L$H�D$ ����H��p�H�\$xH�$H�<$��H�$H�H�\$xH�$H�<$��H�$HH� Qj�H��$�H��$�YYH��uoH�l$8�]Y��u8H�4$H�T$H�H�l$H��H��H�H�H�D$ ����H��p�H�D$�H�$H�^0��H�\$81�@�kY��H��pÐ�H��pÉ%�O����%�%�����c���& + 0runtime.morestack_noctxtN�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Varsfgo.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�@"".(*DockerServer).findContainer� +�net/http.Error�&runtime.deferreturn�(sync.(*RWMutex).Lock�2sync.(*RWMutex).Unlock·f�"runtime.deferproc�@go.string."Container not paused"�net/http.Error�&runtime.deferreturn� +�&runtime.deferreturn�&runtime.deferreturn@�"".autotmp_0695type.string"".autotmp_0693type.string "".err?type.error"".containerobtype.*github.com/fsouza/go-dockerclient.Container +"".id_type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServerN����?J��(�� +��,�H� c; O ?-    &�z! BTgclocals·0b0af158856f2ab75a5e0667d877f9ebTgclocals·0a4b95df80c389fe7e338059324575e1�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�D"".(*DockerServer).attachContainer�*�)eH� %H��$0���H;Aw���H��PH��$pH�$�H�D$H�H�H�kH�H�$H�D$H��$H�T$H��$H�l$�H�\$ H���� H� H�kH��$XH�$H��$�H�L$H��$�H�l$�H�\$H�\$hH�D$(H�\$0H��$�H��H��$�t]H�$H�X ��H�L$H�D$H��$`H�$H��$hH�\$H��$H�L$H��$H�D$H�D$ ��H��P�H�H�$H��$`H�\$H��$hH�\$�H��$`H��$hH�\$H��$�H�\$ H��$��\$(��u5H�$H�L$H�H�l$H��H��H�H�H�D$ ��H��P�H� $H�Z ��H�t$H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H��H�D$�H��$hH�$H��$`H�[0��H��$�H�$H��$�H�[ ��H�T$H��$�H�L$H��$�H�D$ H�\$(H��$�H��H��$�t]H�$H�X ��H�L$H�D$H��$`H�$H��$hH�\$H��$H�L$H��$H�D$H�D$ ��H��P�H�H�$H�T$H�L$�H�L$H�D$ H�H�l$XH��H��H�H��$�H��$�H��$�H��$�H�\$XH�l$PH��H��H�H�H�$H�D$H�D$�H�\$H��$8H�\$ H��$@H�\$(H��$HH�H�$�H�D$H�D$xH�$H�<$��H��$�H�\$H��$�H�\$�H�D$xH���iH�hH�\$PH��H��H�H�$H�<$�=H�$H��$8H�\$H��$@H�\$H��$HH�\$�H�D$xH�l$h�]X���vH�D$`H�D$xH�H�D$p1�H9��H��$H�H�CH��$H����H��H��H��$ H��$(H��$0H�H�$H�\$hH�\$H�|$���H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�D$xH�L$pH��$�H� $H��$�H�D$H�H�l$H��H��H�H�H��$ H�\$ H��$(H�\$(H��$0H�\$0�H�\$`H�\$xH�H�+H��$�H�kH��$�H�H�D$p1�H9���H��$H�H�CH��$H���VH��H��H��$ H��$(H��$0H�H�$H��$�H�\$�H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�D$xH�L$pH��$�H� $H��$�H�D$H��$ H�\$H��$(H�\$H��$0H�\$ �H�\$`H�\$xH�H�+H��$�H�kH��$�H�H�D$p1�H9��H��$H�H�CH��$H����H��H��H��$ H��$(H��$0H�H�$H��$�H�\$�H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�D$xH�L$pH��$�H� $H��$�H�D$H��$ H�\$H��$(H�\$H��$0H�\$ �H��$�H�$H��$�H�[ ��H��PÉ����H�H�$H�H�\$H�H�\$�H�\$H�\$p���������H�H�$H�H�\$H�H�\$�H�\$H�\$p�C����%�Y����� ���H�H�$H�H�\$H�H�\$�H�\$H�\$p����H�D$`H�D$xH�H�D$p1�H9��'H��$H�H�CH��$H����H��H��H��$ H��$(H��$0H�H�$H�\$hH�\$H�|$���H�L$H�D$H��$ H�$H��$�H�L$H��$�H�D$�H�D$xH�L$pH��$�H� $H��$�H�D$H�H�l$H��H��H�H�H��$ H�\$ H��$(H�\$(H��$0H�\$0������%�P��������H�H�$H�H�\$H�H�\$�H�\$H�\$p�����%����������%�V��������� +00runtime.morestack_noctxtd�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Vars|go.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�@"".(*DockerServer).findContainer� +�net/http.Error�,type.net/http.Hijacker�$runtime.assertI2I2�Hgo.string."cannot hijack connection"�net/http.Error� +�0go.string."Content-Type"�Zgo.string."application/vnd.docker.raw-stream"�&net/http.Header.Set� +� +� + +� net/http.Error� type.io.Writer� runtime.convI2I� �github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.Stdout� type.[]uint8� "runtime.makeslice��type.github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.StdWriter�"runtime.newobject�2runtime.writebarrieriface�2runtime.writebarrierslice��go.itab.*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.StdWriter.io.Writer�type.string�runtime.convT2E�2runtime.writebarrieriface�Jgo.string."Container %q is running\n"�fmt.Fprintf�4go.string."What happened?"��go.itab.*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.StdWriter.io.Writer�type.string�runtime.convT2E�2runtime.writebarrieriface�fmt.Fprintln�>J�J#x1�atDjr78`������4>J��@ETgclocals·fcdf49cfa428c5f0402944c8a015fd4eTgclocals·432f0e830fc39b000a597e1ecb706862�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�@"".(*DockerServer).waitContainer� � eH� %H�D$�H;Aw���H���H��$�H�$�H�D$H�H�H�kH�H�$H�D$H��$�H�T$H��$�H�l$�H�\$ H����H� H�kH��$�H�$H�L$hH�L$H�l$pH�l$�H�\$H�\$HH�D$(H�\$0H��$�H��H�D$xt]H�$H�X ��H�L$H�D$H��$�H�$H��$�H�\$H��$�H�L$H��$�H�D$H�D$ ��H����H�$@B�H��$�H�$H�<$��H�$H�H��$�H�l$H�]X����H�$H�<$�mH�$H�H�H�$H�D$�H�D$H�H�+H��$�H�kH��$�H�\$HH�khH�l$8H�H�$H�D$@H�D$H��$�H�\$H�\$8H�\$�H�H�$H��$�H�\$H��$�H�\$�H�L$H�D$ H��$�H�L$XH��$�H�D$`H�H�$�H�L$H��H��tx1��H�L$PH� $H�<$tXH�\$XH�\$H�\$`H�\$�H�L$PH�D$@H� $H��H�H��$�H�D$H��$�H�L$�H���É%량넉%����H�$H�<$tH�$H������%��%�%�����J���4 +*0runtime.morestack_noctxt^�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Varsvgo.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�@"".(*DockerServer).findContainer� +�net/http.Error�time.Sleep�*sync.(*RWMutex).RLock�.sync.(*RWMutex).RUnlock�&type.map[string]int�runtime.makemap�,go.string."StatusCode"�&type.map[string]int�$runtime.mapassign1�type.io.Writer�runtime.convI2I� 4type.encoding/json.Encoder� "runtime.newobject� � runtime.duffzero� +2runtime.writebarrieriface� &type.map[string]int� >encoding/json.(*Encoder).Encode� .sync.(*RWMutex).RUnlock@�"".autotmp_0762�6type.*encoding/json.Encoder"".autotmp_07616type.*encoding/json.Encoder"".autotmp_0760_type.io.Writer"".autotmp_0759�type.int"".autotmp_0758?type.string"".autotmp_0756type.string"".autotmp_0754type.string$encoding/json.w·2�type.io.Writer"".result�&type.map[string]int "".err�type.error"".container�btype.*github.com/fsouza/go-dockerclient.Container +"".id�type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer("�������U�V� "iA U ) +{�    *.��*4>7 6,Tgclocals·f3828558443ce662a87feff12c09632bTgclocals·7669d68cdbc54f3d43537fbce1cd730b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�D"".(*DockerServer).removeContainer��eH� %H�D$�H;Aw���H���H��$H�$�H�D$H�H�H�kH�H�$H�D$H��$�H�T$H��$�H�l$�H�\$ H���UH� H�kH�L$pH�l$xH��$H�kH�,$�H�D$H�H�H�T$`H�KH�L$hH�D$@H�D$H1�H9��>1�1�H�T$@H��$�H�L$HH��$�H��$�H�$H�\$pH�\$H�\$xH�\$�H��$�H�t$ H�L$(H�\$0H��$�H��H��$�tcH�$H�Y ��H�T$H�L$H��$�H�$H��$�H�\$H��$�H�T$H��$�H�L$H�D$ ����H����L��$�I���QI�I�@M�HL��$�H��$�H�t$8H��$�H9��H�,�H�m�]X��tAH����H��$�H�4$H�|$H�5L�D$L��H�H���\$ ���pH�D$�H��$�H�$H��$�H�[0��H��$�H�$H�<$�*H�$H�H��$�H�$H�<$��H�$HH� Qj�H��$�YYH����H�rH� +H�BH�jH�l$8H9���H��H�$H��H��H��H�H�KH�kH��$�H��$�H��$�H9�sdH��H�+H�l$�H��$�H�NH��H�VH9�r4H�>H��$�H�>H��$�H�NH��$�H�V��H����� � � ��H���É%������%�����H�H�H�kH��$�H�$H��$�H�\$H�T$PH�T$H�l$XH�l$H�D$ ����H����� A�����HDŽ$�HDŽ$�HDŽ$�H�H�$H�D$H��$�H�T$H��$�H�L$�H�D$ �\$(H��tPH�0H��$�H�PH��$�H�hH��$���tH��tH��v H�H�N�#���� 1�1������묉����> +*0runtime.morestack_noctxt^�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Varsvgo.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�(net/url.(*URL).Query�"go.string."force"�@"".(*DockerServer).findContainer� +�net/http.Error�&runtime.deferreturn�go.string."1"� runtime.eqstring� +� +(sync.(*RWMutex).Lock� +2sync.(*RWMutex).Unlock·f� +"runtime.deferproc� .runtime.writebarrierptr� &runtime.deferreturn�$runtime.panicslice�$runtime.panicindex�$runtime.panicindex�&runtime.deferreturn��go.string."Error: API error (406): Impossible to remove a running container, please stop it first"�net/http.Error�&runtime.deferreturn�$runtime.panicindex�&type.net/url.Values�4runtime.mapaccess2_faststr�$runtime.panicindex@�""".autotmp_0775type.int"".autotmp_0773type.int"".autotmp_0772type.int"".autotmp_0771type.string"".autotmp_0769type.string"".autotmp_0766type.string "".~r0�type.stringnet/url.vs·4_type.[]stringnet/url.key·3�type.string "".msg�type.string "".err�type.error"".index�type.int"".force�type.string +"".id�type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServerT"��������"��n���� h� +"smDU� +#!>o6 ; � .w}:�UTgclocals·bc335ce91c3a8b5f426dd201465802bdTgclocals·986af82aaae26ef643e71b57d814342a�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�D"".(*DockerServer).commitContainer�8�8eH� %H��$����H;Aw���H��H��$�H�kH�,$�H�D$H�H�H��$H�KH��$ H�D$hH�D$p1�H9��� 1�1�H�L$hH�D$pH��$�H�$H��$XH�L$H��$`H�D$�H�\$H�\$PH�D$(H�\$0H��$pH��H��$ht]H�$H�X ��H�L$H�D$H��$�H�$H��$�H�\$H��$�H�L$H��$�H�D$H�D$ ��H�Ġ�H�D$XH��$�H�kH�,$�H�D$H�H�H��$�H�KH��$HDŽ$�HDŽ$�1�H9�� 1�1�H��$�H��$�H��$�H��$�H����H�H�$�H�D$H�D$XH�D$`H��$�H�$H��$�H�\$�H�\$H�,$H��H��H�H�H�H�L$`H�H��$xH�D$H��$�H�L$ �H�L$(H�D$0H��$pH��H��$ht]H�$H�Y ��H�L$H�D$H��$�H�$H��$�H�\$H��$�H�L$H��$�H�D$H�D$ ��H�Ġ�H�D$�H��$�H�$H��$�H�[0��H��$�H�kH�,$�H�D$H�H�H��$8H�KH��$@HDŽ$�HDŽ$�1�H9���HDŽ$�HDŽ$�H��$�H�kH�,$�H�D$H�H�H��$H�KH��$H�D$xHDŽ$�1�H9��<H�D$xHDŽ$�H�H��$�H��H���H�H�,$H��H��H�H�H�|$PH����H�/H�|$H��H�H��H�D$PH�\$ H��$�H��H��H�H�H����H���H��$�H��H��H�H�H�(H��$�H��H��H�H�H��$�H��$�H��$�H��$�H�\$xH��$hH��$�H��$pH�\$XH��$xH��$�H�kH�,$�H�D$H�H�H��$(H�KH��$0HDŽ$�HDŽ$�1�H9��1�1�H��$�H��$�H��$�H��$�H��$�H�kH�,$�H�D$H�H�H��$HH�KH��$PHDŽ$�HDŽ$�1�H9���1�1�H��$�H��$�H��$�H��$�H��$�H�$H�<$��H�$x�H��$�H���mH�S`H�KhH�[pH��$xH��$�H��$�H��H)�H��}OH�H�$H��$`H�T$H��$hH�L$H��$pH�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$hH��$pH�H�$H��H��$`H��Hi�H�H�\$H��$�H�\$�H��$`H��$hH��$pH��$�H�$H�<$�cH�$`H��$xH�T$H��$�H�L$H��$�H�D$�H��$�H��$�H��$�H����H��tPH��$�H�$H��$�H�D$H�H�l$H��H��H�H�H��$�H�\$ H�L$(�H�T$0H�D$8H��$�H��$�H��$�H��$�H�H�$H��$�H���H�l$H��$�H�\$H��$�H�\$�H��$�H�$H�<$�9H�$x�H��$�H�H�CH��$�H����H��H��H��$HH��$PH��$XH�H�$H��$�H�\$�H�D$H�L$H��$HH�$H��$xH�D$H��$�H�L$�H�H�$H��$�H�\$H��$�H�\$�H�\$H�,$H��H��H�H�H�H�l$H��H��H�H�H��$HH�\$ H��$PH�\$(H��$XH�\$0�H�ĠÉ������%�����%����������%�d���HDŽ$HDŽ$ HDŽ$(H�H�$H�D$H��$�H�T$H��$�H�L$�H�L$ �\$(H��H��H��tOH�H��$H�IH��$ H�kH��$(<tH��tH��v H� +H�B����� 1�1��u�����HDŽ$�HDŽ$�HDŽ$�H�H�$H�D$H��$�H�T$H��$�H�L$�H�L$ �\$(H��H��H��tOH�H��$�H�IH��$�H�kH��$�<tH��tH��v H� +H�B�J���� 1�1��:����뭉�X��������HDŽ$0HDŽ$8HDŽ$@H�H�$H�D$H��$�H�T$H��$�H�L$�H�L$ �\$(H��H��H��tmH�H��$0H�IH��$8H�kH��$@<t,H��t&H��vH�*H�l$xH�jH��$��$���� H�D$xHDŽ$�������HDŽ$�HDŽ$�HDŽ$�H�H�$H�D$H��$�H�T$H��$�H�L$�H�L$ �\$(H��H��H��tsH�H��$�H�IH��$�H�kH��$�<t/H��t)H��vH�*H��$�H�jH��$������� HDŽ$�HDŽ$�������HDŽ$�HDŽ$�HDŽ$�H�H�$H�D$H��$�H�T$H��$�H�L$�H�L$ �\$(H��H��H��tOH�H��$�H�IH��$�H�kH��$�<tH��tH��v H� +H�B�P���� 1�1��@�����HDŽ$HDŽ$HDŽ$H�H�$H�D$H��$�H�T$H��$�H�L$�H�D$ �\$(H��tPH�0H��$H�PH��$H�hH��$��tH��tH��v H�H�F�u���� 1�1��e����묂 +00runtime.morestack_noctxtl(net/url.(*URL).Query�*go.string."container"�@"".(*DockerServer).findContainer� +�net/http.Error�(net/url.(*URL).Query�go.string."run"�Ztype.github.com/fsouza/go-dockerclient.Config�"runtime.newobject�2runtime.stringtoslicebyte�\type.*github.com/fsouza/go-dockerclient.Config� .encoding/json.Unmarshal� +� +net/http.Error� +� (net/url.(*URL).Query� go.string."m"� (net/url.(*URL).Query� $go.string."author"�""".statictmp_0817�� runtime.duffcopy� go.string."img-"�*runtime.concatstring2�(net/url.(*URL).Query� go.string."repo"�(net/url.(*URL).Query�go.string."tag"�(sync.(*RWMutex).Lock�\type.[]github.com/fsouza/go-dockerclient.Image�"runtime.growslice�Xtype.github.com/fsouza/go-dockerclient.Image�.runtime.writebarrierfat�2runtime.writebarrierslice�go.string.":"�*runtime.concatstring3�,type.map[string]string�$runtime.mapassign1� ,sync.(*RWMutex).Unlock�!type.string�!runtime.convT2E�"2runtime.writebarrieriface�"type.io.Writer�#runtime.convI2I�#.go.string."{\"ID\":%q}"�$fmt.Fprintf�&&type.net/url.Values�&4runtime.mapaccess2_faststr�($runtime.panicindex�)&type.net/url.Values�)4runtime.mapaccess2_faststr�+$runtime.panicindex�,&type.net/url.Values�,4runtime.mapaccess2_faststr�.$runtime.panicindex�/&type.net/url.Values�04runtime.mapaccess2_faststr�2$runtime.panicindex�3&type.net/url.Values�34runtime.mapaccess2_faststr�5$runtime.panicindex�6&type.net/url.Values�64runtime.mapaccess2_faststr�8$runtime.panicindex@�~"".autotmp_0828"type.interface {}"".autotmp_0826� &type.[]interface {}"".autotmp_0822type.int"".autotmp_0821�\type.[]github.com/fsouza/go-dockerclient.Image"".autotmp_0820�\type.[]github.com/fsouza/go-dockerclient.Image"".autotmp_0819type.*[]string"".autotmp_0818type.*[]string"".autotmp_0816type.*[]string"".autotmp_0815type.*[]string"".autotmp_0814type.*[]string"".autotmp_0812� (type.[1]interface {}"".autotmp_0811� type.string"".autotmp_0810type.string"".autotmp_0809\type.[]github.com/fsouza/go-dockerclient.Image"".autotmp_0808type.int"".autotmp_0807type.string"".autotmp_0806&type.net/url.Values"".autotmp_0805type.int"".autotmp_0804type.string"".autotmp_0803&type.net/url.Values"".autotmp_0802type.int"".autotmp_0801type.string"".autotmp_0800&type.net/url.Values"".autotmp_0799type.int"".autotmp_0798type.string"".autotmp_0797&type.net/url.Values"".autotmp_0796type.string"".autotmp_0794\type.*github.com/fsouza/go-dockerclient.Config"".autotmp_0793�\type.*github.com/fsouza/go-dockerclient.Config"".autotmp_0792type.int"".autotmp_0791type.string"".autotmp_0790&type.net/url.Values"".autotmp_0789type.string"".autotmp_0787� type.string "".~r0�type.stringnet/url.vs·4� +type.[]stringnet/url.key·3� type.string "".~r0�type.stringnet/url.vs·4� +type.[]stringnet/url.key·3� type.string "".~r0�type.stringnet/url.vs·4� type.[]stringnet/url.key·3�type.string "".~r0�type.stringnet/url.vs·4� type.[]stringnet/url.key·3� type.string "".~r0�type.stringnet/url.vs·4� type.[]stringnet/url.key·3�type.string "".~r0�type.stringnet/url.vs·4� +type.[]stringnet/url.key·3�type.string "".tag�type.string"".repository�type.string"".image�Xtype.github.com/fsouza/go-dockerclient.Image"".runConfig�type.string"".config�\type.*github.com/fsouza/go-dockerclient.Config "".err� type.error"".container�btype.*github.com/fsouza/go-dockerclient.Container +"".id� type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer8%�������� ��� ��� +%cGU wzU# +sm 3�!� +P^!�  ���  +  +�� ��5��-@-v�����`nh!l +�� Z�gsjvZ�Tgclocals·9a68f8414e7859befd5e7341b374c125Tgclocals·4ffc9bba9f386a0dab8c46db5f1d911a�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�@"".(*DockerServer).findContainer��eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$�HDŽ$�HDŽ$�H��$�H�$H�<$�H�$H�H��$�H�$H�<$��H�$HH� Qj�H��$�YYH����H��$�H����L�H�CH�kH��$�1�H��$�H�D$0L��$�H�l$0H9��GL�D$PI�8H�t$8H�t$(H�|$@H���AH�H�GH9���H��$�H� $H��$�H�D$H��$�H�l$H�T$�L�D$PH�|$@H�t$8H��$��\$ ��t;H��$�H�\$(H��$�HDŽ$�HDŽ$���H�İ�H��H����H�� H��$�H��(H��$�H9�uEH�<$H�L$H��$�H�l$H�T$�L�D$PH�|$@H�t$8H��$��\$ ���P���I��H��H�l$0H9������H�H�+H�l$hH�kH�l$pH�D$XH�D$`H�H�$�H�L$H�L$HH� $H�<$��H�\$hH�\$H�\$pH�\$�H�\$HH�\$HH� 1�H9�tRH�T$HH�L$xH��$�HDŽ$�HDŽ$�����H�L$XH��$�H�T$`H��$���H�İ�H�H�$H�H�\$H�H�\$�H�L$�|����%�?�����h����������X�����H�İÉ%�����%�����& +*0runtime.morestack_noctxt�*sync.(*RWMutex).RLock�4sync.(*RWMutex).RUnlock·f�"runtime.deferproc� runtime.eqstring�&runtime.deferreturn� runtime.eqstring� :go.string."No such container"� .type.errors.errorString� "runtime.newobject� +4runtime.writebarrierstring� +Bgo.itab.*errors.errorString.error� &runtime.deferreturn� 0type.*errors.errorString� type.error� Bgo.itab.*errors.errorString.error�  runtime.typ2Itab� &runtime.deferreturnp�$"".autotmp_0846otype.error"".autotmp_0845�0type.*errors.errorString"".autotmp_0844type.string"".autotmp_0843Otype.string"".autotmp_0841�dtype.**github.com/fsouza/go-dockerclient.Container"".autotmp_0840�type.int"".autotmp_0839�type.int"".autotmp_08380type.*errors.errorString"".autotmp_0837/ftype.[]*github.com/fsouza/go-dockerclient.Container "".~r0�type.errorerrors.text·2�type.string"".container�btype.*github.com/fsouza/go-dockerclient.Container"".i�type.int "".~r3Ptype.error "".~r2@type.int "".~r10btype.*github.com/fsouza/go-dockerclient.Container"".idOrNametype.string"".s*type.*"".DockerServerD"�u������`��'�:� R!>\p;u +�,n�SYt2a105Tgclocals·8bd789dcce9d4daa4c4bb84dfe47e247Tgclocals·4459bbba29917b2fee408f0dbbff89b1�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�:"".(*DockerServer).buildImage��eH� %H��$����H;Aw���H��H��$�H�~8H�<$H�5H�|$H�H��H�L$H�D$ H��$H��$�H��$H��$�H����H� $H�D$H�-L�D$L��H��H�H���\$ ���t�D$GH�H�$H��$�H����H�o@H�|$H��H�H��H�L$H�D$ H��$�H�L$xH��$�H��$�H�H�$�H�|$H��H����1��H�L$PH� $H�<$�rH�\$xH�\$H��$�H�\$�H�\$PH�\$HH�\$HH�$�H�D$H�L$H�\$H��$�H��H��$����|$GuuH�D$�H��$�H�$H��$�H�[0��H�H�,$H��H��H�H��H�\$H�l$H��H��H�H�H�H��$�H�$H��$�H�[(��H�Ĉ�H��$�H�$�H�\$H��$H�\$H��$�H�,$�T$H�L$H��$x1��H��$H��$xH��$H��$�H��$���$�H��$�H��$�H�kH�,$�H�D$H��$xH��$�H��$�H��$�H�H�H��$�H�KH��$�H�D$XH�D$`1�H9���1�1�H�L$XH�L$hH�D$`H�D$pH��tH��$�H��$�H��$�H�$H�<$�2H�$x�H��$�H���H�S`H�KhH�[pH��$`H��$hH��$pH��H)�H��}OH�H�$H��$HH�T$H��$PH�L$H��$XH�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$PH��$XH�H�$H��H��$HH��Hi�H�H�\$H��$xH�\$�H��$HH��$PH��$XH��$�H�$H�<$�H�$`H��$`H�T$H��$hH�L$H��$pH�D$�H��$�H��$�H��$�H��$H�H�$H��$�H���H�l$H��$�H�\$H��$xH�\$�H��$�H�$H�<$�SH�$x�H��$�H�H�CH��$�H���H��H��H��$0H��$8H��$@H�H�$H��$xH�\$�H�L$H�D$H��$0H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$0H�\$H��$8H�\$H��$@H�\$ �H�L$(H�D$0H��$H� $H��$H�D$�H�\$H�l$H��H��H�H�H�H��$�H�$H��$�H�[(��H�ĈÉ������%�����%������������%�����HDŽ$HDŽ$ HDŽ$(H�H�$H�D$H��$H�T$H��$H�L$�H�D$ �\$(H��tPH�0H��$H�PH��$ H�hH��$(��tH��tH��v H�H�F������ 1�1��������H��t]H�H��$H�@H��$H�� +�����H� $H�D$H�-L�D$L��H��H�H���\$ ��������D$G�����량%������`���������V +00runtime.morestack_noctxtp0go.string."Content-Type"�&net/http.Header.Get�6go.string."application/tar"� runtime.eqstring�type.io.Reader�runtime.convI2I�.type.archive/tar.Reader�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface�4archive/tar.(*Reader).Next� +�6go.string."miss Dockerfile"�2runtime.stringtoslicebyte� +�:"".(*DockerServer).generateID� time.Now� � runtime.duffzero� +(net/url.(*URL).Query� go.string."t"� (sync.(*RWMutex).Lock�\type.[]github.com/fsouza/go-dockerclient.Image�"runtime.growslice�Xtype.github.com/fsouza/go-dockerclient.Image�.runtime.writebarrierfat�2runtime.writebarrierslice�,type.map[string]string�$runtime.mapassign1�,sync.(*RWMutex).Unlock�type.string�runtime.convT2E�2runtime.writebarrieriface�Bgo.string."Successfully built %s"�fmt.Sprintf�2runtime.stringtoslicebyte� +�&type.net/url.Values�4runtime.mapaccess2_faststr�$runtime.panicindex�,go.string."Dockerfile"� runtime.eqstring@�<"".autotmp_0872� +"type.interface {}"".autotmp_0870� &type.[]interface {}"".autotmp_0866type.int"".autotmp_0865�\type.[]github.com/fsouza/go-dockerclient.Image"".autotmp_0864�\type.[]github.com/fsouza/go-dockerclient.Image"".autotmp_0862type.string"".autotmp_0861� 0type.*archive/tar.Reader"".autotmp_08600type.*archive/tar.Reader"".autotmp_0859� +type.io.Reader"".autotmp_0858type.string"".autotmp_0857� +(type.[1]interface {}"".autotmp_0856� +type.string"".autotmp_0855\type.[]github.com/fsouza/go-dockerclient.Image"".autotmp_0853type.string"".autotmp_0850type.string"".autotmp_0849� type.string "".~r0� type.stringnet/url.vs·4� type.[]stringnet/url.key·3� type.string archive/tar.r·2� type.io.Reader"".t� type.string"".repository� type.string"".image�Xtype.github.com/fsouza/go-dockerclient.Image "".err� type.error +"".tr� 0type.*archive/tar.Reader "".gotDockerFile� type.bool +"".ct� type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer*%����������� %��%#J ++F + Y!�^!�  �#Y  #RE�IVh��`^!lv�UZDTgclocals·f691ab09c838bb4c8855d6461c0f447dTgclocals·089441843964391e375901e8fc7a3c0c�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�8"".(*DockerServer).pullImage��eH� %H��$����H;Aw���H��H��$�H�kH�,$�H�D$H�H�H�T$pH�KH�L$xH�D$@H�D$H1�H9��1�1�H�L$@H��$�H�D$HH��$�H��$�H�kH�,$�H�D$H�H�H��$�H�KH��$�H�D$PH�D$X1�H9���1�1�H�L$PH�L$`H�D$XH�D$hH��$�H�$�H�T$H�L$H��$x1��H��$xH��$�H��$�H�$H�<$�yH�$x�H��$�H���VH�S`H�KhH�[pH��$@H��$HH��$PH��H)�H��}OH�H�$H��$(H�T$H��$0H�L$H��$8H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$0H��$8H�H�$H��H��$(H��Hi�H�H�\$H��$xH�\$�H��$(H��$0H��$8H��$�H�$H�<$�LH�$`H��$@H�T$H��$HH�L$H��$PH�D$�H��$�H�L$hH��$�H����H���_H��$�H��$�H�\$`H��$�H��$�H��$X1��H��$XH����H��H��H��$H��$H��$ H�H�$H��$�H�\$�H�L$H�D$H��$H�$H��$�H�L$H��$�H�D$�H�H�$H��$�H�\$�H�L$H�D$H��$H��H�$H��$�H�L$H��$�H�D$�H�H�,$H��H��H�H�H��$H�\$H��$H�\$H��$ H�\$ �H�T$(H�D$0H��$�H��$�H��$�H��$�H�H�$H��$�H���H�l$H��$�H�\$H��$xH�\$�H��$�H�$H�<$tH�$x�H�ĈÉ%���T����%����������%�{���HDŽ$�HDŽ$HDŽ$H�H�$H�D$H��$�H�T$H��$�H�L$�H�L$ �\$(H��H��H��tOH�H��$�H�IH��$H�kH��$<tH��tH��v H� +H�B�n���� 1�1��^�����HDŽ$�HDŽ$�HDŽ$�H�H�$H�D$H��$�H�T$H��$�H�L$�H�D$ �\$(H��tPH�0H��$�H�PH��$�H�hH��$���tH��tH��v H�H�F�D���� 1�1��4�����@ +00runtime.morestack_noctxtl(net/url.(*URL).Query�*go.string."fromImage"�(net/url.(*URL).Query�go.string."tag"�:"".(*DockerServer).generateID�� runtime.duffzero�(sync.(*RWMutex).Lock�\type.[]github.com/fsouza/go-dockerclient.Image�"runtime.growslice�Xtype.github.com/fsouza/go-dockerclient.Image�.runtime.writebarrierfat� +2runtime.writebarrierslice� � runtime.duffzero� type.string� runtime.convT2E� 2runtime.writebarrieriface�type.string�runtime.convT2E�2runtime.writebarrieriface�"go.string."%s:%s"�fmt.Sprintf�,type.map[string]string�$runtime.mapassign1�,sync.(*RWMutex).Unlock�&type.net/url.Values�4runtime.mapaccess2_faststr�$runtime.panicindex�&type.net/url.Values�4runtime.mapaccess2_faststr�$runtime.panicindex@�<"".autotmp_0903"type.interface {}"".autotmp_0902� "type.interface {}"".autotmp_0900� &type.[]interface {}"".autotmp_0896type.int"".autotmp_0895� \type.[]github.com/fsouza/go-dockerclient.Image"".autotmp_0894� \type.[]github.com/fsouza/go-dockerclient.Image"".autotmp_0893type.*[]string"".autotmp_0891type.string"".autotmp_0890type.string"".autotmp_0889� type.string"".autotmp_0888� type.string"".autotmp_0887�(type.[2]interface {}"".autotmp_0886\type.[]github.com/fsouza/go-dockerclient.Image"".autotmp_0885type.string"".autotmp_0884type.int"".autotmp_0883type.string"".autotmp_0882&type.net/url.Values"".autotmp_0880� +type.string "".~r0� type.stringnet/url.vs·4� +type.[]stringnet/url.key·3� type.string "".~r0� type.stringnet/url.vs·4� +type.[]stringnet/url.key·3� type.string"".image�Xtype.github.com/fsouza/go-dockerclient.Image "".tag� type.string "".fromImageName� type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer%�� ���� X� %mm!� + +�^  ��@5miJ�`��h� Z�Tgclocals·908986cc2bd23e6b2b43c6b331d27560Tgclocals·4fcfe95894f9cd95ba1d5aada184916c�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�8"".(*DockerServer).pushImage��eH� %H�D$�H;Aw���H���H��$H�$�H�D$H�H�H�kH�H�$H�D$H��$�H�T$H��$�H�l$�H�\$ H���H� H�kH�L$`H�l$hH��$H�kH�,$�H�D$H�H�H�T$pH�KH�L$xH�D$@H�D$H1�H9���1�1�H�L$@H�L$PH�D$HH�D$XH��tYH�|$`H�T$hH��$�H�<$H��$�H�T$H�H�|$H��H�H�H�L$ H�D$(�H�\$0H�\$`H�\$8H�\$hH��$�H�$H�<$�eH�$x�H�L$`H�D$hH�H�$H��$�H���H�l$H��$�H�L$H��$�H�D$�H��$�H�L$ �\$(H������u`H�$H�<$tLH�$x�H��$H�$H��$H�t$H�5H�l$H��H�H�H�D$ ��H���É%�H�$H�<$�pH�$x�H�H�+H��$�H�kH��$�H��$�H�H�CH��$�H���H��H��H��$�H��$�H��$�H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H�H�$H��$H�\$H��$H�\$�H�\$H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�H�+H��$�H�kH��$�H��$�H�H�CH��$�H����H��H��H��$�H��$�H��$�H�H�$H��$�H�\$�H�L$H�D$H��$�H�$H��$�H�L$H��$�H�D$�H�H�$H��$H�\$H��$H�\$�H�\$H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H���É�����������%������ ����%����HDŽ$�HDŽ$�HDŽ$�H�H�$H�D$H��$�H�T$H��$�H�L$�H�D$ �\$(H��tPH�0H��$�H�PH��$�H�hH��$���tH��tH��v H�H�F�b���� 1�1��R����묉�����D +*0runtime.morestack_noctxt^�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Varsv go.string."name"�,type.map[string]string�4runtime.mapaccess1_faststr�(net/url.(*URL).Query�go.string."tag"�go.string.":"�*runtime.concatstring3�*sync.(*RWMutex).RLock�,type.map[string]string�4runtime.mapaccess2_faststr�.sync.(*RWMutex).RUnlock�2go.string."No such image"�net/http.Error� .sync.(*RWMutex).RUnlock� ,go.string."Pushing..."� type.string� runtime.convT2E� 2runtime.writebarrieriface� type.io.Writer� runtime.convI2I�fmt.Fprintln�$go.string."Pushed"�type.string�runtime.convT2E�2runtime.writebarrieriface�type.io.Writer�runtime.convI2I�fmt.Fprintln�&type.net/url.Values�4runtime.mapaccess2_faststr�$runtime.panicindex@�*"".autotmp_0929"type.interface {}"".autotmp_0928*type.*[1]interface {}"".autotmp_0927&type.[]interface {}"".autotmp_0926�"type.interface {}"".autotmp_0924/&type.[]interface {}"".autotmp_0921type.string"".autotmp_0920(type.[1]interface {}"".autotmp_0919�type.string"".autotmp_0918�(type.[1]interface {}"".autotmp_0917type.string"".autotmp_0916type.string"".autotmp_0914type.string"".autotmp_0911type.string "".~r0�type.stringnet/url.vs·4_type.[]stringnet/url.key·3�type.string "".tag�type.string"".name�type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer*"��������� b� "sb Y!i: ��  �>.w�5H�����U*Tgclocals·cec9627e2837f98af62e9c7580b3baccTgclocals·a296c3305421c42d8fb7cfcda86446d7�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�6"".(*DockerServer).tagImage��eH� %H��$p���H;Aw���H��H��$0H�$�H�L$H�H�3H�kH�H�$H�L$H��$�H�t$H��$�H�l$�H�\$ H���uH�H�kH��$�H��$�H��$H�$H�<$�;H�$x�H��$�H��$�H�H�$H��$H���H�l$H��$�H�T$H��$�H�L$�H��$H�T$ �\$(H������ufH�,$H�<$tRH�$x�H��$ H�$H��$(H�t$H�5H�l$H��H�H�H�D$ ����H��É%�H�,$H�<$�:H�$x�H��$H�$H�<$� H�$x�H��$H�$H�<$��H�$xH� Qj�YYH����H��$0H�kH�,$�H�D$H�H�H��$�H�KH��$�H�D$@H�D$H1�H9���1�1�H�L$@H�L$pH�D$HH�D$xH��$0H�kH�,$�H�D$H�H�H��$�H�KH��$�H�D$PH�D$X1�H9��{1�1�H�T$PH�T$`H�L$XH�L$hH��tYH�t$pH�|$xH��$�H�4$H��$�H�|$H�H�|$H��H�H�H�T$ H�L$(�H�\$0H�\$pH�\$8H�\$xH�\$pH��$�H�\$xH��$�H��$�H��$�H�H�$H��$H���H�l$H��$�H�T$H��$�H�L$�H�\$ H����H�+H��$�H�kH��$�H�H�$H��$H���H�l$H��$�H�\$H��$�H�\$�H�D$�H��$(H�$H��$ H�[0�Ӑ�H��É�s���HDŽ$�HDŽ$HDŽ$H�H�$H�D$H��$�H�T$H��$�H�L$�H�L$ �\$(H��H��H��tRH�H��$�H�IH��$H�kH��$<t"H��tH��H��v H�H�K������ 1�1��������HDŽ$�HDŽ$�HDŽ$�H�H�$H�D$H��$�H�T$H��$�H�L$�H�D$ �\$(H��tPH�0H��$�H�PH��$�H�hH��$���tH��tH��v H�H�F����� 1�1������묐�H��É%�����%������%������9����%���������H +00runtime.morestack_noctxtd�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Vars| go.string."name"�,type.map[string]string�4runtime.mapaccess1_faststr�*sync.(*RWMutex).RLock�,type.map[string]string�4runtime.mapaccess2_faststr�.sync.(*RWMutex).RUnlock�2go.string."No such image"�net/http.Error�&runtime.deferreturn�.sync.(*RWMutex).RUnlock�(sync.(*RWMutex).Lock�2sync.(*RWMutex).Unlock·f�"runtime.deferproc�(net/url.(*URL).Query� go.string."repo"� +(net/url.(*URL).Query� +go.string."tag"� go.string.":"� *runtime.concatstring3� ,type.map[string]string�4runtime.mapaccess1_faststr�,type.map[string]string�$runtime.mapassign1� +�&runtime.deferreturn�&type.net/url.Values�4runtime.mapaccess2_faststr�$runtime.panicindex�&type.net/url.Values�4runtime.mapaccess2_faststr�$runtime.panicindex�&runtime.deferreturn@�."".autotmp_0952type.*[]string"".autotmp_0949�type.string"".autotmp_0948type.string"".autotmp_0947�type.string"".autotmp_0946type.string"".autotmp_0945type.int"".autotmp_0944type.string"".autotmp_0943&type.net/url.Values"".autotmp_0941type.string"".autotmp_0939type.string"".autotmp_0937type.string "".~r0�type.stringnet/url.vs·4/type.[]stringnet/url.key·3�type.string "".~r0�type.stringnet/url.vs·4_type.[]stringnet/url.key·3�type.string"".newTag�type.string"".newRepo�type.string"".name�type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServerF%����g������K� v� %y!o: !6mh Y�#��   B1��j��|d&e]bUmTgclocals·8375af20f91e3bf26f9f4b100ffb7d0eTgclocals·d85b7b7d175a77beb9795a1053a5aaec�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�<"".(*DockerServer).removeImage��eH� %H��$ ���H;Aw���H��`H��$�H�$�H�D$H�H�H�kH�H�$H�D$H��$�H�T$H��$�H�l$�H�\$ H���!H� H�kH��$�H��$�H��$hH�$H�<$��H�$x�H�D$XH�D$`H��$�H��$�H�H�$H��$hH���H�l$H��$�H�L$H��$�H�D$�H�L$ �\$(H���bH�H�T$xH�IH��$���t:H��$�H��$�H��$�H��$�H��$�H�l$XH��$�H�D$`HDŽ$�HDŽ$�HDŽ$�H��$hH���H��$1��H�H�$H�l$H��$H�\$�H��$�H��$1�H9���H��$H���wH� H�CH��$H���WH�+H�l$hH�kH�l$pH��$�H�L$HH��$�H�D$PH9��H� $H�D$H��$�H�l$H�T$��\$ ����H��$�H��$�H��$�H��H)�H��}OH�H�$H��$�H�T$H��$H�L$H��$H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$H��$H��H��$�H��Hk�H�H�$H�\$hH�\$H�\$pH�\$�H��$�H��$H��$H��$�H��$�H��$�H��$H�$�H��$�H��$1�H9��\���H��$hH�$H�<$��H�$x�H��$hH�$H��$�H�\$H��$�H�\$�H�\$(H�\$@H�L$0H�\$8H��$�H��H��$�tcH�$H�Y ��H�T$H�L$H��$pH�$H��$xH�\$H��$�H�T$H��$�H�L$H�D$ ����H��`�H�D$�H��$xH�$H��$pH�[0��H��$hH�$H�<$��H�$x�H��$hH�$H�<$��H�$xH� Qj�H��$xYYH���QH��$�H���=H�rhH�H�$H�Z`H�BhH�jpH�l$@H9�� Hi�H�H�\$H��H��H��H�R`H�KhH�kpH��$�H��H��$�H��H��$�H9���Hi�H�H�\$�H��$hH�JhH��H�rpH9���H�z`H��$�H�z`H��$�H�JhH��$�H�rpH�\$`H��tCH�\$XH��$�H�\$`H��$�H�H�$H���H�l$H��$�H�\$���H��`�� � � 덐�H��`É%�n����%�A����%�9�������������������%� ���������F +00runtime.morestack_noctxtd�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Vars|go.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�*sync.(*RWMutex).RLock�,type.map[string]string�4runtime.mapaccess2_faststr�� runtime.duffzero�,type.map[string]string�&runtime.mapiterinit� + runtime.eqstring� +type.[]string� "runtime.growslice� 4runtime.writebarrierstring�&runtime.mapiternext�.sync.(*RWMutex).RUnlock�@"".(*DockerServer).findImageByID� +�net/http.Error�&runtime.deferreturn� +�(sync.(*RWMutex).Lock�2sync.(*RWMutex).Unlock·f�"runtime.deferproc�Xtype.github.com/fsouza/go-dockerclient.Image�.runtime.writebarrierfat�,type.map[string]string�"runtime.mapdelete�&runtime.deferreturn�$runtime.panicslice�$runtime.panicindex�$runtime.panicindex�&runtime.deferreturn@�4"".autotmp_0975type.uint64"".autotmp_0974type.uint64"".autotmp_0973type.int"".autotmp_0968�type.[]string"".autotmp_0967type.string"".autotmp_0966type.string"".autotmp_0964�type.string"".autotmp_0963type.int"".autotmp_0962type.int"".autotmp_0961type.int"".autotmp_0960type.string"".autotmp_0958�6type.map.iter[string]string"".autotmp_0957,type.map[string]string"".autotmp_0956type.string"".autotmp_0954�type.string "".err�type.error"".index�type.int"".taggedID�type.string "".tag�type.string"".tags�type.[]string "".img�type.string "".tag�type.string +"".id�type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServerD%����h���$��M� �� %y!{:$�<�, +!GU#!>�< C    N1�`��rWA<+7|���Tgclocals·f691ab09c838bb4c8855d6461c0f447dTgclocals·ffed5bc34d491524ef9a5d2e5f55f2f5�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�>"".(*DockerServer).inspectImage��eH� %H��$����H;Aw���H���H��$H�$�H�L$H�H�3H�kH�H�$H�L$H��$�H�t$H��$�H�l$�H�\$ H��� H�H�kH�T$hH�l$pH��$�H�$H�<$��H�$x�H��$�H�$H�<$��H�$xH� Qj�YYH���H�L$hH�D$pH�H�$H��$�H���H�l$H��$�H�L$H��$�H�D$�H�L$ �\$(H���H�)H�l$xL�IL��$�����H��$�H����H�S`H�ChH�kpH��$�E1�H��$�H�D$0H��$�H��H�l$0I9��ZH�T$PH����H��$�H��H���L�D$8H��$�H��$�H��H���H��$�H��$�H��$�H��$�L9���H�4$H�D$H�l$xH�l$L�L$�L��$�L�D$8H�T$P�\$ ����H��$H�$H��$H�[ ��H�t$H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H��H�D$�H��$H�$H��$H�[0��H�H�$H��$H�\$H��$H�\$�H�T$H�D$ H��$�H�T$XH��$�H�D$`H�H�$�H�|$H��H����1��H�T$HH�$H�<$��H�\$XH�\$H�\$`H�\$�H�t$HH�t$@H��$�H��$�H���H�H�$H��$�H�\$�H�\$H�l$H��H��H�H�H�\$@H�$���H���É%�l�����J���H��I��H�l$0I9������H��$H�$H��$H�t$H�5H�l$H��H�H�H�D$ ����H���É�f����������������H���É%�H����%����������D +00runtime.morestack_noctxtd�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Vars| go.string."name"�,type.map[string]string�4runtime.mapaccess1_faststr�*sync.(*RWMutex).RLock�4sync.(*RWMutex).RUnlock·f�"runtime.deferproc�,type.map[string]string�4runtime.mapaccess2_faststr�� runtime.duffcopy�� runtime.duffcopy� runtime.eqstring� +� +0go.string."Content-Type"� +8go.string."application/json"� +&net/http.Header.Set� +� type.io.Writer� runtime.convI2I� 4type.encoding/json.Encoder� "runtime.newobject� � runtime.duffzero� 2runtime.writebarrieriface�� runtime.duffcopy�Xtype.github.com/fsouza/go-dockerclient.Image�runtime.convT2E�>encoding/json.(*Encoder).Encode�&runtime.deferreturn�*go.string."not found"�net/http.Error�&runtime.deferreturn�&runtime.deferreturn@�("".autotmp_1001�6type.*encoding/json.Encoder"".autotmp_10006type.*encoding/json.Encoder"".autotmp_0999�type.io.Writer"".autotmp_0998type.string"".autotmp_0997�Xtype.github.com/fsouza/go-dockerclient.Image"".autotmp_0996�Ztype.*github.com/fsouza/go-dockerclient.Image"".autotmp_0995�type.int"".autotmp_0994�type.int"".autotmp_0992�Xtype.github.com/fsouza/go-dockerclient.Image"".autotmp_0990�\type.[]github.com/fsouza/go-dockerclient.Image"".autotmp_0989type.string"".autotmp_0987�type.string "".~r0�6type.*encoding/json.Encoder$encoding/json.w·2�type.io.Writer "".img�Xtype.github.com/fsouza/go-dockerclient.Image +"".id�type.string"".name�type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServerD%�����r��"��$� X� %s!6y�fQ#�:  61�~�=�F?" �Tgclocals·d8f81ddf84701f3ac250364dd80cc8faTgclocals·dfb8fdcf4d5f80c6c0f13902dba3acbb�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�:"".(*DockerServer).listEvents��eH� %H��$(���H;Aw���H��XH��$pH�$H��$hH�[ ��H�t$H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H��HDŽ$�HDŽ$�HDŽ$�H�$�H�\$H�\$H1�H�D$@H�l$HH9���H��$`H�$�H�L$H�H�D$xH�$H��$�H�L$�H�\$H��$�H�\$H��$�H�\$ H��$�H�D$(H�\$0H�\$pH��H�D$ht+H�D$�H��$pH�$H��$hH�[0��H��X�H��$�H��$�H��$�H��H)�H��}OH�H�$H��$H�T$H��$H�L$H��$ H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$H��$ H��H��$H��Hk�H�H�$H��$�H�\$H��$�H�\$H��$�H�\$�H��$H��$H��$ H��$�H��$�H��$�H�D$@H��H�D$@H�l$HH9��C���H�D$�H��$pH�$H��$hH�[0��H��$�H��$�H��$�H��$P1�H��$HH�D$PH��$@H��H�l$PH9���H�D$`H����H�H�pH�hH�L$XH��$�H��$H��$H��$�H��$(H��$�H��$0H��$�H��$8H��$�H�H�CH��$�H���'H��H��H��$�H��$�H��$�H�H�$H��$(H�\$�H�D$H�L$H��$�H�$H�D$xH�D$H��$�H�L$�H�H�$H��$hH�\$H��$pH�\$�H�\$H�,$H��H��H�H�H��$�H�\$H��$�H�\$H��$�H�\$ �H�$��H�\$Hi�@BH�$�H�D$`H�L$XH��H��H�l$PH9��Q���H��XÉ�������J���. +00runtime.morestack_noctxtz +�0go.string."Content-Type"�8go.string."application/json"�&net/http.Header.Set�math/rand.Intn�@"".(*DockerServer).generateEvent�btype.*github.com/fsouza/go-dockerclient.APIEvents�*encoding/json.Marshal� +�type.[][]uint8�"runtime.growslice�2runtime.writebarrierslice� + +�type.[]uint8�runtime.convT2E�2runtime.writebarrieriface�type.io.Writer�runtime.convI2I�fmt.Fprintln�math/rand.Intn�time.Sleep@�0"".autotmp_1022"type.interface {}"".autotmp_1020�&type.[]interface {}"".autotmp_1019�type.[]uint8"".autotmp_1018�type.*[]uint8"".autotmp_1017type.int"".autotmp_1016type.int"".autotmp_1012type.int"".autotmp_1011�type.[][]uint8"".autotmp_1010type.int"".autotmp_1009_type.[]uint8"".autotmp_1008�(type.[1]interface {}"".autotmp_1007type.[][]uint8"".autotmp_1006type.int"".autotmp_1005/type.[][]uint8"".autotmp_1003�type.int"".d�type.[]uint8 "".err�type.error"".data�type.[]uint8"".i�type.int"".count�type.int"".events�type.[][]uint8"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer(%�������� T� %Q$n #� #�"0=e�ijq�� \Tgclocals·7a383875e23784cb158d762414ce6278Tgclocals·c566b50610b1be520e5dd3f364f95be9�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�:"".(*DockerServer).pingDocker�xeH� %H;aw���H��H�D$�H�\$(H�$H�\$ H�[0��H��� + 0runtime.morestack_noctxtj +@ "".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer !@�   +5 Tgclocals·ee0e5af169bfc1eef210605652a1df80Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�@"".(*DockerServer).generateEvent��eH� %H;aw���H��`H�D$(H�D$0H�$�H�D$H���YH���(H�H�+H�l$(H�kH�l$0H�\$hH�$�H�\$H�\$8H�\$H�\$@�H�$�L$H�D$�L$PH�D$XH�\$HH� n�����H�H�\$H�H�$�H�D$H�D$ H�$H�<$��H�$H�\$8H�\$H�\$@H�\$�H�\$ H�$H�<$tPH�\$(H�\$H�\$0H�\$�H�D$ H��t(H�h H�H��H��H�H�H�l$H�h0H�D$pH��`É�ԉ%막%�m���H�������H�H�+H�l$(H�kH�l$0�����H��uH�H�+H�l$(H�kH�l$0����H�������H�H�+H�l$(H�kH�l$0���� + 0runtime.morestack_noctxtjmath/rand.Intn�$go.string."create"�:"".(*DockerServer).generateID�time.Now�`type.github.com/fsouza/go-dockerclient.APIEvents�"runtime.newobject�4runtime.writebarrierstring�4runtime.writebarrierstring�2go.string."mybase:latest"�"go.string."start"� go.string."stop"�&go.string."destroy" �"".autotmp_1036btype.*github.com/fsouza/go-dockerclient.APIEvents"".autotmp_1032Otype.string "".~r0�type.int64time.t·2/type.time.Time"".eventTypeotype.string "".~r0btype.*github.com/fsouza/go-dockerclient.APIEvents"".s*type.*"".DockerServer������N�  +"� + + +  + +4Xt)�Tgclocals·7ba969af8c72fca351526f5bd553df36Tgclocals·76950f6d0769389d26192c168dbb78a0�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�8"".(*DockerServer).loadImage�xeH� %H;aw���H��H�D$�H�\$(H�$H�\$ H�[0��H��� + 0runtime.morestack_noctxtj +@ "".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer !@�  +5 Tgclocals·ee0e5af169bfc1eef210605652a1df80Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�6"".(*DockerServer).getImage��eH� %H;aw���H��(H�D$�H�\$@H�$H�\$8H�[0��H�\$@H�$H�\$8H�[ ��H�t$H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H��H��(� + 0runtime.morestack_noctxtj +� +�0go.string."Content-Type"�6go.string."application/tar"�&net/http.Header.Set@P"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServerPlO +��K +5[Tgclocals·ee0e5af169bfc1eef210605652a1df80Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�L"".(*DockerServer).createExecContainer�!� eH� %H��$ ���H;Aw���H��`H��$�H�$�H�D$H�H�H�kH�H�$H�D$H��$H�T$H��$H�l$�H�\$ H����H� H�kH��$hH�$H��$�H�L$H��$�H�l$�H�\$H�\$HH�D$(H�\$0H��$�H��H��$�t]H�$H�X ��H�L$H�D$H��$pH�$H��$xH�\$H��$H�L$H��$H�D$H�D$ ��H��`�H�H�$�H�\$H�\$xH��$hH�$�H�l$H�T$H��$h1��H��$H��$hH��$H��$pH�t$HH���{H��$�H���H�H�$H�\$xH�\$H��$hH�\$�H�H�$�H�\$H�\$pH�H�$H��$�H���H�o@H�|$H��H�H��H�L$H�D$ H��$H��$�H��$H��$�H�H�$�H�L$H��H����1��H�L$`H� $H�<$�wH��$�H�\$H��$�H�\$�H�L$`H�D$pH� $H��H�H��$�H�D$H��$�H�L$�H�T$pH�L$H�D$ H��$�H��H��$�t]H�$H�Y ��H�L$H�D$H��$pH�$H��$xH�\$H��$H�L$H��$H�D$H�D$ ��H��`�H�jH����H�\$xH�$H�$(H�$ H�JH�BH�jH��$0H��$ H��H��$(�DH�l$H��H��H�H��H�D$pH�hH���H�pH�HH����H�\$xH�$H�$(H�$0H�@H��H��H��H��H��tH��H��$ H�D$H��$(H�T$H��$0H�L$�H��$hH�$H�<$��H�$0�H�\$xH�\$XH��$hH���VH�SH�K H�C(H��$PH��$XH��$`H��H)�H��}OH�H�$H��$8H�T$H��$@H�L$H��$HH�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$@H��$HH��$8H��H�$H�\$XH�\$�H��$8H��$@H��$HH��$hH�$H�<$�gH�$H��$PH�T$H��$XH�L$H��$`H�D$�H��$hH�$H�<$�H�$0�H�D$�H��$xH�$H��$pH�[0��H��$xH�$H��$pH�[ ��H�t$H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H��H�H�$H��$pH�\$H��$xH�\$�H�L$H�D$ H��$�H��$�H��$�H��$�H�H�$�H�L$H��H���1��H�L$PH� $H�<$��H��$�H�\$H��$�H�\$�H�\$PH�\$@H�H�$H�D$�H�D$H�H�+H��$�H�kH��$�H�\$xH�+H��$�H�kH��$�H�H�$H�D$hH�D$H��$�H�\$H��$�H�\$�H�\$@H�$H�L$hH�H��$�H�D$H��$�H�L$�H��`É%�����������%������%����������%�q���� �N���� �B����%�}�����[�����������~�����K���p +00runtime.morestack_noctxtd�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Vars|go.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�@"".(*DockerServer).findContainer� +�net/http.Error�dtype.github.com/fsouza/go-dockerclient.ExecInspect�"runtime.newobject�:"".(*DockerServer).generateID�� runtime.duffzero�� runtime.duffcopy�dtype.github.com/fsouza/go-dockerclient.ExecInspect�.runtime.writebarrierfat�ptype.github.com/fsouza/go-dockerclient.CreateExecOptions�"runtime.newobject�type.io.Reader�runtime.convI2I� 4type.encoding/json.Decoder� "runtime.newobject� +� runtime.duffzero� +2runtime.writebarrieriface� rtype.*github.com/fsouza/go-dockerclient.CreateExecOptions� >encoding/json.(*Decoder).Decode� +� net/http.Error�4runtime.writebarrierstring�2runtime.writebarrierslice�(sync.(*RWMutex).Lock�jtype.[]*github.com/fsouza/go-dockerclient.ExecInspect�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice�,sync.(*RWMutex).Unlock� +� +�0go.string."Content-Type"�8go.string."application/json"�&net/http.Header.Set�type.io.Writer�runtime.convI2I�4type.encoding/json.Encoder�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface�,type.map[string]string�runtime.makemap�go.string."Id"�,type.map[string]string�$runtime.mapassign1�,type.map[string]string�>encoding/json.(*Encoder).Encode�$runtime.panicslice� $runtime.panicindex@� D"".autotmp_1068� +type.string"".autotmp_1067� type.string"".autotmp_1066� 6type.*encoding/json.Encoder"".autotmp_10656type.*encoding/json.Encoder"".autotmp_1064� type.io.Writer"".autotmp_1063type.uint64"".autotmp_1062type.uint64"".autotmp_1061type.int"".autotmp_1060type.int"".autotmp_1059�jtype.[]*github.com/fsouza/go-dockerclient.ExecInspect"".autotmp_1058�jtype.[]*github.com/fsouza/go-dockerclient.ExecInspect"".autotmp_1057� ftype.*github.com/fsouza/go-dockerclient.ExecInspect"".autotmp_1054� 6type.*encoding/json.Decoder"".autotmp_10536type.*encoding/json.Decoder"".autotmp_1052� type.io.Reader"".autotmp_1051,type.map[string]string"".autotmp_1049jtype.[]*github.com/fsouza/go-dockerclient.ExecInspect"".autotmp_1046type.string"".autotmp_1043�dtype.github.com/fsouza/go-dockerclient.ExecInspect"".autotmp_1042type.string"".autotmp_1041type.string"".autotmp_1039� type.string"".autotmp_1038� ,type.map[string]string"".&exec� ftype.*github.com/fsouza/go-dockerclient.ExecInspect"".¶ms� rtype.*github.com/fsouza/go-dockerclient.CreateExecOptions "".~r0� 6type.*encoding/json.Encoder$encoding/json.w·2� type.io.Writer$encoding/json.r·2� type.io.Reader "".err� +type.error"".container� btype.*github.com/fsouza/go-dockerclient.Container +"".id� +type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer8%� �� � �� � �� � ����%iGU +u�UZk!�!#Q�    +  `1�<:L72 � +~�D` � L#s4�Tgclocals·a484a676faa0084ad5f98b43c17e101cTgclocals·b3446cef6b648ddae3c01581a04ccc0b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�J"".(*DockerServer).startExecContainer� � eH� %H;aw���H��xH��$�H�$�H�D$H�H�H�kH�H�$H�D$H�T$hH�T$H�l$pH�l$�H�\$ H����H� H�kH��$�H�$H�L$8H�L$H�l$@H�l$�H�\$H�\$0H�D$ H�\$(H�\$PH��H�D$H�<H��$�H�$H�<$�H�$0�H�\$0H��@�kH��$�H�$H�<$��H�$0�H�L$8H�D$@H�H�$H��$�H��H�l$H�L$hH�L$H�D$pH�D$�H�D$ �\$(H�(����H�]H����H�\$8H�\$XH�\$@H�\$`H�H�$H��$�H��H�l$H�\$XH�\$�H��$�H�$H�<$tcH�$0�H�\$01�@�kH��$�H�$H�<$t2H�$0�H�D$�H��$�H�$H��$�H�[0��H��xÉ%�ʼn%�H�H� H�CH�H�$H��$�H��H�l$H�L$hH�L$H�D$pH�D$�H�D$ �\$(H�(���%���H�]H����H�H�+H�l$XH�kH�l$`H�H�$H��$�H��H�l$H�\$XH�\$�������%�����%�����H�D$�H��$�H�$H��$�H�[0��H��xÉ�A���2 + 0runtime.morestack_noctxtN�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Varsfgo.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�4"".(*DockerServer).getExec�(sync.(*RWMutex).Lock�,sync.(*RWMutex).Unlock�,type.map[string]func()�4runtime.mapaccess2_faststr� +�,type.map[string]func()�"runtime.mapdelete�(sync.(*RWMutex).Lock�,sync.(*RWMutex).Unlock� +� go.string."*"� ,type.map[string]func()� +4runtime.mapaccess2_faststr� + +� +go.string."*"� +,type.map[string]func()� "runtime.mapdelete� +@�"".autotmp_1089type.*func()"".autotmp_1087type.string"".autotmp_1086type.string"".autotmp_1085?type.string"".autotmp_1084type.string"".autotmp_1082type.string "".err_type.error"".exec�ftype.*github.com/fsouza/go-dockerclient.ExecInspect +"".idtype.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer(��������h�cM!!X B + #  \ F  #%.&wI�(�eCTgclocals·8375af20f91e3bf26f9f4b100ffb7d0eTgclocals·349a065d14b607627da67d5600b2511a�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�L"".(*DockerServer).resizeExecContainer��eH� %H;aw���H��`H��$�H�$�H�D$H�H�H�kH�H�$H�D$H�T$PH�T$H�l$XH�l$�H�\$ H��t}H� H�kH�\$hH�$H�L$0H�L$H�l$8H�l$�H�L$xH�D$pH�T$ H�\$(H�\$HH��H�T$@uH�D$�H� $H�X0��H��`�H�D$�H� $H�X0��H��`É�|��� + 0runtime.morestack_noctxtN�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Varsfgo.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�4"".(*DockerServer).getExec� +� +@� "".autotmp_1091type.string "".err?type.error +"".id_type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer&�������$�_F  &�Tgclocals·ee0e5af169bfc1eef210605652a1df80Tgclocals·f883d3996c76325fd1714d4e3de9fa33�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�N"".(*DockerServer).inspectExecContainer� � eH� %H�D$�H;Aw���H��H��$�H�$�H�D$H�H�H�kH�H�$H�D$H��$�H�T$H��$�H�l$�H�\$ H����H� H�kH��$�H�$H�L$PH�L$H�l$XH�l$�H��$�H��$�H�\$H�\$0H�T$ H�\$(H�\$hH��H�T$`�FH�D$�H� $H�X0��H��$�H�$H��$�H�[ ��H�t$H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H��H�H�$H��$�H�\$H��$�H�\$�H�L$H�D$ H��$�H�L$@H��$�H�D$HH�H�$�H�L$H��H��tr1��H�L$8H� $H�<$tRH�\$@H�\$H�\$HH�\$�H�L$8H�D$0H� $H��H�H�D$pH�D$H�L$xH�L$�H�ĠÉ%륉�H�D$�H� $H�X0��H�ĠÉ�4���* +*0runtime.morestack_noctxt^�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Varsvgo.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�4"".(*DockerServer).getExec� +� +�0go.string."Content-Type"�8go.string."application/json"�&net/http.Header.Set�type.io.Writer�runtime.convI2I�4type.encoding/json.Encoder�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface�ftype.*github.com/fsouza/go-dockerclient.ExecInspect�>encoding/json.(*Encoder).Encode� +@�"".autotmp_1100�6type.*encoding/json.Encoder"".autotmp_10996type.*encoding/json.Encoder"".autotmp_1098?type.io.Writer"".autotmp_1094type.string$encoding/json.w·2�type.io.Writer "".errtype.error"".exec�ftype.*github.com/fsouza/go-dockerclient.ExecInspect +"".id�type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer&"����'���2�"i]Q� .��>1<Tgclocals·cec9627e2837f98af62e9c7580b3baccTgclocals·04f43ee17c64d5db43a23c286d1bf236�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�4"".(*DockerServer).getExec� � eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$�HDŽ$�H��$�H�$H�<$�yH�$0�H��$�H�$H�<$�LH�$0H� Qj�L��$�YYH���H��$�H����H�SH�C H�k(H��$�1�H��$�H�D$(H��$�H�l$(H9���H�T$HH�H�t$0H�\$8H����H�;H��$�H�KH��$�L9�uoH�<$H�L$H��$�H�l$L�D$�L��$�H�t$0H�T$H�\$ ��t3H�\$8H��$�HDŽ$�HDŽ$���H�Ĩ�H��H��H�l$(H9��D���H�H�+H�l$`H�kH�l$hH�D$PH�D$XH�H�$�H�L$H�L$@H� $H�<$��H�\$`H�\$H�\$hH�\$�H�\$@H�\$@H� 1�H9�tCH�T$@H�L$pH�T$xHDŽ$�H�L$PH��$�H�T$XH��$���H�Ĩ�H�H�$H�H�\$H�H�\$�H�L$뎉%�Q�����W�����������H�ĨÉ%�����%�{���$ +*0runtime.morestack_noctxt�*sync.(*RWMutex).RLock�4sync.(*RWMutex).RUnlock·f�"runtime.deferproc� runtime.eqstring�&runtime.deferreturn�4go.string."exec not found"�.type.errors.errorString�"runtime.newobject�4runtime.writebarrierstring�Bgo.itab.*errors.errorString.error� &runtime.deferreturn� 0type.*errors.errorString� +type.error� +Bgo.itab.*errors.errorString.error� + runtime.typ2Itab� +&runtime.deferreturn`�"".autotmp_1111otype.error"".autotmp_1110�0type.*errors.errorString"".autotmp_1109Otype.string"".autotmp_1107�htype.**github.com/fsouza/go-dockerclient.ExecInspect"".autotmp_1106�type.int"".autotmp_1105�type.int"".autotmp_11040type.*errors.errorString"".autotmp_1103/jtype.[]*github.com/fsouza/go-dockerclient.ExecInspect "".~r0�type.errorerrors.text·2�type.string"".exec�ftype.*github.com/fsouza/go-dockerclient.ExecInspect "".~r2@type.error "".~r10ftype.*github.com/fsouza/go-dockerclient.ExecInspect +"".idtype.string"".s*type.*"".DockerServerD"�i������V��&�6�F!>Xg3 +�(b�FX2R1&4Tgclocals·dbefa26e1f0ee62688488e90e23fcbd7Tgclocals·4459bbba29917b2fee408f0dbbff89b1�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�<"".(*DockerServer).findNetwork��eH� %H�D$�H;Aw���H��HDŽ$�HDŽ$�HDŽ$�HDŽ$�H��$�H�$H�<$�&H�$��H��$�H�$H�<$��H�$�H� Qj�H��$�YYH����H��$�H����L���H���H���H��$�1�H��$�H�D$0L��$�H�l$0H9��AL�D$PI�8H�t$8H�t$(H�|$@H���;H�OH�GH9���H��$�H� $H��$�H�D$H��$�H�l$H�T$�L�D$PH�|$@H�t$8H��$��\$ ��t;H��$�H�\$(H��$�HDŽ$�HDŽ$���H�İ�H��H����H�?H��$�H�KH��$�H9�uEH�<$H�L$H��$�H�l$H�T$�L�D$PH�|$@H�t$8H��$��\$ ���W���I��H��H�l$0H9������H�H�+H�l$hH�kH�l$pH�D$XH�D$`H�H�$�H�L$H�L$HH� $H�<$��H�\$hH�\$H�\$pH�\$�H�\$HH�\$HH� 1�H9�tRH�T$HH�L$xH��$�HDŽ$�HDŽ$�����H�L$XH��$�H�T$`H��$���H�İ�H�H�$H�H�\$H�H�\$�H�L$�|����%�?�����o����������T�����H�İÉ%������%�����& +*0runtime.morestack_noctxt�*sync.(*RWMutex).RLock�4sync.(*RWMutex).RUnlock·f�"runtime.deferproc� runtime.eqstring�&runtime.deferreturn� runtime.eqstring� 6go.string."No such network"� .type.errors.errorString� +"runtime.newobject� +4runtime.writebarrierstring� Bgo.itab.*errors.errorString.error� &runtime.deferreturn� 0type.*errors.errorString� type.error� Bgo.itab.*errors.errorString.error�  runtime.typ2Itab� &runtime.deferreturnp�$"".autotmp_1123otype.error"".autotmp_1122�0type.*errors.errorString"".autotmp_1121type.string"".autotmp_1120Otype.string"".autotmp_1118�`type.**github.com/fsouza/go-dockerclient.Network"".autotmp_1117�type.int"".autotmp_1116�type.int"".autotmp_11150type.*errors.errorString"".autotmp_1114/btype.[]*github.com/fsouza/go-dockerclient.Network "".~r0�type.errorerrors.text·2�type.string"".network�^type.*github.com/fsouza/go-dockerclient.Network"".i�type.int "".~r3Ptype.error "".~r2@type.int "".~r10^type.*github.com/fsouza/go-dockerclient.Network"".idOrNametype.string"".s*type.*"".DockerServerD"�{������`���:�R$Afq;n +�,q�SRt2a10+Tgclocals·8bd789dcce9d4daa4c4bb84dfe47e247Tgclocals·4459bbba29917b2fee408f0dbbff89b1�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�>"".(*DockerServer).listNetworks��eH� %H��$P���H;Aw���H��01�H��$��H��$8H�$H�<$��H�$��H��$8H���H�H�$H�D$H�l$�L�T$L�L$ L�D$(L��$�L��$�L��$�H��$8H���OH���H���H���H��$�1�H��$�H�D$@H��$�H��H�l$@H9��*H�D$`H�0H�T$HH����H��$�H���L��L��L��L��L)�H��}OH�H�$H��$�H�T$H��$�H�L$H��$�H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$�H��$�H�H�$H��H��$�H��Hk�HH�H�\$H��$�H�\$�L��$�L��$�L��$�L��$�L��L��$�L��$�H�D$`H�T$HH��H��H�l$@H9������H��$8H�$H�<$��H�$��H��$HH�$H��$@H�[ ��H�t$H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H��H�D$�H��$HH�$H��$@H�[0��H�H�$H��$@H�\$H��$HH�\$�H�L$H�D$ H�L$xH�L$hH��$�H�D$pH�H�$�H�|$H��H����1��H�L$XH� $H�<$��H�\$hH�\$H�\$pH�\$�H�\$XH�\$PH��$�H��$�H��$�H��$�H��$�H��$�H�H�$H��$�H�\$�H�\$H�l$H��H��H�H�H�\$PH�$�H��0É%�Z�����8����%�>����� ���������%�+���4 +00runtime.morestack_noctxt`� runtime.duffzero�*sync.(*RWMutex).RLock�`type.[]github.com/fsouza/go-dockerclient.Network�"runtime.makeslice�� runtime.duffcopy�`type.[]github.com/fsouza/go-dockerclient.Network�"runtime.growslice�\type.github.com/fsouza/go-dockerclient.Network�.runtime.writebarrierfat� .sync.(*RWMutex).RUnlock� +� 0go.string."Content-Type"� +8go.string."application/json"� +&net/http.Header.Set� +� type.io.Writer� runtime.convI2I� 4type.encoding/json.Encoder� "runtime.newobject� � runtime.duffzero� 2runtime.writebarrieriface�`type.[]github.com/fsouza/go-dockerclient.Network�runtime.convT2E�>encoding/json.(*Encoder).Encode@�$"".autotmp_1144�6type.*encoding/json.Encoder"".autotmp_11436type.*encoding/json.Encoder"".autotmp_1142�type.io.Writer"".autotmp_1137`type.[]github.com/fsouza/go-dockerclient.Network"".autotmp_1136�\type.github.com/fsouza/go-dockerclient.Network"".autotmp_1134�`type.**github.com/fsouza/go-dockerclient.Network"".autotmp_1133�type.int"".autotmp_1132type.int"".autotmp_1131�`type.[]github.com/fsouza/go-dockerclient.Network"".autotmp_1128�btype.[]*github.com/fsouza/go-dockerclient.Network"".autotmp_1127�`type.[]github.com/fsouza/go-dockerclient.Network"".autotmp_1126�type.int "".~r0�6type.*encoding/json.Encoder$encoding/json.w·2�type.io.Writer"".result�`type.[]github.com/fsouza/go-dockerclient.Network"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer%����9�H�4$Td�$Q#�  *S�\v�FW"FTgclocals·908986cc2bd23e6b2b43c6b331d27560Tgclocals·429e38e879552ec65b0c30795e04b14b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�<"".(*DockerServer).networkInfo� +� +eH� %H�D$�H;Aw���H��H��$�H�$�H�D$H�H�H�kH�H�$H�D$H��$�H�T$H��$�H�l$�H�\$ H���H� H�kH��$�H�$H�L$XH�L$H�l$`H�l$�H�\$H�\$8H�D$(H�\$0H�\$pH��H�D$ht]H�$H�X ��H�L$H�D$H��$�H�$H��$�H�\$H��$�H�L$H��$�H�D$H�D$ ��H�Ĩ�H��$�H�$H��$�H�[ ��H�t$H�4$H�5H�l$H��H�H�H�H�l$H��H��H�H��H�D$�H��$�H�$H��$�H�[0��H�H�$H��$�H�\$H��$�H�\$�H�L$H�D$ H��$�H�L$HH��$�H�D$PH�H�$�H�L$H��H��tu1��H�L$@H� $H�<$tUH�\$HH�\$H�\$PH�\$�H�L$@H�D$8H� $H��H�H�D$xH�D$H��$�H�L$�H�ĨÉ%뢉뇉�����, +*0runtime.morestack_noctxt^�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Varsvgo.string."id"�,type.map[string]string�4runtime.mapaccess1_faststr�<"".(*DockerServer).findNetwork� +�net/http.Error� +�0go.string."Content-Type"�8go.string."application/json"�&net/http.Header.Set� +�type.io.Writer�runtime.convI2I�4type.encoding/json.Encoder�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface� ^type.*github.com/fsouza/go-dockerclient.Network� >encoding/json.(*Encoder).Encode@�"".autotmp_1156�6type.*encoding/json.Encoder"".autotmp_11556type.*encoding/json.Encoder"".autotmp_1154?type.io.Writer"".autotmp_1151type.string"".autotmp_1149type.string$encoding/json.w·2�type.io.Writer "".errtype.error"".network�^type.*github.com/fsouza/go-dockerclient.Network +"".id�type.string"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer("�������#�6�"i> UQ#� .��>40Tgclocals·cec9627e2837f98af62e9c7580b3baccTgclocals·04f43ee17c64d5db43a23c286d1bf236�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�"".isValidName��eH� %H;aw���H��(H�D$8H��t:H�t$0H�4$H�D$H�5H�l$H��H�H���\$ ��u +�D$@H��(��D$@H��(� + 0runtime.morestack_noctxtlgo.string."."� strings.Contains0P "".~r1 type.bool"".nametype.stringPDOP Op�6 + +F*Tgclocals·a08e9001cb8f9d822225de3b8e406515Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�@"".(*DockerServer).createNetwork��eH� %H��$����H;Aw���H��H�H�$�H�\$H�\$hH��$�H����H�S@H�kHH��$�H�,$H��$�H����H�Z Sj�YYH����H�H�$H��$�H���xH�o@H�|$H��H�H��H�L$H�D$ H��$�H��$�H��$�H��$�H�H�$�H�|$H��H���1��H�L$XH� $H�<$��H��$�H�\$H��$�H�\$�H�T$XH�L$hH�$H��H� H��$�H�L$H��$�H�T$�H�L$H�T$ H��$�H��H��$�tcH�$H�Y ��H�T$H�L$H��$�H�$H��$�H�\$H��$H�T$H��$H�L$H�D$ ����H�Ę�H�t$hH�>H����H�7H�<$H�H���\$��uHH��$�H�$H��$�H�t$H�5H�l$H��H�H�H�D$ ����H�Ę�H��$�H�$H�t$hH�>H���pH�7H�|$H�H��H�\$1�H9�tHH��$�H�$H��$�H�t$H�5H�l$H��H�H�H�D$ ����H�Ę�H��$�H�$�H�L$H�D$H��$�H��$�H�H�$�H�T$hH�L$H��$P1��H�:H����H�H��$PH��H�H�H��$�H��$`H��$�H��$hH�:H���]H�wH��$pH�H�H�H�$H�L$`H�L$H��$PH�\$�H��$�H�$H�<$�H�$��H�\$`H�\$PH��$�H����H���H���H���H��$8H��$@H��$HH��H)�H��}OH�H�$H��$ H�T$H��$(H�L$H��$0H�D$H�D$ �H�T$(H�L$0H�D$8H��H��H��$(H��$0H��$ H��H�$H�\$PH�\$�H��$ H��$(H��$0H��$�H�$H�<$��H�$�H��$8H�T$H��$@H�L$H��$HH�D$�H��$�H�$H�<$��H�$��H�D$�H��$�H�$H��$�H�[0��H��$�H�H�CH�|$`H�oH��$�H��H�H�H�H�$H��$�H�\$H��$�H�\$�H�L$H�D$ H��$�H�L$pH��$�H�D$xH�H�$�H�|$H��H����1��H�L$HH� $H�<$��H�\$pH�\$H�\$xH�\$�H�t$HH�t$@H��$�H��$H��H�H�H�H�$H��$H�\$�H�\$H�l$H��H��H�H�H�\$@H�$���H�ĘÉ%�m�����K����%�p����%������&����%������������V��������������%� ����������������H�ĘÉ�8����� ���` +00runtime.morestack_noctxtPxtype.*github.com/fsouza/go-dockerclient.CreateNetworkOptionsb"runtime.newobject�"runtime.deferproc�type.io.Reader�runtime.convI2I�4type.encoding/json.Decoder�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface�ztype.**github.com/fsouza/go-dockerclient.CreateNetworkOptions�>encoding/json.(*Decoder).Decode� +�net/http.Error�&runtime.deferreturn�"".isValidName� @go.string."Invalid network name"� net/http.Error� &runtime.deferreturn� +<"".(*DockerServer).findNetwork� Dgo.string."network already exists"� net/http.Error� &runtime.deferreturn� :"".(*DockerServer).generateID� \type.github.com/fsouza/go-dockerclient.Network� "runtime.newobject� � runtime.duffzero�\type.github.com/fsouza/go-dockerclient.Network�.runtime.writebarrierfat�(sync.(*RWMutex).Lock�btype.[]*github.com/fsouza/go-dockerclient.Network�"runtime.growslice�.runtime.writebarrierptr�2runtime.writebarrierslice�,sync.(*RWMutex).Unlock� +�type.io.Writer�runtime.convI2I�4type.encoding/json.Encoder�"runtime.newobject�� runtime.duffzero�2runtime.writebarrieriface�2type.struct { ID string }�runtime.convT2E�>encoding/json.(*Encoder).Encode�&runtime.deferreturn�&runtime.deferreturn@�2"".autotmp_1180�6type.*encoding/json.Encoder"".autotmp_11796type.*encoding/json.Encoder"".autotmp_1178�type.io.Writer"".autotmp_1173�btype.[]*github.com/fsouza/go-dockerclient.Network"".autotmp_1172�btype.[]*github.com/fsouza/go-dockerclient.Network"".autotmp_1171�^type.*github.com/fsouza/go-dockerclient.Network"".autotmp_1170�6type.*encoding/json.Decoder"".autotmp_11696type.*encoding/json.Decoder"".autotmp_1168�type.io.Reader"".autotmp_1167�2type.struct { ID string }"".autotmp_1166btype.[]*github.com/fsouza/go-dockerclient.Network"".autotmp_1165�\type.github.com/fsouza/go-dockerclient.Network"".autotmp_1164type.string"".autotmp_1162�type.string"".&network�^type.*github.com/fsouza/go-dockerclient.Network"".&config�ztype.**github.com/fsouza/go-dockerclient.CreateNetworkOptions "".~r0�6type.*encoding/json.Encoder$encoding/json.w·2�type.io.Writer$encoding/json.r·2�type.io.Reader"".c�2type.struct { ID string }"".generatedID�type.string "".err�type.error"".r0,type.*net/http.Request"".w8type.net/http.ResponseWriter"".s*type.*"".DockerServer`%�W���s��������������%O�U,:;:+ +� +$�$#/�    &j0NwL7-xD=F*�$�Dc�4F>"�Tgclocals·a484a676faa0084ad5f98b43c17e101cTgclocals·770edcce2fa22bec4aa18365adf4562e�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�R"".*DockerServer.("".commitContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�D"".(*DockerServer).commitContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�P"".*DockerServer.("".listContainers)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�B"".(*DockerServer).listContainers0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�R"".*DockerServer.("".createContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�D"".(*DockerServer).createContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�T"".*DockerServer.("".inspectContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�F"".(*DockerServer).inspectContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�R"".*DockerServer.("".renameContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�D"".(*DockerServer).renameContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�L"".*DockerServer.("".topContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�>"".(*DockerServer).topContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�P"".*DockerServer.("".startContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�B"".(*DockerServer).startContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�N"".*DockerServer.("".stopContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�@"".(*DockerServer).stopContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�P"".*DockerServer.("".pauseContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�B"".(*DockerServer).pauseContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�T"".*DockerServer.("".unpauseContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�F"".(*DockerServer).unpauseContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�N"".*DockerServer.("".waitContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�@"".(*DockerServer).waitContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�R"".*DockerServer.("".attachContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�D"".(*DockerServer).attachContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�R"".*DockerServer.("".removeContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�D"".(*DockerServer).removeContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�Z"".*DockerServer.("".createExecContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�L"".(*DockerServer).createExecContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�P"".*DockerServer.("".statsContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�B"".(*DockerServer).statsContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�Z"".*DockerServer.("".resizeExecContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�L"".(*DockerServer).resizeExecContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�X"".*DockerServer.("".startExecContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�J"".(*DockerServer).startExecContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�\"".*DockerServer.("".inspectExecContainer)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�N"".(*DockerServer).inspectExecContainer0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�F"".*DockerServer.("".pullImage)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�8"".(*DockerServer).pullImage0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�H"".*DockerServer.("".buildImage)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�:"".(*DockerServer).buildImage0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�H"".*DockerServer.("".listImages)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�:"".(*DockerServer).listImages0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�J"".*DockerServer.("".removeImage)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�<"".(*DockerServer).removeImage0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�L"".*DockerServer.("".inspectImage)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�>"".(*DockerServer).inspectImage0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�F"".*DockerServer.("".pushImage)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�8"".(*DockerServer).pushImage0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�D"".*DockerServer.("".tagImage)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�6"".(*DockerServer).tagImage0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�H"".*DockerServer.("".listEvents)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�:"".(*DockerServer).listEvents0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�H"".*DockerServer.("".pingDocker)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�:"".(*DockerServer).pingDocker0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�F"".*DockerServer.("".loadImage)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�8"".(*DockerServer).loadImage0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�D"".*DockerServer.("".getImage)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�6"".(*DockerServer).getImage0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�L"".*DockerServer.("".listNetworks)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�>"".(*DockerServer).listNetworks0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�J"".*DockerServer.("".networkInfo)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�<"".(*DockerServer).networkInfo0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�N"".*DockerServer.("".createNetwork)·fm��eH� %H;aw���H�� H�ZH�$H�\$(H�\$H�\$0H�\$H�\$8H�\$�H�� � + "runtime.morestack�@"".(*DockerServer).createNetwork0@ +"".a1 ,type.*net/http.Request +"".a08type.net/http.ResponseWriter@/?P�P +@Tgclocals·099986b79bd4df464b634a14757f9178Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�"".func·001��eH� %H��$����H;Aw���H��H�BH�ZH�\$pH�D$hH�H���H��$81��H�H�$H�l$H��$8H�\$�H��$81�H9��AH��$@H���H� H�CH��$8H����H�+H��$�H�kH��$�H��$�H��$�H�L$xH� $H��$�H�D$H��$�H�~H����H�w8H�|$H�H���\$ H��H�L$(H�\$0H��$�H��H��$�t]H�$H�Y ��H�L$H�D$H��$�H�$H��$�H�\$H��$�H�L$H��$�H�D$H�D$ ��H�Ĉ�<��H��$8H�$�H��$81�H9������H�\$hH�+H���H���H���H��H��$01�H��$(H�L$HH��$ H��H�l$HH9��VH�L$`H�)H�D$PH�D$@H��H�H�H�kH�H�$H�D$XH�D$H��$�H�T$H��$�H�l$�H�\$ H����H� H�kH��$�H� $H��$�H�l$H��$�H�~H����H�w8H�|$H�H���\$ H��H�L$(H�\$0H��$�H��H��$�t]H�$H�Y ��H�L$H�D$H��$�H�$H��$�H�\$H��$�H�L$H��$�H�D$H�D$ ��H�Ĉ�<uZH�L$`H�D$PH��H��H�l$HH9������H��$�H�$H��$�H�\$H��$�H�\$H�\$pH�H���H�Ĉ�H�H� H�kH�H�$H�\$XH�\$H��$�H�L$H��$�H�l$�H�\$ H���CH� H�kH��$�H�$H��$�H�\$H��$�H�L$H��$�H�l$H�D$ ��H�|$@H�L$hH�)H��H9���H�L���I��I��H��H��H�)H��H�)H���H9���H�H��H���H)�H��H)�H��H��t H��H��H�H��H��$�H��H��$L��$L��L��$L��$L��L��$�H��$�H�L��$�L)�H��~[H�H�$H��$�H�t$L�L$L�D$H�D$ �L��$H��$�H�t$(H�\$0H��$�H�\$8H��$�H��$�J�,�H�,$H��$�H�\$H��H��H�\$�H��$H��$�H��$�H��$�H�H��$�H��$�H��$�H�\$hH�+H�,$H�<$tm�5i��� +�]� }%Tgclocals·9672a07f1a450fc594d7cd9cb2c95495Tgclocals·0809678294a6ccf1679e4ac422a0f629�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�"".init��eH� %H;aw���H�����t���uH���� ����������������H�H�,$H��H��H�H��H�D$H�H�$H�D$��H���4 + 0runtime.morestack_noctxt:"".initdone·R"".initdone·p"runtime.throwinit�"".initdone·��github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.init��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.init�Lgithub.com/fsouza/go-dockerclient.init�time.init�sync.init�strings.init�strconv.init�regexp.init�net/http.init�net.init�math/rand.init�fmt.init�$encoding/json.init� crypto/rand.init� archive/tar.init�Pgo.string."^[a-zA-Z0-9][a-zA-Z0-9_.-]+$"�$regexp.MustCompile�"".nameRegexp�.runtime.writebarrierptr�"".initdone·00/0�/����4�  7�Tgclocals·3280bececceccd33cb74587feedb1f9fTgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�(type..hash.[8]string��eH� %H;aw���H��0H�L$H1�H�D$ H�l$ H9�}eH�D$(H��H��H��H��c��k�RH��H��H�\$8H��tDHk�H�H�$H�D$H�D$HH�D$�H�L$H�D$(H��H�l$ H9�|�H�L$PH��0É� + 0runtime.morestack_noctxt�runtime.strhash@` "".autotmp_1229type.int"".autotmp_1228type.int "".~r30type.uintptr"".h type.uintptr"".stype.uintptr"".ptype.*[8]string`�_` �� +}3Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783Tgclocals·3280bececceccd33cb74587feedb1f9f�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�$type..eq.[8]string��eH� %H;aw���H��X1�H�D$(H�l$(H9���H�D$0H�\$`H����H��Hk�H�H�3H�KH�\$hH��tvH��Hk�H�H�H�CH9�uVH�t$HH�4$H�L$PH�L$H�T$8H�T$H�D$@H�D$��\$ ��t H�D$0H��H�l$(H9��n����D$xH��X��D$xH��XÉ놉�c��� + 0runtime.morestack_noctxt� runtime.eqstring@�"".autotmp_1233?type.string"".autotmp_1232type.string"".autotmp_1231_type.int"".autotmp_1230Otype.int "".~r30type.bool"".s type.uintptr"".qtype.*[8]string"".ptype.*[8]string&���� ���� �PTgclocals·9c703c5c7b9c1932c840b69f8ebce236Tgclocals·44568aa369055d8938d809aa5d80843b�/Users/csparr/ws/go/src/github.com/influxdb/telegraf/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go�go.string."/containers/{id:.*}"PH/containers/{id:.*} >go.string."/containers/{id:.*}"�$go.string."DELETE"0.DELETE $go.string."DELETE"�Hgo.string."/containers/{id:.*}/exec"`R/containers/{id:.*}/exec Hgo.string."/containers/{id:.*}/exec"�Jgo.string."/containers/{id:.*}/stats"`T/containers/{id:.*}/stats Jgo.string."/containers/{id:.*}/stats"�@go.string."/exec/{id:.*}/resize"PJ/exec/{id:.*}/resize @go.string."/exec/{id:.*}/resize"�>go.string."/exec/{id:.*}/start"PH/exec/{id:.*}/start >go.string."/exec/{id:.*}/start"�/images/create 4go.string."/images/create"�$go.string."/build"0./build $go.string."/build"�0go.string."/images/json"@: /images/json 0go.string."/images/json"�6go.string."/images/{id:.*}"@@/images/{id:.*} 6go.string."/images/{id:.*}"�Dgo.string."/images/{name:.*}/json"PN/images/{name:.*}/json Dgo.string."/images/{name:.*}/json"�Dgo.string."/images/{name:.*}/push"PN/images/{name:.*}/push Dgo.string."/images/{name:.*}/push"�Bgo.string."/images/{name:.*}/tag"PL/images/{name:.*}/tag Bgo.string."/images/{name:.*}/tag"�&go.string."/events"00/events &go.string."/events"�$go.string."/_ping"0./_ping $go.string."/_ping"�0go.string."/images/load"@: /images/load 0go.string."/images/load"�>go.string."/images/{id:.*}/get"PH/images/{id:.*}/get >go.string."/images/{id:.*}/get"�*go.string."/networks"@4 /networks *go.string."/networks"�:go.string."/networks/{id:.*}"PD/networks/{id:.*} :go.string."/networks/{id:.*}"�Tgclocals·76225bbef6ae6e9e5960f6f7925b8185@@� "�Tgclocals·2c09ec81c5cb12328d7183f25bc48833@@�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·e8c55b930b09fa5028b5e4b78b8932dc +�Tgclocals·31214a5fe2ac06a8b2e85038c37289d6  +�Tgclocals·5197b04b6fafdc0c7d1822cc34066683 ���Tgclocals·31214a5fe2ac06a8b2e85038c37289d6  +�Tgclocals·5197b04b6fafdc0c7d1822cc34066683 ���Tgclocals·f29b89ce4cd57d8100665fbda8fdf405 "�Tgclocals·1765c43755fbf91dfae87195c1ec24fb  +���"go.string."error"0,error "go.string."error"�go.string."url"0(url go.string."url"�Tgclocals·20f42599a700c1a4b6c6ede24ef4e8a600��Tgclocals·9a90374975a8610a14ef231e086acf2900 +�����Tgclocals·8d600a433c6aaa81a4fe446d95c5546b �Tgclocals·bd51743682bd6c0f7b9f2e8e6dffed99  + +�Tgclocals·0528ab8f76149a707fd2f0025c2178a3�Tgclocals·519efd86263089ddb84df3cfe7fd2992�Tgclocals·85223f890d4c8f80203775beed82eadd +�Tgclocals·925be0824eaf197a56a5d7050bf29309  +���,Bgo.itab.*errors.errorString.error�>go.string."container not found"PHcontainer not found >go.string."container not found"�Tgclocals·0e8ff9f111235a6bccca3fa33f624774``."��%�Tgclocals·1e2d550ac4f017d716d87ff44946577f88 J%e J%e J%e J%e J%e �Tgclocals·d64e51a4c4bfeaa840e480961ec6b0b3�Tgclocals·519efd86263089ddb84df3cfe7fd2992�&go.string."http://"00http:// &go.string."http://"�go.string."/"0$/ go.string."/"�Tgclocals·f883d3996c76325fd1714d4e3de9fa33 �Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·3901c619f635162fa423fe138099ace5(( ��U���U�Tgclocals·bc335ce91c3a8b5f426dd201465802bd((����,�go.itab.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Router.net/http.Handler�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�Tgclocals·be18fcff1e4d1cf801d0b47f660b980688("�Tgclocals·ab01a2d55089ff50c402006df1039c3988 + + + + +�go.string."all"0(all go.string."all"�go.string."1"0$1 go.string."1"�go.string." "0$  go.string." "�"go.string."%s %s"0,%s %s "go.string."%s %s"�go.string."/%s"0(/%s go.string."/%s"�0go.string."Content-Type"@: Content-Type 0go.string."Content-Type"�8go.string."application/json"PBapplication/json 8go.string."application/json"�Tgclocals·a74ca190396b92ed76efe93c61653942����H"��H"���H"�� ��H"����H"��"��H"�� "��H"��/" ��H"��/"��H"�� ��H"����H"����H"�� ��H"��H"���H"� �H"��H"��H"��Tgclocals·02bfe185cbfa386cc6696a665007ff28����������������������Tgclocals·e0b091cc964057ade987c1196ae02e2e� � v�P� �P����""�$� �� �"J ��V���""�$� �� �"J ��V���""�$� �� �"J��V ���V���""�$� �� �"J��Tgclocals·f7ba1512b6938de3ab7810c798567682pp �������������Tgclocals·660c52760819425e2fa6ae9a8a8ae931  �Tgclocals·f7cb58e18cf0f9d3ee7dc7385e94aef7  +�,�2go.string."No such image"@< No such image 2go.string."No such image"�Tgclocals·52c6e5e411ef106b9194437a527f5a0f��"�H)"RI� � ��� (��Tgclocals·c958acb0df1ea67178a15bee7623bbbd88 +�� + + +� go.string."name"0*name go.string."name"�Dgo.string."Invalid container name"PNInvalid container name Dgo.string."Invalid container name"�&go.string."0.0.0.0"000.0.0.0 &go.string."0.0.0.0"�0go.string."172.16.42.%d"@: 172.16.42.%d 0go.string."172.16.42.%d"�.go.string."172.16.42.1"@8 172.16.42.1 .go.string."172.16.42.1"�&go.string."docker0"00docker0 &go.string."docker0"�lgo.string."there's already a container with this name"�v*there's already a container with this name lgo.string."there's already a container with this name"�Tgclocals·37832594314d999d6d2b5ee19691d36f��,� ��Z ��Z ���Z � �Z � �� ��� ���@  ���e  ���� e  ����e  ��� �VR��"""� + �� �VR��"""� + �� �VR��"""� + �� �VR��"""� + ��� ������Tgclocals·d1e6514bc516778716e9d38209cf4ab8���������������������������������go.string."%x"0&%x go.string."%x"�Tgclocals·80320eec1018401d2b0daec3b250b99e00���Tgclocals·1ee14e32cec51f1cde6c2b0577d8188700�go.string."id"0&id go.string."id"�Tgclocals·5699c890da9a4c1a61d89d978591d077@@(�Tgclocals·bc335ce91c3a8b5f426dd201465802bd((����Tgclocals·04f43ee17c64d5db43a23c286d1bf23600� +�Tgclocals·cec9627e2837f98af62e9c7580b3bacc00�����$go.string."stream"0.stream $go.string."stream"�Tgclocals·bdc1cfaf863af97c7b8d007001384e8a��8,( + +eUUUUUUUUUU� � BU%TU�Tgclocals·7a383875e23784cb158d762414ce6278HH��������Ngo.string."Container %s is not running"`XContainer %s is not running Ngo.string."Container %s is not running"�go.string."UID"0(UID go.string."UID"�go.string."PID"0(PID go.string."PID"� go.string."PPID"0*PPID go.string."PPID"�go.string."C"0$C go.string."C"�"go.string."STIME"0,STIME "go.string."STIME"�go.string."TTY"0(TTY go.string."TTY"� go.string."TIME"0*TIME go.string."TIME"�go.string."CMD"0(CMD go.string."CMD"� go.string."root"0*root go.string."root"� go.string."7535"0*7535 go.string."7535"� go.string."7516"0*7516 go.string."7516"�go.string."0"0$0 go.string."0"�"go.string."03:20"0,03:20 "go.string."03:20"�go.string."?"0$? go.string."?"�(go.string."00:00:00"@200:00:00 (go.string."00:00:00"�Tgclocals·950db48493931155c4d72d2be7776567��P������ ������ �  �Tgclocals·a484a676faa0084ad5f98b43c17e101c�����������������Jgo.string."Container already running"`TContainer already running Jgo.string."Container already running"�Tgclocals·9cd0f1c7734d56b3c926d71ae19f8ec3pp$�� � �Tgclocals·f3828558443ce662a87feff12c09632b@@�������Bgo.string."Container not running"PLContainer not running Bgo.string."Container not running"�Tgclocals·97d2741936c7bda613787afceb8adff3((�Tgclocals·bc335ce91c3a8b5f426dd201465802bd((����Hgo.string."Container already paused"`RContainer already paused Hgo.string."Container already paused"�Tgclocals·0a4b95df80c389fe7e338059324575e1 �Tgclocals·0b0af158856f2ab75a5e0667d877f9eb ���@go.string."Container not paused"PJContainer not paused @go.string."Container not paused"�Tgclocals·0a4b95df80c389fe7e338059324575e1 �Tgclocals·0b0af158856f2ab75a5e0667d877f9eb ���,�go.itab.*github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.StdWriter.io.Writer�Hgo.string."cannot hijack connection"`Rcannot hijack connection Hgo.string."cannot hijack connection"�Zgo.string."application/vnd.docker.raw-stream"pd!application/vnd.docker.raw-stream Zgo.string."application/vnd.docker.raw-stream"�Jgo.string."Container %q is running\n"`RContainer %q is running + Jgo.string."Container %q is running\n"�Rgo.string."Container %q is not running\n"`ZContainer %q is not running + Rgo.string."Container %q is not running\n"�4go.string."What happened?"@>What happened? 4go.string."What happened?"�� <<�Tgclocals·7a383875e23784cb158d762414ce6278HH��������Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·ee0e5af169bfc1eef210605652a1df80��$go.string."create"0.create $go.string."create"�"go.string."start"0,start "go.string."start"� go.string."stop"0*stop go.string."stop"�&go.string."destroy"00destroy &go.string."destroy"�2go.string."mybase:latest"@< mybase:latest 2go.string."mybase:latest"�Tgclocals·76950f6d0769389d26192c168dbb78a088� +�Tgclocals·7ba969af8c72fca351526f5bd553df3688�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·ee0e5af169bfc1eef210605652a1df80��Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·ee0e5af169bfc1eef210605652a1df80��go.string."Id"0&Id go.string."Id"�Tgclocals·b3446cef6b648ddae3c01581a04ccc0b�����H�$��"XIY�����*$���������� �� �Tgclocals·a484a676faa0084ad5f98b43c17e101c�����������������go.string."*"0$* go.string."*"�Tgclocals·349a065d14b607627da67d5600b2511a88 +�Tgclocals·8375af20f91e3bf26f9f4b100ffb7d0e88������Tgclocals·f883d3996c76325fd1714d4e3de9fa33 �Tgclocals·ee0e5af169bfc1eef210605652a1df80��Tgclocals·04f43ee17c64d5db43a23c286d1bf23600� +�Tgclocals·cec9627e2837f98af62e9c7580b3bacc00�����4go.string."exec not found"@>exec not found 4go.string."exec not found"�Tgclocals·4459bbba29917b2fee408f0dbbff89b188"�Tgclocals·dbefa26e1f0ee62688488e90e23fcbd788 +� + + +�6go.string."No such network"@@No such network 6go.string."No such network"�Tgclocals·4459bbba29917b2fee408f0dbbff89b188"�Tgclocals·8bd789dcce9d4daa4c4bb84dfe47e24788 +�- + + +�Tgclocals·429e38e879552ec65b0c30795e04b14b�� 8�� �� ������������ �� ���Tgclocals·908986cc2bd23e6b2b43c6b331d27560XX ����������Tgclocals·04f43ee17c64d5db43a23c286d1bf23600� +�Tgclocals·cec9627e2837f98af62e9c7580b3bacc00�����go.string."."0$. go.string."."�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·a08e9001cb8f9d822225de3b8e406515�@go.string."Invalid network name"PJInvalid network name @go.string."Invalid network name"�Dgo.string."network already exists"PNnetwork already exists Dgo.string."network already exists"�Tgclocals·770edcce2fa22bec4aa18365adf4562e��V �  "  �  �Tgclocals·a484a676faa0084ad5f98b43c17e101c�����������������Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·099986b79bd4df464b634a14757f9178+�Tgclocals·0809678294a6ccf1679e4ac422a0f629��L��Z� �Z�Z� � � �Tgclocals·9672a07f1a450fc594d7cd9cb2c95495PP++++++++�Pgo.string."^[a-zA-Z0-9][a-zA-Z0-9_.-]+$"`Z^[a-zA-Z0-9][a-zA-Z0-9_.-]+$ Pgo.string."^[a-zA-Z0-9][a-zA-Z0-9_.-]+$"�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·3280bececceccd33cb74587feedb1f9f�*"".nameRegexp&type.*regexp.Regexp�""".statictmp_0019�(type."".DockerServer�""".statictmp_0126 type.[1]string  go.string."POST"�""".statictmp_0130 type.[1]string  go.string."GET"�""".statictmp_0134 type.[1]string  go.string."POST"�""".statictmp_0138 type.[1]string  go.string."GET"�""".statictmp_0142 type.[1]string  go.string."POST"�""".statictmp_0146 type.[1]string  go.string."GET"�""".statictmp_0150 type.[1]string  go.string."POST"�""".statictmp_0154 type.[1]string  go.string."POST"�""".statictmp_0158 type.[1]string  go.string."POST"�""".statictmp_0162 type.[1]string  go.string."POST"�""".statictmp_0166 type.[1]string  go.string."POST"�""".statictmp_0170 type.[1]string  go.string."POST"�""".statictmp_0174 type.[1]string  go.string."POST"�""".statictmp_0178 type.[1]string  $go.string."DELETE"�""".statictmp_0182 type.[1]string  go.string."POST"�""".statictmp_0186 type.[1]string  go.string."GET"�""".statictmp_0190 type.[1]string  go.string."POST"�""".statictmp_0194 type.[1]string  go.string."POST"�""".statictmp_0198 type.[1]string  go.string."GET"�""".statictmp_0202 type.[1]string  go.string."POST"�""".statictmp_0206 type.[1]string  go.string."POST"�""".statictmp_0210 type.[1]string  go.string."GET"�""".statictmp_0214 type.[1]string  $go.string."DELETE"�""".statictmp_0218 type.[1]string  go.string."GET"�""".statictmp_0222 type.[1]string  go.string."POST"�""".statictmp_0226 type.[1]string  go.string."POST"�""".statictmp_0230 type.[1]string  go.string."GET"�""".statictmp_0234 type.[1]string  go.string."GET"�""".statictmp_0238 type.[1]string  go.string."POST"�""".statictmp_0242 type.[1]string  go.string."GET"�""".statictmp_0246 type.[1]string  go.string."GET"�""".statictmp_0250 type.[1]string  go.string."GET"�""".statictmp_0254 type.[1]string  go.string."POST"�""".statictmp_0435�htype.github.com/fsouza/go-dockerclient.APIContainers�""".statictmp_0534@jtype.[1]github.com/fsouza/go-dockerclient.PortBinding  &go.string."0.0.0.0"�""".statictmp_0542�`type.github.com/fsouza/go-dockerclient.Container��""".statictmp_0647�type.[8]string� go.string."UID"  go.string."PID"@ go.string."PPID"` go.string."C"� "go.string."STIME"� go.string."TTY"� go.string."TIME"� go.string."CMD"�""".statictmp_0652�type.[8]string� go.string."root"  go.string."7535"@ go.string."7516"` go.string."0"� "go.string."03:20"� go.string."?"� (go.string."00:00:00"�""".statictmp_0817�Xtype.github.com/fsouza/go-dockerclient.Image�,"".initdone·type.uint8�"".NewServer·f"".NewServer�net.Listen·fnet.Listen�(runtime.newobject·f"runtime.newobject�$runtime.makemap·fruntime.makemap�4runtime.writebarrierfat·f.runtime.writebarrierfat�@"".(*DockerServer).buildMuxer·f:"".(*DockerServer).buildMuxer�&runtime.typ2Itab·f runtime.typ2Itab�"net/http.Serve·fnet/http.Serve�$runtime.newproc·fruntime.newproc�,runtime.throwreturn·f&runtime.throwreturn�8"".(*DockerServer).notify·f2"".(*DockerServer).notify�(runtime.chansend1·f"runtime.chansend1�4runtime.writebarrierptr·f.runtime.writebarrierptr��github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).Path·f�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).Path��github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).Methods·f�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).Methods�X"".*DockerServer.("".commitContainer)·fm·fR"".*DockerServer.("".commitContainer)·fm�H"".(*DockerServer).handlerWrapper·fB"".(*DockerServer).handlerWrapper��github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).HandlerFunc·f�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Route).HandlerFunc�V"".*DockerServer.("".listContainers)·fm·fP"".*DockerServer.("".listContainers)·fm�X"".*DockerServer.("".createContainer)·fm·fR"".*DockerServer.("".createContainer)·fm�Z"".*DockerServer.("".inspectContainer)·fm·fT"".*DockerServer.("".inspectContainer)·fm�X"".*DockerServer.("".renameContainer)·fm·fR"".*DockerServer.("".renameContainer)·fm�R"".*DockerServer.("".topContainer)·fm·fL"".*DockerServer.("".topContainer)·fm�V"".*DockerServer.("".startContainer)·fm·fP"".*DockerServer.("".startContainer)·fm�T"".*DockerServer.("".stopContainer)·fm·fN"".*DockerServer.("".stopContainer)·fm�V"".*DockerServer.("".pauseContainer)·fm·fP"".*DockerServer.("".pauseContainer)·fm�Z"".*DockerServer.("".unpauseContainer)·fm·fT"".*DockerServer.("".unpauseContainer)·fm�T"".*DockerServer.("".waitContainer)·fm·fN"".*DockerServer.("".waitContainer)·fm�X"".*DockerServer.("".attachContainer)·fm·fR"".*DockerServer.("".attachContainer)·fm�X"".*DockerServer.("".removeContainer)·fm·fR"".*DockerServer.("".removeContainer)·fm�`"".*DockerServer.("".createExecContainer)·fm·fZ"".*DockerServer.("".createExecContainer)·fm�V"".*DockerServer.("".statsContainer)·fm·fP"".*DockerServer.("".statsContainer)·fm�`"".*DockerServer.("".resizeExecContainer)·fm·fZ"".*DockerServer.("".resizeExecContainer)·fm�^"".*DockerServer.("".startExecContainer)·fm·fX"".*DockerServer.("".startExecContainer)·fm�b"".*DockerServer.("".inspectExecContainer)·fm·f\"".*DockerServer.("".inspectExecContainer)·fm�L"".*DockerServer.("".pullImage)·fm·fF"".*DockerServer.("".pullImage)·fm�N"".*DockerServer.("".buildImage)·fm·fH"".*DockerServer.("".buildImage)·fm�N"".*DockerServer.("".listImages)·fm·fH"".*DockerServer.("".listImages)·fm�P"".*DockerServer.("".removeImage)·fm·fJ"".*DockerServer.("".removeImage)·fm�R"".*DockerServer.("".inspectImage)·fm·fL"".*DockerServer.("".inspectImage)·fm�L"".*DockerServer.("".pushImage)·fm·fF"".*DockerServer.("".pushImage)·fm�J"".*DockerServer.("".tagImage)·fm·fD"".*DockerServer.("".tagImage)·fm�N"".*DockerServer.("".listEvents)·fm·fH"".*DockerServer.("".listEvents)·fm�N"".*DockerServer.("".pingDocker)·fm·fH"".*DockerServer.("".pingDocker)·fm�L"".*DockerServer.("".loadImage)·fm·fF"".*DockerServer.("".loadImage)·fm�J"".*DockerServer.("".getImage)·fm·fD"".*DockerServer.("".getImage)·fm�R"".*DockerServer.("".listNetworks)·fm·fL"".*DockerServer.("".listNetworks)·fm�P"".*DockerServer.("".networkInfo)·fm·fJ"".*DockerServer.("".networkInfo)·fm�T"".*DockerServer.("".createNetwork)·fm·fN"".*DockerServer.("".createNetwork)·fm�:"".(*DockerServer).SetHook·f4"".(*DockerServer).SetHook�B"".(*DockerServer).PrepareExec·f<"".(*DockerServer).PrepareExec�*runtime.mapassign1·f$runtime.mapassign1�D"".(*DockerServer).PrepareStats·f>"".(*DockerServer).PrepareStats�H"".(*DockerServer).PrepareFailure·fB"".(*DockerServer).PrepareFailure�T"".(*DockerServer).PrepareMultiFailures·fN"".(*DockerServer).PrepareMultiFailures�(runtime.growslice·f"runtime.growslice�8runtime.writebarrierslice·f2runtime.writebarrierslice�D"".(*DockerServer).ResetFailure·f>"".(*DockerServer).ResetFailure�(runtime.mapdelete·f"runtime.mapdelete�P"".(*DockerServer).ResetMultiFailures·fJ"".(*DockerServer).ResetMultiFailures�F"".(*DockerServer).CustomHandler·f@"".(*DockerServer).CustomHandler�.sync.(*RWMutex).Lock·f(sync.(*RWMutex).Lock�2sync.(*RWMutex).Unlock·f,sync.(*RWMutex).Unlock�J"".(*DockerServer).MutateContainer·fD"".(*DockerServer).MutateContainer�&runtime.eqstring·f runtime.eqstring�:runtime.writebarrierstring·f4runtime.writebarrierstring�4"".(*DockerServer).Stop·f."".(*DockerServer).Stop�2"".(*DockerServer).URL·f,"".(*DockerServer).URL�0runtime.concatstring3·f*runtime.concatstring3�>"".(*DockerServer).ServeHTTP·f8"".(*DockerServer).ServeHTTP�0sync.(*RWMutex).RLock·f*sync.(*RWMutex).RLock�4sync.(*RWMutex).RUnlock·f.sync.(*RWMutex).RUnlock�(runtime.deferproc·f"runtime.deferproc�,runtime.deferreturn·f&runtime.deferreturn�,runtime.mapiterinit·f&runtime.mapiterinit�,runtime.mapiternext·f&runtime.mapiternext�*regexp.MatchString·f$regexp.MatchString��github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).ServeHTTP·f�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.(*Router).ServeHTTP�H"".(*DockerServer).DefaultHandler·fB"".(*DockerServer).DefaultHandler�"".func·001·f"".func·001�H"".(*DockerServer).listContainers·fB"".(*DockerServer).listContainers�.net/url.(*URL).Query·f(net/url.(*URL).Query�:runtime.mapaccess2_faststr·f4runtime.mapaccess2_faststr�*runtime.panicindex·f$runtime.panicindex�(runtime.makeslice·f"runtime.makeslice�strings.Join·fstrings.Join�$runtime.convT2E·fruntime.convT2E�8runtime.writebarrieriface·f2runtime.writebarrieriface�fmt.Sprintf·ffmt.Sprintf�hgithub.com/fsouza/go-dockerclient.(*State).String·fbgithub.com/fsouza/go-dockerclient.(*State).String��github.com/fsouza/go-dockerclient.(*NetworkSettings).PortMappingAPI·f�github.com/fsouza/go-dockerclient.(*NetworkSettings).PortMappingAPI�,net/http.Header.Set·f&net/http.Header.Set�$runtime.convI2I·fruntime.convI2I�Dencoding/json.(*Encoder).Encode·f>encoding/json.(*Encoder).Encode�@"".(*DockerServer).listImages·f:"".(*DockerServer).listImages�>"".(*DockerServer).findImage·f8"".(*DockerServer).findImage�F"".(*DockerServer).findImageByID·f@"".(*DockerServer).findImageByID�J"".(*DockerServer).createContainer·fD"".(*DockerServer).createContainer�Dencoding/json.(*Decoder).Decode·f>encoding/json.(*Decoder).Decode�"net/http.Error·fnet/http.Error�>regexp.(*Regexp).MatchString·f8regexp.(*Regexp).MatchString� math/rand.Int·fmath/rand.Int�strconv.Itoa·fstrconv.Itoa�*runtime.panicslice·f$runtime.panicslice�@"".(*DockerServer).generateID·f:"".(*DockerServer).generateID�time.Now·ftime.Now�&crypto/rand.Read·f crypto/rand.Read�J"".(*DockerServer).renameContainer·fD"".(*DockerServer).renameContainer��github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Vars·f�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Vars�:runtime.mapaccess1_faststr·f4runtime.mapaccess1_faststr�F"".(*DockerServer).findContainer·f@"".(*DockerServer).findContainer�L"".(*DockerServer).inspectContainer·fF"".(*DockerServer).inspectContainer�H"".(*DockerServer).statsContainer·fB"".(*DockerServer).statsContainer�(strconv.ParseBool·f"strconv.ParseBool�D"".(*DockerServer).topContainer·f>"".(*DockerServer).topContainer�fmt.Fprintf·ffmt.Fprintf�H"".(*DockerServer).startContainer·fB"".(*DockerServer).startContainer�F"".(*DockerServer).stopContainer·f@"".(*DockerServer).stopContainer�H"".(*DockerServer).pauseContainer·fB"".(*DockerServer).pauseContainer�L"".(*DockerServer).unpauseContainer·fF"".(*DockerServer).unpauseContainer�J"".(*DockerServer).attachContainer·fD"".(*DockerServer).attachContainer�*runtime.assertI2I2·f$runtime.assertI2I2�fmt.Fprintln·ffmt.Fprintln�F"".(*DockerServer).waitContainer·f@"".(*DockerServer).waitContainer�time.Sleep·ftime.Sleep�J"".(*DockerServer).removeContainer·fD"".(*DockerServer).removeContainer�J"".(*DockerServer).commitContainer·fD"".(*DockerServer).commitContainer�8runtime.stringtoslicebyte·f2runtime.stringtoslicebyte�4encoding/json.Unmarshal·f.encoding/json.Unmarshal�0runtime.concatstring2·f*runtime.concatstring2�@"".(*DockerServer).buildImage·f:"".(*DockerServer).buildImage�,net/http.Header.Get·f&net/http.Header.Get�:archive/tar.(*Reader).Next·f4archive/tar.(*Reader).Next�>"".(*DockerServer).pullImage·f8"".(*DockerServer).pullImage�>"".(*DockerServer).pushImage·f8"".(*DockerServer).pushImage�<"".(*DockerServer).tagImage·f6"".(*DockerServer).tagImage�B"".(*DockerServer).removeImage·f<"".(*DockerServer).removeImage�D"".(*DockerServer).inspectImage·f>"".(*DockerServer).inspectImage�@"".(*DockerServer).listEvents·f:"".(*DockerServer).listEvents�"math/rand.Intn·fmath/rand.Intn�F"".(*DockerServer).generateEvent·f@"".(*DockerServer).generateEvent�0encoding/json.Marshal·f*encoding/json.Marshal�@"".(*DockerServer).pingDocker·f:"".(*DockerServer).pingDocker�>"".(*DockerServer).loadImage·f8"".(*DockerServer).loadImage�<"".(*DockerServer).getImage·f6"".(*DockerServer).getImage�R"".(*DockerServer).createExecContainer·fL"".(*DockerServer).createExecContainer�P"".(*DockerServer).startExecContainer·fJ"".(*DockerServer).startExecContainer�:"".(*DockerServer).getExec·f4"".(*DockerServer).getExec�R"".(*DockerServer).resizeExecContainer·fL"".(*DockerServer).resizeExecContainer�T"".(*DockerServer).inspectExecContainer·fN"".(*DockerServer).inspectExecContainer�B"".(*DockerServer).findNetwork·f<"".(*DockerServer).findNetwork�D"".(*DockerServer).listNetworks·f>"".(*DockerServer).listNetworks�B"".(*DockerServer).networkInfo·f<"".(*DockerServer).networkInfo�""".isValidName·f"".isValidName�&strings.Contains·f strings.Contains�F"".(*DockerServer).createNetwork·f@"".(*DockerServer).createNetwork�$runtime.memmove·fruntime.memmove�"".init·f"".init�(runtime.throwinit·f"runtime.throwinit��github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.init·f�github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.init��github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.init·f�github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy.init�Rgithub.com/fsouza/go-dockerclient.init·fLgithub.com/fsouza/go-dockerclient.init�time.init·ftime.init�sync.init·fsync.init�strings.init·fstrings.init�strconv.init·fstrconv.init�regexp.init·fregexp.init� net/http.init·fnet/http.init�net.init·fnet.init�"math/rand.init·fmath/rand.init�fmt.init·ffmt.init�*encoding/json.init·f$encoding/json.init�&crypto/rand.init·f crypto/rand.init�&archive/tar.init·f archive/tar.init�*regexp.MustCompile·f$regexp.MustCompile�bruntime.gcbits.0x48844400000000000000000000000000 H�D�(go.string."[]string"@2[]string (go.string."[]string"�type.[]string��Ө� + � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P(go.string."[]string"p,go.weak.type.*[]string�"runtime.zerovalue�type.string�:go.typelink.[]string/[]stringtype.[]string�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�0type..hashfunc.[8]string(type..hash.[8]string�,type..eqfunc.[8]string$type..eq.[8]string�&type..alg.[8]string 0type..hashfunc.[8]string,type..eqfunc.[8]string�bruntime.gcbits.0x48484848484848480000000000000000 HHHHHHHH�*go.string."[8]string"@4 [8]string *go.string."[8]string"�type.[8]string���US�> &type..alg.[8]string0bruntime.gcbits.0x48484848484848480000000000000000P*go.string."[8]string"p.go.weak.type.*[8]string�"runtime.zerovalue�type.string�type.[]string�>go.typelink.[8]string/[8]stringtype.[8]string�bruntime.gcbits.0x88000000000000000000000000000000 ��Jgo.string."*map.bucket[string]string"`T*map.bucket[string]string Jgo.string."*map.bucket[string]string"�Y� � runtime.algarray0Btype..gc.map.bucket[string]string@Jtype..gcprog.map.bucket[string]stringPHgo.string."map.bucket[string]string"pLgo.weak.type.*map.bucket[string]string�"runtime.zerovalue��:type.map.bucket[string]string� go.string."keys"�type.[8]string�$go.string."values"�type.[8]string�(go.string."overflow"�go.weak.type.*map[string]string�"runtime.zerovalue�type.string�type.string�:type.map.bucket[string]string�4type.map.hdr[string]string�^go.typelink.map[string]string/map[string]string,type.map[string]string�$go.string."func()"0.func() $go.string."func()"�type.func()������3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P$go.string."func()"p(go.weak.type.*func()�"runtime.zerovalue��type.func()��type.func()�(go.string."[]func()"@2[]func() (go.string."[]func()"�type.[]func()��=��% � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P(go.string."[]func()"p,go.weak.type.*[]func()�"runtime.zerovalue�type.func()�:go.typelink.[]func()/[]func()type.[]func()�bruntime.gcbits.0x88888888000000000000000000000000 �����*go.string."[8]func()"@4 [8]func() *go.string."[8]func()"�type.[8]func()��@���o � runtime.algarray0bruntime.gcbits.0x88888888000000000000000000000000P*go.string."[8]func()"p.go.weak.type.*[8]func()�"runtime.zerovalue�type.func()�type.[]func()�>go.typelink.[8]func()/[8]func()type.[8]func()�Jgo.string."*map.bucket[string]func()"`T*map.bucket[string]func() Jgo.string."*map.bucket[string]func()"�go.weak.type.*map[string]func()�"runtime.zerovalue�type.string�type.func()�:type.map.bucket[string]func()�4type.map.hdr[string]func()�^go.typelink.map[string]func()/map[string]func(),type.map[string]func()�Jgo.string."func(string) docker.Stats"`Tfunc(string) docker.Stats Jgo.string."func(string) docker.Stats"�rtype.func(string) github.com/fsouza/go-dockerclient.Stats����,3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PJgo.string."func(string) docker.Stats"p�go.weak.type.*func(string) github.com/fsouza/go-dockerclient.Stats�"runtime.zerovalue��rtype.func(string) github.com/fsouza/go-dockerclient.Stats��rtype.func(string) github.com/fsouza/go-dockerclient.Stats�type.string�Xtype.github.com/fsouza/go-dockerclient.Stats�Ngo.string."[]func(string) docker.Stats"`X[]func(string) docker.Stats Ngo.string."[]func(string) docker.Stats"�vtype.[]func(string) github.com/fsouza/go-dockerclient.Stats����� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000PNgo.string."[]func(string) docker.Stats"p�go.weak.type.*[]func(string) github.com/fsouza/go-dockerclient.Stats�"runtime.zerovalue�rtype.func(string) github.com/fsouza/go-dockerclient.Stats��go.typelink.[]func(string) docker.Stats/[]func(string) github.com/fsouza/go-dockerclient.Statsvtype.[]func(string) github.com/fsouza/go-dockerclient.Stats�Pgo.string."[8]func(string) docker.Stats"`Z[8]func(string) docker.Stats Pgo.string."[8]func(string) docker.Stats"�xtype.[8]func(string) github.com/fsouza/go-dockerclient.Stats��@��� � runtime.algarray0bruntime.gcbits.0x88888888000000000000000000000000PPgo.string."[8]func(string) docker.Stats"p�go.weak.type.*[8]func(string) github.com/fsouza/go-dockerclient.Stats�"runtime.zerovalue�rtype.func(string) github.com/fsouza/go-dockerclient.Stats�vtype.[]func(string) github.com/fsouza/go-dockerclient.Stats��go.typelink.[8]func(string) docker.Stats/[8]func(string) github.com/fsouza/go-dockerclient.Statsxtype.[8]func(string) github.com/fsouza/go-dockerclient.Stats�pgo.string."*map.bucket[string]func(string) docker.Stats"�z,*map.bucket[string]func(string) docker.Stats pgo.string."*map.bucket[string]func(string) docker.Stats"��type.*map.bucket[string]func(string) github.com/fsouza/go-dockerclient.Stats��t}8�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ppgo.string."*map.bucket[string]func(string) docker.Stats"p�go.weak.type.**map.bucket[string]func(string) github.com/fsouza/go-dockerclient.Stats�"runtime.zerovalue��type.map.bucket[string]func(string) github.com/fsouza/go-dockerclient.Stats�ngo.string."map.bucket[string]func(string) docker.Stats"�x+map.bucket[string]func(string) docker.Stats ngo.string."map.bucket[string]func(string) docker.Stats"��type.map.bucket[string]func(string) github.com/fsouza/go-dockerclient.Stats������5�� � runtime.algarray0bruntime.gcbits.0x84848484848484848488888888000000Pngo.string."map.bucket[string]func(string) docker.Stats"p�go.weak.type.*map.bucket[string]func(string) github.com/fsouza/go-dockerclient.Stats�"runtime.zerovalue���type.map.bucket[string]func(string) github.com/fsouza/go-dockerclient.Stats� go.string."keys"�type.[8]string�$go.string."values"�xtype.[8]func(string) github.com/fsouza/go-dockerclient.Stats�(go.string."overflow"��type.*map.bucket[string]func(string) github.com/fsouza/go-dockerclient.Stats�hgo.string."map.hdr[string]func(string) docker.Stats"�r(map.hdr[string]func(string) docker.Stats hgo.string."map.hdr[string]func(string) docker.Stats"��type.map.hdr[string]func(string) github.com/fsouza/go-dockerclient.Stats��0ǟ  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000Phgo.string."map.hdr[string]func(string) docker.Stats"p�go.weak.type.*map.hdr[string]func(string) github.com/fsouza/go-dockerclient.Stats�"runtime.zerovalue���type.map.hdr[string]func(string) github.com/fsouza/go-dockerclient.Stats�&go.string."buckets"��type.*map.bucket[string]func(string) github.com/fsouza/go-dockerclient.Stats�,go.string."oldbuckets"��type.*map.bucket[string]func(string) github.com/fsouza/go-dockerclient.Stats�`go.string."map[string]func(string) docker.Stats"pj$map[string]func(string) docker.Stats `go.string."map[string]func(string) docker.Stats"��type.map[string]func(string) github.com/fsouza/go-dockerclient.Stats��z��5� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P`go.string."map[string]func(string) docker.Stats"p�go.weak.type.*map[string]func(string) github.com/fsouza/go-dockerclient.Stats�"runtime.zerovalue�type.string�rtype.func(string) github.com/fsouza/go-dockerclient.Stats��type.map.bucket[string]func(string) github.com/fsouza/go-dockerclient.Stats��type.map.hdr[string]func(string) github.com/fsouza/go-dockerclient.Stats��go.typelink.map[string]func(string) docker.Stats/map[string]func(string) github.com/fsouza/go-dockerclient.Stats�type.map[string]func(string) github.com/fsouza/go-dockerclient.Stats�4go.string."[]http.Handler"@>[]http.Handler 4go.string."[]http.Handler"�.type.[]net/http.Handler��c�� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P4go.string."[]http.Handler"p@go.weak.type.*[]net/http.Handler�"runtime.zerovalue�*type.net/http.Handler�Zgo.typelink.[]http.Handler/[]net/http.Handler.type.[]net/http.Handler�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�Dtype..hashfunc.[8]net/http.Handlergo.string."[]*docker.Container"PH[]*docker.Container >go.string."[]*docker.Container"�ftype.[]*github.com/fsouza/go-dockerclient.Container������ � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P>go.string."[]*docker.Container"pxgo.weak.type.*[]*github.com/fsouza/go-dockerclient.Container�"runtime.zerovalue�btype.*github.com/fsouza/go-dockerclient.Container��go.typelink.[]*docker.Container/[]*github.com/fsouza/go-dockerclient.Containerftype.[]*github.com/fsouza/go-dockerclient.Container�Bgo.string."[]*docker.ExecInspect"PL[]*docker.ExecInspect Bgo.string."[]*docker.ExecInspect"�jtype.[]*github.com/fsouza/go-dockerclient.ExecInspect��Y�&6 � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000PBgo.string."[]*docker.ExecInspect"p|go.weak.type.*[]*github.com/fsouza/go-dockerclient.ExecInspect�"runtime.zerovalue�ftype.*github.com/fsouza/go-dockerclient.ExecInspect��go.typelink.[]*docker.ExecInspect/[]*github.com/fsouza/go-dockerclient.ExecInspectjtype.[]*github.com/fsouza/go-dockerclient.ExecInspect�4go.string."[]docker.Image"@>[]docker.Image 4go.string."[]docker.Image"�\type.[]github.com/fsouza/go-dockerclient.Image�����r � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P4go.string."[]docker.Image"pngo.weak.type.*[]github.com/fsouza/go-dockerclient.Image�"runtime.zerovalue�Xtype.github.com/fsouza/go-dockerclient.Image��go.typelink.[]docker.Image/[]github.com/fsouza/go-dockerclient.Image\type.[]github.com/fsouza/go-dockerclient.Image�:go.string."[]*docker.Network"PD[]*docker.Network :go.string."[]*docker.Network"�btype.[]*github.com/fsouza/go-dockerclient.Network���\,i � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P:go.string."[]*docker.Network"ptgo.weak.type.*[]*github.com/fsouza/go-dockerclient.Network�"runtime.zerovalue�^type.*github.com/fsouza/go-dockerclient.Network��go.typelink.[]*docker.Network/[]*github.com/fsouza/go-dockerclient.Networkbtype.[]*github.com/fsouza/go-dockerclient.Network�>go.string."func(*http.Request)"PHfunc(*http.Request) >go.string."func(*http.Request)"�8type.func(*net/http.Request)��$I�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."func(*http.Request)"pJgo.weak.type.*func(*net/http.Request)�"runtime.zerovalue��8type.func(*net/http.Request)��8type.func(*net/http.Request)�,type.*net/http.Request�>go.string."[]map[string]string"PH[]map[string]string >go.string."[]map[string]string"�0type.[]map[string]string�����f � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P>go.string."[]map[string]string"pBgo.weak.type.*[]map[string]string�"runtime.zerovalue�,type.map[string]string�fgo.typelink.[]map[string]string/[]map[string]string0type.[]map[string]string�Hgo.string."chan<- *docker.Container"`Rchan<- *docker.Container Hgo.string."chan<- *docker.Container"�ptype.chan<- *github.com/fsouza/go-dockerclient.Container���w��2 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PHgo.string."chan<- *docker.Container"p�go.weak.type.*chan<- *github.com/fsouza/go-dockerclient.Container�"runtime.zerovalue�btype.*github.com/fsouza/go-dockerclient.Container��go.typelink.chan<- *docker.Container/chan<- *github.com/fsouza/go-dockerclient.Containerptype.chan<- *github.com/fsouza/go-dockerclient.Container�Bgo.string."*testing.DockerServer"PL*testing.DockerServer Bgo.string."*testing.DockerServer"�zgo.string."func(*testing.DockerServer, string, http.Handler)"��1func(*testing.DockerServer, string, http.Handler) zgo.string."func(*testing.DockerServer, string, http.Handler)"�jtype.func(*"".DockerServer, string, net/http.Handler)���30�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pzgo.string."func(*testing.DockerServer, string, http.Handler)"p|go.weak.type.*func(*"".DockerServer, string, net/http.Handler)�"runtime.zerovalue��jtype.func(*"".DockerServer, string, net/http.Handler)��jtype.func(*"".DockerServer, string, net/http.Handler)�*type.*"".DockerServer�type.string�*type.net/http.Handler�hgo.string."func(*testing.DockerServer) http.Handler"�r(func(*testing.DockerServer) http.Handler hgo.string."func(*testing.DockerServer) http.Handler"�Xtype.func(*"".DockerServer) net/http.Handler�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Phgo.string."func(*testing.DockerServer) http.Handler"pjgo.weak.type.*func(*"".DockerServer) net/http.Handler�"runtime.zerovalue��Xtype.func(*"".DockerServer) net/http.Handler��Xtype.func(*"".DockerServer) net/http.Handler�*type.*"".DockerServer�*type.net/http.Handler��go.string."func(*testing.DockerServer, string, docker.State) error"��7func(*testing.DockerServer, string, docker.State) error �go.string."func(*testing.DockerServer, string, docker.State) error"��type.func(*"".DockerServer, string, github.com/fsouza/go-dockerclient.State) error��M3P�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*testing.DockerServer, string, docker.State) error"p�go.weak.type.*func(*"".DockerServer, string, github.com/fsouza/go-dockerclient.State) error�"runtime.zerovalue���type.func(*"".DockerServer, string, github.com/fsouza/go-dockerclient.State) error���type.func(*"".DockerServer, string, github.com/fsouza/go-dockerclient.State) error�*type.*"".DockerServer�type.string�Xtype.github.com/fsouza/go-dockerclient.State�type.error�ngo.string."func(*testing.DockerServer, string, func())"�x+func(*testing.DockerServer, string, func()) ngo.string."func(*testing.DockerServer, string, func())"�Vtype.func(*"".DockerServer, string, func())���Աo3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pngo.string."func(*testing.DockerServer, string, func())"phgo.weak.type.*func(*"".DockerServer, string, func())�"runtime.zerovalue��Vtype.func(*"".DockerServer, string, func())��Vtype.func(*"".DockerServer, string, func())�*type.*"".DockerServer�type.string�type.func()�ngo.string."func(*testing.DockerServer, string, string)"�x+func(*testing.DockerServer, string, string) ngo.string."func(*testing.DockerServer, string, string)"�Vtype.func(*"".DockerServer, string, string)��-qX�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pngo.string."func(*testing.DockerServer, string, string)"phgo.weak.type.*func(*"".DockerServer, string, string)�"runtime.zerovalue��Vtype.func(*"".DockerServer, string, string)��Vtype.func(*"".DockerServer, string, string)�*type.*"".DockerServer�type.string�type.string��go.string."func(*testing.DockerServer, string, func(string) docker.Stats)"��>func(*testing.DockerServer, string, func(string) docker.Stats) �go.string."func(*testing.DockerServer, string, func(string) docker.Stats)"��type.func(*"".DockerServer, string, func(string) github.com/fsouza/go-dockerclient.Stats)��uB�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*testing.DockerServer, string, func(string) docker.Stats)"p�go.weak.type.*func(*"".DockerServer, string, func(string) github.com/fsouza/go-dockerclient.Stats)�"runtime.zerovalue���type.func(*"".DockerServer, string, func(string) github.com/fsouza/go-dockerclient.Stats)���type.func(*"".DockerServer, string, func(string) github.com/fsouza/go-dockerclient.Stats)�*type.*"".DockerServer�type.string�rtype.func(string) github.com/fsouza/go-dockerclient.Stats�^go.string."func(*testing.DockerServer, string)"ph#func(*testing.DockerServer, string) ^go.string."func(*testing.DockerServer, string)"�Ftype.func(*"".DockerServer, string)��:i2�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P^go.string."func(*testing.DockerServer, string)"pXgo.weak.type.*func(*"".DockerServer, string)�"runtime.zerovalue��Ftype.func(*"".DockerServer, string)��Ftype.func(*"".DockerServer, string)�*type.*"".DockerServer�type.string�Ngo.string."func(*testing.DockerServer)"`Xfunc(*testing.DockerServer) Ngo.string."func(*testing.DockerServer)"�6type.func(*"".DockerServer)���v�@3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PNgo.string."func(*testing.DockerServer)"pHgo.weak.type.*func(*"".DockerServer)�"runtime.zerovalue��6type.func(*"".DockerServer)��6type.func(*"".DockerServer)�*type.*"".DockerServer��go.string."func(*testing.DockerServer, http.ResponseWriter, *http.Request)"��?func(*testing.DockerServer, http.ResponseWriter, *http.Request) �go.string."func(*testing.DockerServer, http.ResponseWriter, *http.Request)"��type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)���]� +3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*testing.DockerServer, http.ResponseWriter, *http.Request)"p�go.weak.type.*func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�"runtime.zerovalue���type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)���type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�*type.*"".DockerServer�8type.net/http.ResponseWriter�,type.*net/http.Request�xgo.string."func(*testing.DockerServer, func(*http.Request))"��0func(*testing.DockerServer, func(*http.Request)) xgo.string."func(*testing.DockerServer, func(*http.Request))"�htype.func(*"".DockerServer, func(*net/http.Request))��I�*�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pxgo.string."func(*testing.DockerServer, func(*http.Request))"pzgo.weak.type.*func(*"".DockerServer, func(*net/http.Request))�"runtime.zerovalue��htype.func(*"".DockerServer, func(*net/http.Request))��htype.func(*"".DockerServer, func(*net/http.Request))�*type.*"".DockerServer�8type.func(*net/http.Request)�\go.string."func(*testing.DockerServer) string"pf"func(*testing.DockerServer) string \go.string."func(*testing.DockerServer) string"�Dtype.func(*"".DockerServer) string��F��p3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P\go.string."func(*testing.DockerServer) string"pVgo.weak.type.*func(*"".DockerServer) string�"runtime.zerovalue��Dtype.func(*"".DockerServer) string��Dtype.func(*"".DockerServer) string�*type.*"".DockerServer�type.string��go.string."func(*testing.DockerServer, string) (*docker.Container, int, error)"��Cfunc(*testing.DockerServer, string) (*docker.Container, int, error) �go.string."func(*testing.DockerServer, string) (*docker.Container, int, error)"��type.func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.Container, int, error)���r�$3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*testing.DockerServer, string) (*docker.Container, int, error)"p�go.weak.type.*func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.Container, int, error)�"runtime.zerovalue���type.func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.Container, int, error)���type.func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.Container, int, error)�*type.*"".DockerServer�type.string�btype.*github.com/fsouza/go-dockerclient.Container�type.int�type.error�~go.string."func(*testing.DockerServer, string) (string, error)"��3func(*testing.DockerServer, string) (string, error) ~go.string."func(*testing.DockerServer, string) (string, error)"�ftype.func(*"".DockerServer, string) (string, error)����ti3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P~go.string."func(*testing.DockerServer, string) (string, error)"pxgo.weak.type.*func(*"".DockerServer, string) (string, error)�"runtime.zerovalue��ftype.func(*"".DockerServer, string) (string, error)��ftype.func(*"".DockerServer, string) (string, error)�*type.*"".DockerServer�type.string�type.string�type.error��go.string."func(*testing.DockerServer, string) (string, int, error)"��8func(*testing.DockerServer, string) (string, int, error) �go.string."func(*testing.DockerServer, string) (string, int, error)"�ptype.func(*"".DockerServer, string) (string, int, error)��C �O3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*testing.DockerServer, string) (string, int, error)"p�go.weak.type.*func(*"".DockerServer, string) (string, int, error)�"runtime.zerovalue��ptype.func(*"".DockerServer, string) (string, int, error)��ptype.func(*"".DockerServer, string) (string, int, error)�*type.*"".DockerServer�type.string�type.string�type.int�type.error��go.string."func(*testing.DockerServer, string) (*docker.Network, int, error)"��Afunc(*testing.DockerServer, string) (*docker.Network, int, error) �go.string."func(*testing.DockerServer, string) (*docker.Network, int, error)"��type.func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.Network, int, error)�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*testing.DockerServer, string) (*docker.Network, int, error)"p�go.weak.type.*func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.Network, int, error)�"runtime.zerovalue���type.func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.Network, int, error)���type.func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.Network, int, error)�*type.*"".DockerServer�type.string�^type.*github.com/fsouza/go-dockerclient.Network�type.int�type.error�rgo.string."func(*testing.DockerServer) *docker.APIEvents"�|-func(*testing.DockerServer) *docker.APIEvents rgo.string."func(*testing.DockerServer) *docker.APIEvents"��type.func(*"".DockerServer) *github.com/fsouza/go-dockerclient.APIEvents��v�Pv3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Prgo.string."func(*testing.DockerServer) *docker.APIEvents"p�go.weak.type.*func(*"".DockerServer) *github.com/fsouza/go-dockerclient.APIEvents�"runtime.zerovalue���type.func(*"".DockerServer) *github.com/fsouza/go-dockerclient.APIEvents���type.func(*"".DockerServer) *github.com/fsouza/go-dockerclient.APIEvents�*type.*"".DockerServer�btype.*github.com/fsouza/go-dockerclient.APIEvents��go.string."func(*testing.DockerServer, string) (*docker.ExecInspect, error)"��@func(*testing.DockerServer, string) (*docker.ExecInspect, error) �go.string."func(*testing.DockerServer, string) (*docker.ExecInspect, error)"��type.func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.ExecInspect, error)��,� =3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(*testing.DockerServer, string) (*docker.ExecInspect, error)"p�go.weak.type.*func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.ExecInspect, error)�"runtime.zerovalue���type.func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.ExecInspect, error)���type.func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.ExecInspect, error)�*type.*"".DockerServer�type.string�ftype.*github.com/fsouza/go-dockerclient.ExecInspect�type.error�hgo.string."func(http.ResponseWriter, *http.Request)"�r(func(http.ResponseWriter, *http.Request) hgo.string."func(http.ResponseWriter, *http.Request)"�jtype.func(net/http.ResponseWriter, *net/http.Request)�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Phgo.string."func(http.ResponseWriter, *http.Request)"p|go.weak.type.*func(net/http.ResponseWriter, *net/http.Request)�"runtime.zerovalue��jtype.func(net/http.ResponseWriter, *net/http.Request)��jtype.func(net/http.ResponseWriter, *net/http.Request)�8type.net/http.ResponseWriter�,type.*net/http.Request�""..gostring.1��nfunc(*testing.DockerServer, func(http.ResponseWriter, *http.Request)) func(http.ResponseWriter, *http.Request) ""..gostring.1��type.func(*"".DockerServer, func(net/http.ResponseWriter, *net/http.Request)) func(net/http.ResponseWriter, *net/http.Request)��0|)3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P""..gostring.1p�go.weak.type.*func(*"".DockerServer, func(net/http.ResponseWriter, *net/http.Request)) func(net/http.ResponseWriter, *net/http.Request)�"runtime.zerovalue���type.func(*"".DockerServer, func(net/http.ResponseWriter, *net/http.Request)) func(net/http.ResponseWriter, *net/http.Request)���type.func(*"".DockerServer, func(net/http.ResponseWriter, *net/http.Request)) func(net/http.ResponseWriter, *net/http.Request)�*type.*"".DockerServer�jtype.func(net/http.ResponseWriter, *net/http.Request)�jtype.func(net/http.ResponseWriter, *net/http.Request)�tgo.string."func(*testing.DockerServer, *docker.Container)"�~.func(*testing.DockerServer, *docker.Container) tgo.string."func(*testing.DockerServer, *docker.Container)"��type.func(*"".DockerServer, *github.com/fsouza/go-dockerclient.Container)���Ƹs3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ptgo.string."func(*testing.DockerServer, *docker.Container)"p�go.weak.type.*func(*"".DockerServer, *github.com/fsouza/go-dockerclient.Container)�"runtime.zerovalue���type.func(*"".DockerServer, *github.com/fsouza/go-dockerclient.Container)���type.func(*"".DockerServer, *github.com/fsouza/go-dockerclient.Container)�*type.*"".DockerServer�btype.*github.com/fsouza/go-dockerclient.Container�2go.string."CustomHandler"@< CustomHandler 2go.string."CustomHandler"�Lgo.string."func(string, http.Handler)"`Vfunc(string, http.Handler) Lgo.string."func(string, http.Handler)"�Ftype.func(string, net/http.Handler)��)m� 3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PLgo.string."func(string, http.Handler)"pXgo.weak.type.*func(string, net/http.Handler)�"runtime.zerovalue��Ftype.func(string, net/http.Handler)��Ftype.func(string, net/http.Handler)�type.string�*type.net/http.Handler�4go.string."DefaultHandler"@>DefaultHandler 4go.string."DefaultHandler"�>go.string."func() http.Handler"PHfunc() http.Handler >go.string."func() http.Handler"�8type.func() net/http.Handler����53 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P>go.string."func() http.Handler"pJgo.weak.type.*func() net/http.Handler�"runtime.zerovalue��8type.func() net/http.Handler��8type.func() net/http.Handler�*type.net/http.Handler�6go.string."MutateContainer"@@MutateContainer 6go.string."MutateContainer"�Xgo.string."func(string, docker.State) error"pb func(string, docker.State) error Xgo.string."func(string, docker.State) error"��type.func(string, github.com/fsouza/go-dockerclient.State) error��.� 3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PXgo.string."func(string, docker.State) error"p�go.weak.type.*func(string, github.com/fsouza/go-dockerclient.State) error�"runtime.zerovalue���type.func(string, github.com/fsouza/go-dockerclient.State) error���type.func(string, github.com/fsouza/go-dockerclient.State) error�type.string�Xtype.github.com/fsouza/go-dockerclient.State�type.error�.go.string."PrepareExec"@8 PrepareExec .go.string."PrepareExec"�@go.string."func(string, func())"PJfunc(string, func()) @go.string."func(string, func())"�2type.func(string, func())���+>�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P@go.string."func(string, func())"pDgo.weak.type.*func(string, func())�"runtime.zerovalue��2type.func(string, func())��2type.func(string, func())�type.string�type.func()�4go.string."PrepareFailure"@>PrepareFailure 4go.string."PrepareFailure"�@go.string."func(string, string)"PJfunc(string, string) @go.string."func(string, string)"�2type.func(string, string)���!��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P@go.string."func(string, string)"pDgo.weak.type.*func(string, string)�"runtime.zerovalue��2type.func(string, string)��2type.func(string, string)�type.string�type.string�@go.string."PrepareMultiFailures"PJPrepareMultiFailures @go.string."PrepareMultiFailures"�0go.string."PrepareStats"@: PrepareStats 0go.string."PrepareStats"�fgo.string."func(string, func(string) docker.Stats)"pp'func(string, func(string) docker.Stats) fgo.string."func(string, func(string) docker.Stats)"��type.func(string, func(string) github.com/fsouza/go-dockerclient.Stats)��;�=3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pfgo.string."func(string, func(string) docker.Stats)"p�go.weak.type.*func(string, func(string) github.com/fsouza/go-dockerclient.Stats)�"runtime.zerovalue���type.func(string, func(string) github.com/fsouza/go-dockerclient.Stats)���type.func(string, func(string) github.com/fsouza/go-dockerclient.Stats)�type.string�rtype.func(string) github.com/fsouza/go-dockerclient.Stats�0go.string."ResetFailure"@: ResetFailure 0go.string."ResetFailure"�0go.string."func(string)"@: func(string) 0go.string."func(string)"�"type.func(string)���ǹ�3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P0go.string."func(string)"p4go.weak.type.*func(string)�"runtime.zerovalue��"type.func(string)��"type.func(string)�type.string�go.string."createExecContainer"PHcreateExecContainer >go.string."createExecContainer"�2go.string."createNetwork"@< createNetwork 2go.string."createNetwork"�2go.string."findContainer"@< findContainer 2go.string."findContainer"�pgo.string."func(string) (*docker.Container, int, error)"�z,func(string) (*docker.Container, int, error) pgo.string."func(string) (*docker.Container, int, error)"��type.func(string) (*github.com/fsouza/go-dockerclient.Container, int, error)��4��`3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ppgo.string."func(string) (*docker.Container, int, error)"p�go.weak.type.*func(string) (*github.com/fsouza/go-dockerclient.Container, int, error)�"runtime.zerovalue���type.func(string) (*github.com/fsouza/go-dockerclient.Container, int, error)���type.func(string) (*github.com/fsouza/go-dockerclient.Container, int, error)�type.string�btype.*github.com/fsouza/go-dockerclient.Container�type.int�type.error�*go.string."findImage"@4 findImage *go.string."findImage"�Pgo.string."func(string) (string, error)"`Zfunc(string) (string, error) Pgo.string."func(string) (string, error)"�Btype.func(string) (string, error)��D +j+3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PPgo.string."func(string) (string, error)"pTgo.weak.type.*func(string) (string, error)�"runtime.zerovalue��Btype.func(string) (string, error)��Btype.func(string) (string, error)�type.string�type.string�type.error�2go.string."findImageByID"@< findImageByID 2go.string."findImageByID"�Zgo.string."func(string) (string, int, error)"pd!func(string) (string, int, error) Zgo.string."func(string) (string, int, error)"�Ltype.func(string) (string, int, error)����(>3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PZgo.string."func(string) (string, int, error)"p^go.weak.type.*func(string) (string, int, error)�"runtime.zerovalue��Ltype.func(string) (string, int, error)��Ltype.func(string) (string, int, error)�type.string�type.string�type.int�type.error�.go.string."findNetwork"@8 findNetwork .go.string."findNetwork"�lgo.string."func(string) (*docker.Network, int, error)"�v*func(string) (*docker.Network, int, error) lgo.string."func(string) (*docker.Network, int, error)"��type.func(string) (*github.com/fsouza/go-dockerclient.Network, int, error)��%��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Plgo.string."func(string) (*docker.Network, int, error)"p�go.weak.type.*func(string) (*github.com/fsouza/go-dockerclient.Network, int, error)�"runtime.zerovalue���type.func(string) (*github.com/fsouza/go-dockerclient.Network, int, error)���type.func(string) (*github.com/fsouza/go-dockerclient.Network, int, error)�type.string�^type.*github.com/fsouza/go-dockerclient.Network�type.int�type.error�2go.string."generateEvent"@< generateEvent 2go.string."generateEvent"�Hgo.string."func() *docker.APIEvents"`Rfunc() *docker.APIEvents Hgo.string."func() *docker.APIEvents"�ptype.func() *github.com/fsouza/go-dockerclient.APIEvents�����3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PHgo.string."func() *docker.APIEvents"p�go.weak.type.*func() *github.com/fsouza/go-dockerclient.APIEvents�"runtime.zerovalue��ptype.func() *github.com/fsouza/go-dockerclient.APIEvents��ptype.func() *github.com/fsouza/go-dockerclient.APIEvents�btype.*github.com/fsouza/go-dockerclient.APIEvents�,go.string."generateID"@6 +generateID ,go.string."generateID"�&go.string."getExec"00getExec &go.string."getExec"�jgo.string."func(string) (*docker.ExecInspect, error)"�t)func(string) (*docker.ExecInspect, error) jgo.string."func(string) (*docker.ExecInspect, error)"��type.func(string) (*github.com/fsouza/go-dockerclient.ExecInspect, error)�� ��3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pjgo.string."func(string) (*docker.ExecInspect, error)"p�go.weak.type.*func(string) (*github.com/fsouza/go-dockerclient.ExecInspect, error)�"runtime.zerovalue���type.func(string) (*github.com/fsouza/go-dockerclient.ExecInspect, error)���type.func(string) (*github.com/fsouza/go-dockerclient.ExecInspect, error)�type.string�ftype.*github.com/fsouza/go-dockerclient.ExecInspect�type.error�(go.string."getImage"@2getImage (go.string."getImage"�4go.string."handlerWrapper"@>handlerWrapper 4go.string."handlerWrapper"��go.string."func(func(http.ResponseWriter, *http.Request)) func(http.ResponseWriter, *http.Request)"��Wfunc(func(http.ResponseWriter, *http.Request)) func(http.ResponseWriter, *http.Request) �go.string."func(func(http.ResponseWriter, *http.Request)) func(http.ResponseWriter, *http.Request)"��type.func(func(net/http.ResponseWriter, *net/http.Request)) func(net/http.ResponseWriter, *net/http.Request)���7�x3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."func(func(http.ResponseWriter, *http.Request)) func(http.ResponseWriter, *http.Request)"p�go.weak.type.*func(func(net/http.ResponseWriter, *net/http.Request)) func(net/http.ResponseWriter, *net/http.Request)�"runtime.zerovalue���type.func(func(net/http.ResponseWriter, *net/http.Request)) func(net/http.ResponseWriter, *net/http.Request)���type.func(func(net/http.ResponseWriter, *net/http.Request)) func(net/http.ResponseWriter, *net/http.Request)�jtype.func(net/http.ResponseWriter, *net/http.Request)�jtype.func(net/http.ResponseWriter, *net/http.Request)�8go.string."inspectContainer"PBinspectContainer 8go.string."inspectContainer"�@go.string."inspectExecContainer"PJinspectExecContainer @go.string."inspectExecContainer"�0go.string."inspectImage"@: inspectImage 0go.string."inspectImage"�4go.string."listContainers"@>listContainers 4go.string."listContainers"�,go.string."listEvents"@6 +listEvents ,go.string."listEvents"�,go.string."listImages"@6 +listImages ,go.string."listImages"�0go.string."listNetworks"@: listNetworks 0go.string."listNetworks"�*go.string."loadImage"@4 loadImage *go.string."loadImage"�.go.string."networkInfo"@8 networkInfo .go.string."networkInfo"�$go.string."notify"0.notify $go.string."notify"�Fgo.string."func(*docker.Container)"PPfunc(*docker.Container) Fgo.string."func(*docker.Container)"�ntype.func(*github.com/fsouza/go-dockerclient.Container)����:v3 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PFgo.string."func(*docker.Container)"p�go.weak.type.*func(*github.com/fsouza/go-dockerclient.Container)�"runtime.zerovalue��ntype.func(*github.com/fsouza/go-dockerclient.Container)��ntype.func(*github.com/fsouza/go-dockerclient.Container)�btype.*github.com/fsouza/go-dockerclient.Container�4go.string."pauseContainer"@>pauseContainer 4go.string."pauseContainer"�,go.string."pingDocker"@6 +pingDocker ,go.string."pingDocker"�*go.string."pullImage"@4 pullImage *go.string."pullImage"�*go.string."pushImage"@4 pushImage *go.string."pushImage"�6go.string."removeContainer"@@removeContainer 6go.string."removeContainer"�.go.string."removeImage"@8 removeImage .go.string."removeImage"�6go.string."renameContainer"@@renameContainer 6go.string."renameContainer"�>go.string."resizeExecContainer"PHresizeExecContainer >go.string."resizeExecContainer"�4go.string."startContainer"@>startContainer 4go.string."startContainer"�statsContainer 4go.string."statsContainer"�2go.string."stopContainer"@< stopContainer 2go.string."stopContainer"�(go.string."tagImage"@2tagImage (go.string."tagImage"�0go.string."topContainer"@: topContainer 0go.string."topContainer"�8go.string."unpauseContainer"PBunpauseContainer 8go.string."unpauseContainer"�2go.string."waitContainer"@< waitContainer 2go.string."waitContainer"�*type.*"".DockerServer�+�+��P�677� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."*testing.DockerServer"p"".(*DockerServer).PrepareStats�>"".(*DockerServer).PrepareStats�0go.string."ResetFailure"�"type.func(string)�Ftype.func(*"".DockerServer, string)�>"".(*DockerServer).ResetFailure�>"".(*DockerServer).ResetFailure�go.string."createExecContainer"�"go.importpath."".�jtype.func(net/http.ResponseWriter, *net/http.Request)��type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�L"".(*DockerServer).createExecContainer�L"".(*DockerServer).createExecContainer�2go.string."createNetwork"�"go.importpath."".�jtype.func(net/http.ResponseWriter, *net/http.Request)��type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�@"".(*DockerServer).createNetwork�@"".(*DockerServer).createNetwork�2go.string."findContainer"�"go.importpath."".��type.func(string) (*github.com/fsouza/go-dockerclient.Container, int, error)��type.func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.Container, int, error)�@"".(*DockerServer).findContainer�@"".(*DockerServer).findContainer�*go.string."findImage"�"go.importpath."".�Btype.func(string) (string, error)�ftype.func(*"".DockerServer, string) (string, error)�8"".(*DockerServer).findImage�8"".(*DockerServer).findImage�2go.string."findImageByID"�"go.importpath."".�Ltype.func(string) (string, int, error)�ptype.func(*"".DockerServer, string) (string, int, error)�@"".(*DockerServer).findImageByID�@"".(*DockerServer).findImageByID�.go.string."findNetwork"�"go.importpath."".��type.func(string) (*github.com/fsouza/go-dockerclient.Network, int, error)��type.func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.Network, int, error)�<"".(*DockerServer).findNetwork�<"".(*DockerServer).findNetwork�2go.string."generateEvent"�"go.importpath."".�ptype.func() *github.com/fsouza/go-dockerclient.APIEvents��type.func(*"".DockerServer) *github.com/fsouza/go-dockerclient.APIEvents�@"".(*DockerServer).generateEvent�@"".(*DockerServer).generateEvent�,go.string."generateID"�"go.importpath."".�$type.func() string�Dtype.func(*"".DockerServer) string�:"".(*DockerServer).generateID�:"".(*DockerServer).generateID�&go.string."getExec"�"go.importpath."".��type.func(string) (*github.com/fsouza/go-dockerclient.ExecInspect, error)��type.func(*"".DockerServer, string) (*github.com/fsouza/go-dockerclient.ExecInspect, error)�4"".(*DockerServer).getExec�4"".(*DockerServer).getExec�(go.string."getImage"�"go.importpath."".�jtype.func(net/http.ResponseWriter, *net/http.Request)��type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�6"".(*DockerServer).getImage�6"".(*DockerServer).getImage�4go.string."handlerWrapper"�"go.importpath."".��type.func(func(net/http.ResponseWriter, *net/http.Request)) func(net/http.ResponseWriter, *net/http.Request)��type.func(*"".DockerServer, func(net/http.ResponseWriter, *net/http.Request)) func(net/http.ResponseWriter, *net/http.Request)�B"".(*DockerServer).handlerWrapper�B"".(*DockerServer).handlerWrapper�8go.string."inspectContainer"�"go.importpath."".�jtype.func(net/http.ResponseWriter, *net/http.Request)��type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�F"".(*DockerServer).inspectContainer�F"".(*DockerServer).inspectContainer�@go.string."inspectExecContainer"�"go.importpath."".�jtype.func(net/http.ResponseWriter, *net/http.Request)��type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�N"".(*DockerServer).inspectExecContainer�N"".(*DockerServer).inspectExecContainer�0go.string."inspectImage"�"go.importpath."".�jtype.func(net/http.ResponseWriter, *net/http.Request)��type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�>"".(*DockerServer).inspectImage�>"".(*DockerServer).inspectImage�4go.string."listContainers"�"go.importpath."".�jtype.func(net/http.ResponseWriter, *net/http.Request)��type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�B"".(*DockerServer).listContainers�B"".(*DockerServer).listContainers�,go.string."listEvents"�"go.importpath."".�jtype.func(net/http.ResponseWriter, *net/http.Request)��type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�:"".(*DockerServer).listEvents�:"".(*DockerServer).listEvents�,go.string."listImages"�"go.importpath."".�jtype.func(net/http.ResponseWriter, *net/http.Request)��type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�:"".(*DockerServer).listImages�:"".(*DockerServer).listImages�0go.string."listNetworks"�"go.importpath."".�jtype.func(net/http.ResponseWriter, *net/http.Request)��type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�>"".(*DockerServer).listNetworks�>"".(*DockerServer).listNetworks�*go.string."loadImage"�"go.importpath."".�jtype.func(net/http.ResponseWriter, *net/http.Request)��type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�8"".(*DockerServer).loadImage�8"".(*DockerServer).loadImage�.go.string."networkInfo"�"go.importpath."".�jtype.func(net/http.ResponseWriter, *net/http.Request)��type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�<"".(*DockerServer).networkInfo�<"".(*DockerServer).networkInfo�$go.string."notify"�"go.importpath."".�ntype.func(*github.com/fsouza/go-dockerclient.Container)��type.func(*"".DockerServer, *github.com/fsouza/go-dockerclient.Container)�2"".(*DockerServer).notify�2"".(*DockerServer).notify�4go.string."pauseContainer"�"go.importpath."".�jtype.func(net/http.ResponseWriter, *net/http.Request)��type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�B"".(*DockerServer).pauseContainer�B"".(*DockerServer).pauseContainer�,go.string."pingDocker"� "go.importpath."".� jtype.func(net/http.ResponseWriter, *net/http.Request)� �type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)� :"".(*DockerServer).pingDocker� :"".(*DockerServer).pingDocker� *go.string."pullImage"� "go.importpath."".� jtype.func(net/http.ResponseWriter, *net/http.Request)�!�type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�!8"".(*DockerServer).pullImage�!8"".(*DockerServer).pullImage�!*go.string."pushImage"�!"go.importpath."".�!jtype.func(net/http.ResponseWriter, *net/http.Request)�!�type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�!8"".(*DockerServer).pushImage�"8"".(*DockerServer).pushImage�"6go.string."removeContainer"�""go.importpath."".�"jtype.func(net/http.ResponseWriter, *net/http.Request)�"�type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�"D"".(*DockerServer).removeContainer�"D"".(*DockerServer).removeContainer�".go.string."removeImage"�#"go.importpath."".�#jtype.func(net/http.ResponseWriter, *net/http.Request)�#�type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�#<"".(*DockerServer).removeImage�#<"".(*DockerServer).removeImage�#6go.string."renameContainer"�#"go.importpath."".�#jtype.func(net/http.ResponseWriter, *net/http.Request)�$�type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�$D"".(*DockerServer).renameContainer�$D"".(*DockerServer).renameContainer�$>go.string."resizeExecContainer"�$"go.importpath."".�$jtype.func(net/http.ResponseWriter, *net/http.Request)�$�type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�$L"".(*DockerServer).resizeExecContainer�%L"".(*DockerServer).resizeExecContainer�%4go.string."startContainer"�%"go.importpath."".�%jtype.func(net/http.ResponseWriter, *net/http.Request)�%�type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�%B"".(*DockerServer).startContainer�%B"".(*DockerServer).startContainer�%"".(*DockerServer).topContainer�)>"".(*DockerServer).topContainer�)8go.string."unpauseContainer"�)"go.importpath."".�)jtype.func(net/http.ResponseWriter, *net/http.Request)�*�type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�*F"".(*DockerServer).unpauseContainer�*F"".(*DockerServer).unpauseContainer�*2go.string."waitContainer"�*"go.importpath."".�*jtype.func(net/http.ResponseWriter, *net/http.Request)�*�type.func(*"".DockerServer, net/http.ResponseWriter, *net/http.Request)�*@"".(*DockerServer).waitContainer�+@"".(*DockerServer).waitContainer�,0type..gc."".DockerServer*�8type..gcprog."".DockerServer(�UUV�U�j���@go.string."testing.DockerServer"PJtesting.DockerServer @go.string."testing.DockerServer"�,go.string."containers"@6 +containers ,go.string."containers"�"go.string."execs"0,execs "go.string."execs"�&go.string."execMut"00execMut &go.string."execMut"� go.string."cMut"0*cMut go.string."cMut"�$go.string."images"0.images $go.string."images"� go.string."iMut"0*iMut go.string."iMut"�$go.string."imgIDs"0.imgIDs $go.string."imgIDs"�(go.string."networks"@2networks (go.string."networks"�$go.string."netMut"0.netMut $go.string."netMut"�(go.string."listener"@2listener (go.string."listener"�go.string."mux"0(mux go.string."mux"� go.string."hook"0*hook go.string."hook"�(go.string."failures"@2failures (go.string."failures"�2go.string."multiFailures"@< multiFailures 2go.string."multiFailures"�2go.string."execCallbacks"@< execCallbacks 2go.string."execCallbacks"�4go.string."statsCallbacks"@>statsCallbacks 4go.string."statsCallbacks"�4go.string."customHandlers"@>customHandlers 4go.string."customHandlers"�0go.string."handlerMutex"@: handlerMutex 0go.string."handlerMutex"�"go.string."cChan"0,cChan "go.string."cChan"�0go.string."DockerServer"@: DockerServer 0go.string."DockerServer"�(type."".DockerServer��@� +�Y0H`x�������� 8� � runtime.algarray00type..gc."".DockerServer@8type..gcprog."".DockerServerP@go.string."testing.DockerServer"p*type.*"".DockerServer�"runtime.zerovalue��(type."".DockerServer�,go.string."containers"�"go.importpath."".�ftype.[]*github.com/fsouza/go-dockerclient.Container�"go.string."execs"�"go.importpath."".�jtype.[]*github.com/fsouza/go-dockerclient.ExecInspect�&go.string."execMut"�"go.importpath."".�"type.sync.RWMutex� go.string."cMut"�"go.importpath."".�"type.sync.RWMutex�$go.string."images"�"go.importpath."".�\type.[]github.com/fsouza/go-dockerclient.Image� go.string."iMut"�"go.importpath."".�"type.sync.RWMutex�$go.string."imgIDs"�"go.importpath."".�,type.map[string]string�(go.string."networks"�"go.importpath."".�btype.[]*github.com/fsouza/go-dockerclient.Network�$go.string."netMut"�"go.importpath."".�"type.sync.RWMutex�(go.string."listener"�"go.importpath."".�"type.net.Listener�go.string."mux"�"go.importpath."".��type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Router� go.string."hook"�"go.importpath."".�8type.func(*net/http.Request)� (go.string."failures"� "go.importpath."".� ,type.map[string]string� 2go.string."multiFailures"� "go.importpath."".� 0type.[]map[string]string� +2go.string."execCallbacks"� +"go.importpath."".� +,type.map[string]func()� +4go.string."statsCallbacks"� "go.importpath."".� �type.map[string]func(string) github.com/fsouza/go-dockerclient.Stats� 4go.string."customHandlers"� "go.importpath."".� @type.map[string]net/http.Handler� 0go.string."handlerMutex"� "go.importpath."".� "type.sync.RWMutex� "go.string."cChan"� "go.importpath."".� ptype.chan<- *github.com/fsouza/go-dockerclient.Container`� (type."".DockerServer� 0go.string."DockerServer"� "go.importpath."".� �(type."".DockerServer�0go.string."[]*mux.Route"@: []*mux.Route 0go.string."[]*mux.Route"��type.[]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route��HN� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P0go.string."[]*mux.Route"p�go.weak.type.*[]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�"runtime.zerovalue��type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route��go.typelink.[]*mux.Route/[]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�type.[]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�2go.string."[8]*mux.Route"@< [8]*mux.Route 2go.string."[8]*mux.Route"��type.[8]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route��@O�`  runtime.algarray0bruntime.gcbits.0x88888888000000000000000000000000P2go.string."[8]*mux.Route"p�go.weak.type.*[8]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�"runtime.zerovalue��type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route��type.[]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route��go.typelink.[8]*mux.Route/[8]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�type.[8]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�Rgo.string."*map.bucket[string]*mux.Route"`\*map.bucket[string]*mux.Route Rgo.string."*map.bucket[string]*mux.Route"��type.*map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route���!Om6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PRgo.string."*map.bucket[string]*mux.Route"p�go.weak.type.**map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�"runtime.zerovalue��type.map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�Pgo.string."map.bucket[string]*mux.Route"`Zmap.bucket[string]*mux.Route Pgo.string."map.bucket[string]*mux.Route"��type.map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�������� � runtime.algarray0bruntime.gcbits.0x84848484848484848488888888000000PPgo.string."map.bucket[string]*mux.Route"p�go.weak.type.*map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�"runtime.zerovalue���type.map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route� go.string."keys"�type.[8]string�$go.string."values"��type.[8]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�(go.string."overflow"��type.*map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�Jgo.string."map.hdr[string]*mux.Route"`Tmap.hdr[string]*mux.Route Jgo.string."map.hdr[string]*mux.Route"��type.map.hdr[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route��0 �  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000PJgo.string."map.hdr[string]*mux.Route"p�go.weak.type.*map.hdr[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�"runtime.zerovalue���type.map.hdr[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�&go.string."buckets"��type.*map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�,go.string."oldbuckets"��type.*map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�Bgo.string."map[string]*mux.Route"PLmap[string]*mux.Route Bgo.string."map[string]*mux.Route"��type.map[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route���U�5� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."map[string]*mux.Route"p�go.weak.type.*map[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�"runtime.zerovalue�type.string��type.*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route��type.map.bucket[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route��type.map.hdr[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route��go.typelink.map[string]*mux.Route/map[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�type.map[string]*github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux.Route�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�0type..hashfunc.[1]string(type..hash.[1]string�,type..eqfunc.[1]string$type..eq.[1]string�&type..alg.[1]string 0type..hashfunc.[1]string,type..eqfunc.[1]string�bruntime.gcbits.0x48000000000000000000000000000000 H�*go.string."[1]string"@4 [1]string *go.string."[1]string"�type.[1]string��ĸb  &type..alg.[1]string0bruntime.gcbits.0x48000000000000000000000000000000P*go.string."[1]string"p.go.weak.type.*[1]string�"runtime.zerovalue�type.string�type.[]string�>go.typelink.[1]string/[1]stringtype.[1]string�bruntime.gcbits.0x84000000000000000000000000000000 ��rgo.string."struct { F uintptr; R *testing.DockerServer }"�|-struct { F uintptr; R *testing.DockerServer } rgo.string."struct { F uintptr; R *testing.DockerServer }"�go.string."F"0$F go.string."F"�go.string."R"0$R go.string."R"�Ztype.struct { F uintptr; R *"".DockerServer }���|(I � runtime.algarray0bruntime.gcbits.0x84000000000000000000000000000000Prgo.string."struct { F uintptr; R *testing.DockerServer }"plgo.weak.type.*struct { F uintptr; R *"".DockerServer }�"runtime.zerovalue��Ztype.struct { F uintptr; R *"".DockerServer }�go.string."F"�type.uintptr�go.string."R"�*type.*"".DockerServer�,go.string."*[1]string"@6 +*[1]string ,go.string."*[1]string"�type.*[1]string��l.!�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."*[1]string"p0go.weak.type.**[1]string�"runtime.zerovalue�type.[1]string�tgo.string."*struct { F uintptr; R *testing.DockerServer }"�~.*struct { F uintptr; R *testing.DockerServer } tgo.string."*struct { F uintptr; R *testing.DockerServer }"�\type.*struct { F uintptr; R *"".DockerServer }����B6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ptgo.string."*struct { F uintptr; R *testing.DockerServer }"pngo.weak.type.**struct { F uintptr; R *"".DockerServer }�"runtime.zerovalue�Ztype.struct { F uintptr; R *"".DockerServer }�^runtime.gcbits.0x000000000000000000000000000000 �@go.string."[0]map[string]string"PJ[0]map[string]string @go.string."[0]map[string]string"�2type.[0]map[string]string���l}�� � runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P@go.string."[0]map[string]string"pDgo.weak.type.*[0]map[string]string�"runtime.zerovalue�,type.map[string]string�0type.[]map[string]string�jgo.typelink.[0]map[string]string/[0]map[string]string2type.[0]map[string]string�Bgo.string."*[0]map[string]string"PL*[0]map[string]string Bgo.string."*[0]map[string]string"�4type.*[0]map[string]string���r<�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PBgo.string."*[0]map[string]string"pFgo.weak.type.**[0]map[string]string�"runtime.zerovalue�2type.[0]map[string]string�go.typelink.[]uintptr/[]uintptrtype.[]uintptr�,go.string."[4]uintptr"@6 +[4]uintptr ,go.string."[4]uintptr"�type.[4]uintptr�� l<�  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P,go.string."[4]uintptr"p0go.weak.type.*[4]uintptr�"runtime.zerovalue�type.uintptr�type.[]uintptr�Bgo.typelink.[4]uintptr/[4]uintptrtype.[4]uintptr�bruntime.gcbits.0x88888844440000000000000000000000 ���DD�Pgo.string."map.iter[string]http.Handler"`Zmap.iter[string]http.Handler Pgo.string."map.iter[string]http.Handler"�go.string."key"0(key go.string."key"�go.string."val"0(val go.string."val"�go.string."h"0$h go.string."h"� go.string."bptr"0*bptr go.string."bptr"�"go.string."other"0,other "go.string."other"�Jtype.map.iter[string]net/http.Handler��P�  (0( � runtime.algarray0bruntime.gcbits.0x88888844440000000000000000000000PPgo.string."map.iter[string]http.Handler"p\go.weak.type.*map.iter[string]net/http.Handler�"runtime.zerovalue��Jtype.map.iter[string]net/http.Handler�go.string."key"�type.*string�go.string."val"�,type.*net/http.Handler�go.string."t"�type.*uint8�go.string."h"�Jtype.*map.hdr[string]net/http.Handler�&go.string."buckets"�Ptype.*map.bucket[string]net/http.Handler� go.string."bptr"�Ptype.*map.bucket[string]net/http.Handler�"go.string."other"�type.[4]uintptr�Dgo.string."**testing.DockerServer"PN**testing.DockerServer Dgo.string."**testing.DockerServer"�,type.**"".DockerServer��Ep�6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PDgo.string."**testing.DockerServer"p>go.weak.type.***"".DockerServer�"runtime.zerovalue�*type.*"".DockerServer�jgo.string."*func(http.ResponseWriter, *http.Request)"�t)*func(http.ResponseWriter, *http.Request) jgo.string."*func(http.ResponseWriter, *http.Request)"�ltype.*func(net/http.ResponseWriter, *net/http.Request)���'~P6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Pjgo.string."*func(http.ResponseWriter, *http.Request)"p~go.weak.type.**func(net/http.ResponseWriter, *net/http.Request)�"runtime.zerovalue�jtype.func(net/http.ResponseWriter, *net/http.Request)�bruntime.gcbits.0x84488800000000000000000000000000 �H���go.string."struct { F uintptr; A0 **testing.DockerServer; A1 *func(http.ResponseWriter, *http.Request) }"��]struct { F uintptr; A0 **testing.DockerServer; A1 *func(http.ResponseWriter, *http.Request) } �go.string."struct { F uintptr; A0 **testing.DockerServer; A1 *func(http.ResponseWriter, *http.Request) }"�go.string."A0"0&A0 go.string."A0"�go.string."A1"0&A1 go.string."A1"��type.struct { F uintptr; A0 **"".DockerServer; A1 *func(net/http.ResponseWriter, *net/http.Request) }��v$�(  runtime.algarray0bruntime.gcbits.0x84488800000000000000000000000000P�go.string."struct { F uintptr; A0 **testing.DockerServer; A1 *func(http.ResponseWriter, *http.Request) }"p�go.weak.type.*struct { F uintptr; A0 **"".DockerServer; A1 *func(net/http.ResponseWriter, *net/http.Request) }�"runtime.zerovalue���type.struct { F uintptr; A0 **"".DockerServer; A1 *func(net/http.ResponseWriter, *net/http.Request) }�go.string."F"�type.uintptr�go.string."A0"�,type.**"".DockerServer�go.string."A1"�ltype.*func(net/http.ResponseWriter, *net/http.Request)��go.string."*struct { F uintptr; A0 **testing.DockerServer; A1 *func(http.ResponseWriter, *http.Request) }"��^*struct { F uintptr; A0 **testing.DockerServer; A1 *func(http.ResponseWriter, *http.Request) } �go.string."*struct { F uintptr; A0 **testing.DockerServer; A1 *func(http.ResponseWriter, *http.Request) }"��type.*struct { F uintptr; A0 **"".DockerServer; A1 *func(net/http.ResponseWriter, *net/http.Request) }���r��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P�go.string."*struct { F uintptr; A0 **testing.DockerServer; A1 *func(http.ResponseWriter, *http.Request) }"p�go.weak.type.**struct { F uintptr; A0 **"".DockerServer; A1 *func(net/http.ResponseWriter, *net/http.Request) }�"runtime.zerovalue��type.struct { F uintptr; A0 **"".DockerServer; A1 *func(net/http.ResponseWriter, *net/http.Request) }�Dgo.string."[]docker.APIContainers"PN[]docker.APIContainers Dgo.string."[]docker.APIContainers"�ltype.[]github.com/fsouza/go-dockerclient.APIContainers��� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000PDgo.string."[]docker.APIContainers"p~go.weak.type.*[]github.com/fsouza/go-dockerclient.APIContainers�"runtime.zerovalue�htype.github.com/fsouza/go-dockerclient.APIContainers��go.typelink.[]docker.APIContainers/[]github.com/fsouza/go-dockerclient.APIContainersltype.[]github.com/fsouza/go-dockerclient.APIContainers�bruntime.gcbits.0xcc000000000000000000000000000000 ��0go.string."interface {}"@: interface {} 0go.string."interface {}"�"type.interface {}���W� � runtime.algarray0bruntime.gcbits.0xcc000000000000000000000000000000P0go.string."interface {}"p4go.weak.type.*interface {}�"runtime.zerovalue��"type.interface {}�4go.string."[]interface {}"@>[]interface {} 4go.string."[]interface {}"�&type.[]interface {}��p��/ � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P4go.string."[]interface {}"p8go.weak.type.*[]interface {}�"runtime.zerovalue�"type.interface {}�Rgo.typelink.[]interface {}/[]interface {}&type.[]interface {}�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·44568aa369055d8938d809aa5d80843b�Tgclocals·9c703c5c7b9c1932c840b69f8ebce236�[8]docker.Port 4go.string."[8]docker.Port"�\type.[8]github.com/fsouza/go-dockerclient.Port���~/(� ftype..alg.[8]github.com/fsouza/go-dockerclient.Port0bruntime.gcbits.0x48484848484848480000000000000000P4go.string."[8]docker.Port"pngo.weak.type.*[8]github.com/fsouza/go-dockerclient.Port�"runtime.zerovalue�Vtype.github.com/fsouza/go-dockerclient.Port�Ztype.[]github.com/fsouza/go-dockerclient.Port��go.typelink.[8]docker.Port/[8]github.com/fsouza/go-dockerclient.Port\type.[8]github.com/fsouza/go-dockerclient.Port�Dgo.string."[][]docker.PortBinding"PN[][]docker.PortBinding Dgo.string."[][]docker.PortBinding"�ltype.[][]github.com/fsouza/go-dockerclient.PortBinding��q�Ǣ � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000PDgo.string."[][]docker.PortBinding"p~go.weak.type.*[][]github.com/fsouza/go-dockerclient.PortBinding�"runtime.zerovalue�htype.[]github.com/fsouza/go-dockerclient.PortBinding��go.typelink.[][]docker.PortBinding/[][]github.com/fsouza/go-dockerclient.PortBindingltype.[][]github.com/fsouza/go-dockerclient.PortBinding�bruntime.gcbits.0x48844448844448844448844400000000 H�DH�DH�DH�D�Fgo.string."[8][]docker.PortBinding"PP[8][]docker.PortBinding Fgo.string."[8][]docker.PortBinding"�ntype.[8][]github.com/fsouza/go-dockerclient.PortBinding����(�h � runtime.algarray0bruntime.gcbits.0x48844448844448844448844400000000PFgo.string."[8][]docker.PortBinding"p�go.weak.type.*[8][]github.com/fsouza/go-dockerclient.PortBinding�"runtime.zerovalue�htype.[]github.com/fsouza/go-dockerclient.PortBinding�ltype.[][]github.com/fsouza/go-dockerclient.PortBinding��go.typelink.[8][]docker.PortBinding/[8][]github.com/fsouza/go-dockerclient.PortBindingntype.[8][]github.com/fsouza/go-dockerclient.PortBinding�pgo.string."*map.bucket[docker.Port][]docker.PortBinding"�z,*map.bucket[docker.Port][]docker.PortBinding pgo.string."*map.bucket[docker.Port][]docker.PortBinding"��type.*map.bucket[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding��䴝~6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000Ppgo.string."*map.bucket[docker.Port][]docker.PortBinding"p�go.weak.type.**map.bucket[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding�"runtime.zerovalue��type.map.bucket[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding�,�type..gc.map.bucket[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding,��type..gcprog.map.bucket[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding*����Y�eY�e �ngo.string."map.bucket[docker.Port][]docker.PortBinding"�x+map.bucket[docker.Port][]docker.PortBinding ngo.string."map.bucket[docker.Port][]docker.PortBinding"��type.map.bucket[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding��P��r�Y�H � runtime.algarray0�type..gc.map.bucket[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding@�type..gcprog.map.bucket[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBindingPngo.string."map.bucket[docker.Port][]docker.PortBinding"p�go.weak.type.*map.bucket[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding�"runtime.zerovalue���type.map.bucket[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding� go.string."keys"�\type.[8]github.com/fsouza/go-dockerclient.Port�$go.string."values"�ntype.[8][]github.com/fsouza/go-dockerclient.PortBinding�(go.string."overflow"��type.*map.bucket[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding�hgo.string."map.hdr[docker.Port][]docker.PortBinding"�r(map.hdr[docker.Port][]docker.PortBinding hgo.string."map.hdr[docker.Port][]docker.PortBinding"��type.map.hdr[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding��09�h�  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000Phgo.string."map.hdr[docker.Port][]docker.PortBinding"p�go.weak.type.*map.hdr[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding�"runtime.zerovalue���type.map.hdr[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding�&go.string."buckets"��type.*map.bucket[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding�,go.string."oldbuckets"��type.*map.bucket[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding�`go.string."map[docker.Port][]docker.PortBinding"pj$map[docker.Port][]docker.PortBinding `go.string."map[docker.Port][]docker.PortBinding"��type.map[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding��� 5P � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P`go.string."map[docker.Port][]docker.PortBinding"p�go.weak.type.*map[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding�"runtime.zerovalue�Vtype.github.com/fsouza/go-dockerclient.Port�htype.[]github.com/fsouza/go-dockerclient.PortBinding��type.map.bucket[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding��type.map.hdr[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding��go.typelink.map[docker.Port][]docker.PortBinding/map[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding�type.map[github.com/fsouza/go-dockerclient.Port][]github.com/fsouza/go-dockerclient.PortBinding�*go.string."struct {}"@4 struct {} *go.string."struct {}"�type.struct {}����'�  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P*go.string."struct {}"p.go.weak.type.*struct {}�"runtime.zerovalue��type.struct {}�.go.string."[]struct {}"@8 []struct {} .go.string."[]struct {}"� type.[]struct {}���̥� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P.go.string."[]struct {}"p2go.weak.type.*[]struct {}�"runtime.zerovalue�type.struct {}�Fgo.typelink.[]struct {}/[]struct {} type.[]struct {}�0go.string."[8]struct {}"@: [8]struct {} 0go.string."[8]struct {}"�"type.[8]struct {}��>�y �  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P0go.string."[8]struct {}"p4go.weak.type.*[8]struct {}�"runtime.zerovalue�type.struct {}� type.[]struct {}�Jgo.typelink.[8]struct {}/[8]struct {}"type.[8]struct {}�Zgo.string."*map.bucket[docker.Port]struct {}"pd!*map.bucket[docker.Port]struct {} Zgo.string."*map.bucket[docker.Port]struct {}"��type.*map.bucket[github.com/fsouza/go-dockerclient.Port]struct {}��1S6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PZgo.string."*map.bucket[docker.Port]struct {}"p�go.weak.type.**map.bucket[github.com/fsouza/go-dockerclient.Port]struct {}�"runtime.zerovalue��type.map.bucket[github.com/fsouza/go-dockerclient.Port]struct {}�bruntime.gcbits.0x84848484848484848400000000000000 ����������Xgo.string."map.bucket[docker.Port]struct {}"pb map.bucket[docker.Port]struct {} Xgo.string."map.bucket[docker.Port]struct {}"��type.map.bucket[github.com/fsouza/go-dockerclient.Port]struct {}����(��� � runtime.algarray0bruntime.gcbits.0x84848484848484848400000000000000PXgo.string."map.bucket[docker.Port]struct {}"p�go.weak.type.*map.bucket[github.com/fsouza/go-dockerclient.Port]struct {}�"runtime.zerovalue���type.map.bucket[github.com/fsouza/go-dockerclient.Port]struct {}� go.string."keys"�\type.[8]github.com/fsouza/go-dockerclient.Port�$go.string."values"�"type.[8]struct {}�(go.string."overflow"��type.*map.bucket[github.com/fsouza/go-dockerclient.Port]struct {}�Rgo.string."map.hdr[docker.Port]struct {}"`\map.hdr[docker.Port]struct {} Rgo.string."map.hdr[docker.Port]struct {}"�ztype.map.hdr[github.com/fsouza/go-dockerclient.Port]struct {}��0���  � runtime.algarray0bruntime.gcbits.0x44844800000000000000000000000000PRgo.string."map.hdr[docker.Port]struct {}"p�go.weak.type.*map.hdr[github.com/fsouza/go-dockerclient.Port]struct {}�"runtime.zerovalue��ztype.map.hdr[github.com/fsouza/go-dockerclient.Port]struct {}�&go.string."buckets"��type.*map.bucket[github.com/fsouza/go-dockerclient.Port]struct {}�,go.string."oldbuckets"��type.*map.bucket[github.com/fsouza/go-dockerclient.Port]struct {}�Jgo.string."map[docker.Port]struct {}"`Tmap[docker.Port]struct {} Jgo.string."map[docker.Port]struct {}"�rtype.map[github.com/fsouza/go-dockerclient.Port]struct {}���'�75� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PJgo.string."map[docker.Port]struct {}"p�go.weak.type.*map[github.com/fsouza/go-dockerclient.Port]struct {}�"runtime.zerovalue�Vtype.github.com/fsouza/go-dockerclient.Port�type.struct {}��type.map.bucket[github.com/fsouza/go-dockerclient.Port]struct {}�ztype.map.hdr[github.com/fsouza/go-dockerclient.Port]struct {}��go.typelink.map[docker.Port]struct {}/map[github.com/fsouza/go-dockerclient.Port]struct {}rtype.map[github.com/fsouza/go-dockerclient.Port]struct {}�Tgclocals·3280bececceccd33cb74587feedb1f9f�Tgclocals·2dc77d960dd3e4b3de2361f9cbd75783�Tgclocals·65526a5f07004f02424fe51b799cdd23  +�Tgclocals·fa7203fd5ed88aea99b7be572f707eb0 �~type..hashfunc.[1]github.com/fsouza/go-dockerclient.PortBindingvtype..hash.[1]github.com/fsouza/go-dockerclient.PortBinding�ztype..eqfunc.[1]github.com/fsouza/go-dockerclient.PortBindingrtype..eq.[1]github.com/fsouza/go-dockerclient.PortBinding�ttype..alg.[1]github.com/fsouza/go-dockerclient.PortBinding ~type..hashfunc.[1]github.com/fsouza/go-dockerclient.PortBindingztype..eqfunc.[1]github.com/fsouza/go-dockerclient.PortBinding�bruntime.gcbits.0x48480000000000000000000000000000 HH�Bgo.string."[1]docker.PortBinding"PL[1]docker.PortBinding Bgo.string."[1]docker.PortBinding"�jtype.[1]github.com/fsouza/go-dockerclient.PortBinding�� �� ttype..alg.[1]github.com/fsouza/go-dockerclient.PortBinding0bruntime.gcbits.0x48480000000000000000000000000000PBgo.string."[1]docker.PortBinding"p|go.weak.type.*[1]github.com/fsouza/go-dockerclient.PortBinding�"runtime.zerovalue�dtype.github.com/fsouza/go-dockerclient.PortBinding�htype.[]github.com/fsouza/go-dockerclient.PortBinding��go.typelink.[1]docker.PortBinding/[1]github.com/fsouza/go-dockerclient.PortBindingjtype.[1]github.com/fsouza/go-dockerclient.PortBinding�@go.string."struct { ID string }"PJstruct { ID string } @go.string."struct { ID string }"�go.string."ID"0&ID go.string."ID"�2type.struct { ID string }��K�� � runtime.algarray0bruntime.gcbits.0x48000000000000000000000000000000P@go.string."struct { ID string }"pDgo.weak.type.*struct { ID string }�"runtime.zerovalue��2type.struct { ID string }�go.string."ID"�type.string�,go.string."*struct {}"@6 +*struct {} ,go.string."*struct {}"�type.*struct {}��J$��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."*struct {}"p0go.weak.type.**struct {}�"runtime.zerovalue�type.struct {}�Tgo.string."*map.hdr[docker.Port]struct {}"`^*map.hdr[docker.Port]struct {} Tgo.string."*map.hdr[docker.Port]struct {}"�|type.*map.hdr[github.com/fsouza/go-dockerclient.Port]struct {}��5�BN6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PTgo.string."*map.hdr[docker.Port]struct {}"p�go.weak.type.**map.hdr[github.com/fsouza/go-dockerclient.Port]struct {}�"runtime.zerovalue�ztype.map.hdr[github.com/fsouza/go-dockerclient.Port]struct {}�Tgo.string."map.iter[docker.Port]struct {}"`^map.iter[docker.Port]struct {} Tgo.string."map.iter[docker.Port]struct {}"�|type.map.iter[github.com/fsouza/go-dockerclient.Port]struct {}��P�tυ (0( � runtime.algarray0bruntime.gcbits.0x88888844440000000000000000000000PTgo.string."map.iter[docker.Port]struct {}"p�go.weak.type.*map.iter[github.com/fsouza/go-dockerclient.Port]struct {}�"runtime.zerovalue��|type.map.iter[github.com/fsouza/go-dockerclient.Port]struct {}�go.string."key"�Xtype.*github.com/fsouza/go-dockerclient.Port�go.string."val"�type.*struct {}�go.string."t"�type.*uint8�go.string."h"�|type.*map.hdr[github.com/fsouza/go-dockerclient.Port]struct {}�&go.string."buckets"��type.*map.bucket[github.com/fsouza/go-dockerclient.Port]struct {}� go.string."bptr"��type.*map.bucket[github.com/fsouza/go-dockerclient.Port]struct {}�"go.string."other"�type.[4]uintptr�Dgo.string."*[1]docker.PortBinding"PN*[1]docker.PortBinding Dgo.string."*[1]docker.PortBinding"�ltype.*[1]github.com/fsouza/go-dockerclient.PortBinding�����:6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PDgo.string."*[1]docker.PortBinding"p~go.weak.type.**[1]github.com/fsouza/go-dockerclient.PortBinding�"runtime.zerovalue�jtype.[1]github.com/fsouza/go-dockerclient.PortBinding�&go.string."[]uint8"00[]uint8 &go.string."[]uint8"�type.[]uint8���~.8 � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P&go.string."[]uint8"p*go.weak.type.*[]uint8�"runtime.zerovalue�type.uint8�6go.typelink.[]uint8/[]uint8type.[]uint8�*go.string."[16]uint8"@4 [16]uint8 *go.string."[16]uint8"�type.[16]uint8���}5G� � runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P*go.string."[16]uint8"p.go.weak.type.*[16]uint8�"runtime.zerovalue�type.uint8�type.[]uint8�>go.typelink.[16]uint8/[16]uint8type.[16]uint8�,go.string."*[16]uint8"@6 +*[16]uint8 ,go.string."*[16]uint8"�type.*[16]uint8���/ Q6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."*[16]uint8"p0go.weak.type.**[16]uint8�"runtime.zerovalue�type.[16]uint8�,go.string."[][]string"@6 +[][]string ,go.string."[][]string"�type.[][]string���:� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P,go.string."[][]string"p0go.weak.type.*[][]string�"runtime.zerovalue�type.[]string�Bgo.typelink.[][]string/[][]stringtype.[][]string�.go.string."[1][]string"@8 [1][]string .go.string."[1][]string"� type.[1][]string���y=Y � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P.go.string."[1][]string"p2go.weak.type.*[1][]string�"runtime.zerovalue�type.[]string�type.[][]string�Fgo.typelink.[1][]string/[1][]string type.[1][]string�,go.string."*[8]string"@6 +*[8]string ,go.string."*[8]string"�type.*[8]string����o6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P,go.string."*[8]string"p0go.weak.type.**[8]string�"runtime.zerovalue�type.[8]string�0go.string."*[1][]string"@: *[1][]string 0go.string."*[1][]string"�"type.*[1][]string���^��6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P0go.string."*[1][]string"p4go.weak.type.**[1][]string�"runtime.zerovalue� type.[1][]string�"go.string."[]int"0,[]int "go.string."[]int"�type.[]int���f� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P"go.string."[]int"p&go.weak.type.*[]int�"runtime.zerovalue�type.int�.go.typelink.[]int/[]inttype.[]int�$go.string."[8]int"0.[8]int $go.string."[8]int"�type.[8]int��@����  runtime.algarray0^runtime.gcbits.0x000000000000000000000000000000P$go.string."[8]int"p(go.weak.type.*[8]int�"runtime.zerovalue�type.int�type.[]int�2go.typelink.[8]int/[8]inttype.[8]int�Dgo.string."*map.bucket[string]int"PN*map.bucket[string]int Dgo.string."*map.bucket[string]int"�6type.*map.bucket[string]int��ɾ̜6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PDgo.string."*map.bucket[string]int"pHgo.weak.type.**map.bucket[string]int�"runtime.zerovalue�4type.map.bucket[string]int�bruntime.gcbits.0x84848484848484844444444484000000 ��������DDDD��Bgo.string."map.bucket[string]int"PLmap.bucket[string]int Bgo.string."map.bucket[string]int"�4type.map.bucket[string]int���]hcq�� � runtime.algarray0bruntime.gcbits.0x84848484848484844444444484000000PBgo.string."map.bucket[string]int"pFgo.weak.type.*map.bucket[string]int�"runtime.zerovalue��4type.map.bucket[string]int� go.string."keys"�type.[8]string�$go.string."values"�type.[8]int�(go.string."overflow"�6type.*map.bucket[string]int�map[string]int 4go.string."map[string]int"�&type.map[string]int�����J5� � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P4go.string."map[string]int"p8go.weak.type.*map[string]int�"runtime.zerovalue�type.string�type.int�4type.map.bucket[string]int�.type.map.hdr[string]int�Rgo.typelink.map[string]int/map[string]int&type.map[string]int�*go.string."[][]uint8"@4 [][]uint8 *go.string."[][]uint8"�type.[][]uint8����}� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P*go.string."[][]uint8"p.go.weak.type.*[][]uint8�"runtime.zerovalue�type.[]uint8�>go.typelink.[][]uint8/[][]uint8type.[][]uint8�(go.string."*[]uint8"@2*[]uint8 (go.string."*[]uint8"�type.*[]uint8�����i6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P(go.string."*[]uint8"p,go.weak.type.**[]uint8�"runtime.zerovalue�type.[]uint8�&go.string."*func()"00*func() &go.string."*func()"�type.*func()����u6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P&go.string."*func()"p*go.weak.type.**func()�"runtime.zerovalue�type.func()�@go.string."**docker.ExecInspect"PJ**docker.ExecInspect @go.string."**docker.ExecInspect"�htype.**github.com/fsouza/go-dockerclient.ExecInspect��V���6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P@go.string."**docker.ExecInspect"pzgo.weak.type.***github.com/fsouza/go-dockerclient.ExecInspect�"runtime.zerovalue�ftype.*github.com/fsouza/go-dockerclient.ExecInspect�8go.string."**docker.Network"PB**docker.Network 8go.string."**docker.Network"�`type.**github.com/fsouza/go-dockerclient.Network���bS6 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000P8go.string."**docker.Network"prgo.weak.type.***github.com/fsouza/go-dockerclient.Network�"runtime.zerovalue�^type.*github.com/fsouza/go-dockerclient.Network�8go.string."[]docker.Network"PB[]docker.Network 8go.string."[]docker.Network"�`type.[]github.com/fsouza/go-dockerclient.Network���&H� � runtime.algarray0bruntime.gcbits.0x48844400000000000000000000000000P8go.string."[]docker.Network"prgo.weak.type.*[]github.com/fsouza/go-dockerclient.Network�"runtime.zerovalue�\type.github.com/fsouza/go-dockerclient.Network��go.typelink.[]docker.Network/[]github.com/fsouza/go-dockerclient.Network`type.[]github.com/fsouza/go-dockerclient.Network�Rgo.string."**docker.CreateNetworkOptions"`\**docker.CreateNetworkOptions Rgo.string."**docker.CreateNetworkOptions"�ztype.**github.com/fsouza/go-dockerclient.CreateNetworkOptions����36 � runtime.algarray0bruntime.gcbits.0x88000000000000000000000000000000PRgo.string."**docker.CreateNetworkOptions"p�go.weak.type.***github.com/fsouza/go-dockerclient.CreateNetworkOptions�"runtime.zerovalue�xtype.*github.com/fsouza/go-dockerclient.CreateNetworkOptions�type..eq.[8]net/http.Handler·f8type..eq.[8]net/http.Handler�$runtime.ifaceeq·fruntime.ifaceeq�.type..hash.[1]string·f(type..hash.[1]string�*type..eq.[1]string·f$type..eq.[1]string�:type..hash.[2]interface {}·f4type..hash.[2]interface {}�.runtime.nilinterhash·f(runtime.nilinterhash�6type..eq.[2]interface {}·f0type..eq.[2]interface {}�$runtime.efaceeq·fruntime.efaceeq�:type..hash.[1]interface {}·f4type..hash.[1]interface {}�6type..eq.[1]interface {}·f0type..eq.[1]interface {}�ntype..hash.[8]github.com/fsouza/go-dockerclient.Port·fhtype..hash.[8]github.com/fsouza/go-dockerclient.Port�jtype..eq.[8]github.com/fsouza/go-dockerclient.Port·fdtype..eq.[8]github.com/fsouza/go-dockerclient.Port�|type..hash.[1]github.com/fsouza/go-dockerclient.PortBinding·fvtype..hash.[1]github.com/fsouza/go-dockerclient.PortBinding�vtype..hash.github.com/fsouza/go-dockerclient.PortBinding·fptype..hash.github.com/fsouza/go-dockerclient.PortBinding�xtype..eq.[1]github.com/fsouza/go-dockerclient.PortBinding·frtype..eq.[1]github.com/fsouza/go-dockerclient.PortBinding�"runtime.zerovalue0��go13ld \ No newline at end of file diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/.gitignore b/Godeps/_workspace/src/github.com/Shopify/sarama/.gitignore new file mode 100644 index 0000000000000..3591f9ff30533 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/.gitignore @@ -0,0 +1,24 @@ +# Compiled Object files, Static and Dynamic libs (Shared Objects) +*.o +*.a +*.so +*.test + +# Folders +_obj +_test +.vagrant + +# Architecture specific extensions/prefixes +*.[568vq] +[568vq].out + +*.cgo1.go +*.cgo2.c +_cgo_defun.c +_cgo_gotypes.go +_cgo_export.* + +_testmain.go + +*.exe diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/.travis.yml b/Godeps/_workspace/src/github.com/Shopify/sarama/.travis.yml new file mode 100644 index 0000000000000..a9e5cc3cbf056 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/.travis.yml @@ -0,0 +1,41 @@ +language: go +go: +- 1.3.3 +- 1.4.2 + +env: + global: + - KAFKA_PEERS=localhost:9091,localhost:9092,localhost:9093,localhost:9094,localhost:9095 + - TOXIPROXY_ADDR=http://localhost:8474 + - KAFKA_INSTALL_ROOT=/home/travis/kafka + - KAFKA_HOSTNAME=localhost + - DEBUG=true + matrix: + - KAFKA_VERSION=0.8.1.1 + - KAFKA_VERSION=0.8.2.1 + +before_install: +- export REPOSITORY_ROOT=${TRAVIS_BUILD_DIR} +- vagrant/install_cluster.sh +- vagrant/boot_cluster.sh +- vagrant/create_topics.sh + + +install: +- make install_dependencies + +script: +- make test +- make vet +- make errcheck +- make fmt + +matrix: + include: + - go: tip + env: KAFKA_VERSION=0.8.2.1 + allow_failures: + - go: tip + fast_finish: true + +sudo: false diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/CHANGELOG.md b/Godeps/_workspace/src/github.com/Shopify/sarama/CHANGELOG.md new file mode 100644 index 0000000000000..5bea6bc3b615f --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/CHANGELOG.md @@ -0,0 +1,157 @@ +# Changelog + +#### Version 1.5.0 (unreleased) + +New Features: + - TLS-encrypted network connections are now supported. This feature is subject + to change when Kafka releases built-in TLS support, but for now this is + enough to work with TLS-terminating proxies + ([#154](https://github.com/Shopify/sarama/pull/154)). + +Improvements: + - The consumer will not block if a single partition is not drained by the user; + all other partitions will continue to consume normally + ([#485](https://github.com/Shopify/sarama/pull/485)). + - Formatting of error strings has been much improved + ([#495](https://github.com/Shopify/sarama/pull/495)). + - Internal refactoring of the producer for code cleanliness and to enable + future work ([#300](https://github.com/Shopify/sarama/pull/300)). + +Bug Fixes: + - Fix a potential deadlock in the consumer on shutdown + ([#475](https://github.com/Shopify/sarama/pull/475)). + +#### Version 1.4.3 (2015-07-21) + +Bug Fixes: + - Don't include the partitioner in the producer's "fetch partitions" + circuit-breaker ([#466](https://github.com/Shopify/sarama/pull/466)). + - Don't retry messages until the broker is closed when abandoning a broker in + the producer ([#468](https://github.com/Shopify/sarama/pull/468)). + - Update the import path for snappy-go, it has moved again and the API has + changed slightly ([#486](https://github.com/Shopify/sarama/pull/486)). + +#### Version 1.4.2 (2015-05-27) + +Bug Fixes: + - Update the import path for snappy-go, it has moved from google code to github + ([#456](https://github.com/Shopify/sarama/pull/456)). + +#### Version 1.4.1 (2015-05-25) + +Improvements: + - Optimizations when decoding snappy messages, thanks to John Potocny + ([#446](https://github.com/Shopify/sarama/pull/446)). + +Bug Fixes: + - Fix hypothetical race conditions on producer shutdown + ([#450](https://github.com/Shopify/sarama/pull/450), + [#451](https://github.com/Shopify/sarama/pull/451)). + +#### Version 1.4.0 (2015-05-01) + +New Features: + - The consumer now implements `Topics()` and `Partitions()` methods to enable + users to dynamically choose what topics/partitions to consume without + instantiating a full client + ([#431](https://github.com/Shopify/sarama/pull/431)). + - The partition-consumer now exposes the high water mark offset value returned + by the broker via the `HighWaterMarkOffset()` method ([#339](https://github.com/Shopify/sarama/pull/339)). + - Added a `kafka-console-consumer` tool capable of handling multiple + partitions, and deprecated the now-obsolete `kafka-console-partitionConsumer` + ([#439](https://github.com/Shopify/sarama/pull/439), + [#442](https://github.com/Shopify/sarama/pull/442)). + +Improvements: + - The producer's logging during retry scenarios is more consistent, more + useful, and slightly less verbose + ([#429](https://github.com/Shopify/sarama/pull/429)). + - The client now shuffles its initial list of seed brokers in order to prevent + thundering herd on the first broker in the list + ([#441](https://github.com/Shopify/sarama/pull/441)). + +Bug Fixes: + - The producer now correctly manages its state if retries occur when it is + shutting down, fixing several instances of confusing behaviour and at least + one potential deadlock ([#419](https://github.com/Shopify/sarama/pull/419)). + - The consumer now handles messages for different partitions asynchronously, + making it much more resilient to specific user code ordering + ([#325](https://github.com/Shopify/sarama/pull/325)). + +#### Version 1.3.0 (2015-04-16) + +New Features: + - The client now tracks consumer group coordinators using + ConsumerMetadataRequests similar to how it tracks partition leadership using + regular MetadataRequests ([#411](https://github.com/Shopify/sarama/pull/411)). + This adds two methods to the client API: + - `Coordinator(consumerGroup string) (*Broker, error)` + - `RefreshCoordinator(consumerGroup string) error` + +Improvements: + - ConsumerMetadataResponses now automatically create a Broker object out of the + ID/address/port combination for the Coordinator; accessing the fields + individually has been deprecated + ([#413](https://github.com/Shopify/sarama/pull/413)). + - Much improved handling of `OffsetOutOfRange` errors in the consumer. + Consumers will fail to start if the provided offset is out of range + ([#418](https://github.com/Shopify/sarama/pull/418)) + and they will automatically shut down if the offset falls out of range + ([#424](https://github.com/Shopify/sarama/pull/424)). + - Small performance improvement in encoding and decoding protocol messages + ([#427](https://github.com/Shopify/sarama/pull/427)). + +Bug Fixes: + - Fix a rare race condition in the client's background metadata refresher if + it happens to be activated while the client is being closed + ([#422](https://github.com/Shopify/sarama/pull/422)). + +#### Version 1.2.0 (2015-04-07) + +Improvements: + - The producer's behaviour when `Flush.Frequency` is set is now more intuitive + ([#389](https://github.com/Shopify/sarama/pull/389)). + - The producer is now somewhat more memory-efficient during and after retrying + messages due to an improved queue implementation + ([#396](https://github.com/Shopify/sarama/pull/396)). + - The consumer produces much more useful logging output when leadership + changes ([#385](https://github.com/Shopify/sarama/pull/385)). + - The client's `GetOffset` method will now automatically refresh metadata and + retry once in the event of stale information or similar + ([#394](https://github.com/Shopify/sarama/pull/394)). + - Broker connections now have support for using TCP keepalives + ([#407](https://github.com/Shopify/sarama/issues/407)). + +Bug Fixes: + - The OffsetCommitRequest message now correctly implements all three possible + API versions ([#390](https://github.com/Shopify/sarama/pull/390), + [#400](https://github.com/Shopify/sarama/pull/400)). + +#### Version 1.1.0 (2015-03-20) + +Improvements: + - Wrap the producer's partitioner call in a circuit-breaker so that repeatedly + broken topics don't choke throughput + ([#373](https://github.com/Shopify/sarama/pull/373)). + +Bug Fixes: + - Fix the producer's internal reference counting in certain unusual scenarios + ([#367](https://github.com/Shopify/sarama/pull/367)). + - Fix the consumer's internal reference counting in certain unusual scenarios + ([#369](https://github.com/Shopify/sarama/pull/369)). + - Fix a condition where the producer's internal control messages could have + gotten stuck ([#368](https://github.com/Shopify/sarama/pull/368)). + - Fix an issue where invalid partition lists would be cached when asking for + metadata for a non-existant topic ([#372](https://github.com/Shopify/sarama/pull/372)). + + +#### Version 1.0.0 (2015-03-17) + +Version 1.0.0 is the first tagged version, and is almost a complete rewrite. The primary differences with previous untagged versions are: + +- The producer has been rewritten; there is now a `SyncProducer` with a blocking API, and an `AsyncProducer` that is non-blocking. +- The consumer has been rewritten to only open one connection per broker instead of one connection per partition. +- The main types of Sarama are now interfaces to make depedency injection easy; mock implementations for `Consumer`, `SyncProducer` and `AsyncProducer` are provided in the `github.com/Shopify/sarama/mocks` package. +- For most uses cases, it is no longer necessary to open a `Client`; this will be done for you. +- All the configuration values have been unified in the `Config` struct. +- Much improved test suite. diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/CONTRIBUTING.md b/Godeps/_workspace/src/github.com/Shopify/sarama/CONTRIBUTING.md new file mode 100644 index 0000000000000..b0f107cbc76b6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/CONTRIBUTING.md @@ -0,0 +1,31 @@ +# Contributing + +Contributions are always welcome, both reporting issues and submitting pull requests! + +### Reporting issues + +Please make sure to include any potentially useful information in the issue, so we can pinpoint the issue faster without going back and forth. + +- What SHA of Sarama are you running? If this is not the latest SHA on the master branch, please try if the problem persists with the latest version. +- You can set `sarama.Logger` to a [log.Logger](http://golang.org/pkg/log/#Logger) instance to capture debug output. Please include it in your issue description. +- Also look at the logs of the Kafka broker you are connected to. If you see anything out of the ordinary, please include it. + +Also, please include the following information about your environment, so we can help you faster: + +- What version of Kafka are you using? +- What version of Go are you using? +- What are the values of your Producer/Consumer/Client configuration? + + +### Submitting pull requests + +We will gladly accept bug fixes, or additions to this library. Please fork this library, commit & push your changes, and open a pull request. Because this library is in production use by many people and applications, we code review all additions. To make the review process go as smooth as possible, please consider the following. + +- If you plan to work on something major, please open an issue to discuss the design first. +- Don't break backwards compatibility. If you really have to, open an issue to discuss this first. +- Make sure to use the `go fmt` command to format your code according to the standards. Even better, set up your editor to do this for you when saving. +- Run [go vet](https://godoc.org/golang.org/x/tools/cmd/vet) to detect any suspicious constructs in your code that could be bugs. +- Explicitly handle all error return values. If you really want to ignore an error value, you can assign it to `_`.You can use [errcheck](https://github.com/kisielk/errcheck) to verify whether you have handled all errors. +- You may also want to run [golint](https://github.com/golang/lint) as well to detect style problems. +- Add tests that cover the changes you made. Make sure to run `go test` with the `-race` argument to test for race conditions. +- Make sure your code is supported by all the Go versions we support. You can rely on [Travis CI](https://travis-ci.org/Shopify/sarama) for testing older Go versions diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/MIT-LICENSE b/Godeps/_workspace/src/github.com/Shopify/sarama/MIT-LICENSE new file mode 100644 index 0000000000000..8121b63b1c4ac --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/MIT-LICENSE @@ -0,0 +1,20 @@ +Copyright (c) 2013 Evan Huus + +Permission is hereby granted, free of charge, to any person obtaining +a copy of this software and associated documentation files (the +"Software"), to deal in the Software without restriction, including +without limitation the rights to use, copy, modify, merge, publish, +distribute, sublicense, and/or sell copies of the Software, and to +permit persons to whom the Software is furnished to do so, subject to +the following conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE +LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION +OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION +WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/Makefile b/Godeps/_workspace/src/github.com/Shopify/sarama/Makefile new file mode 100644 index 0000000000000..b76e97a97808b --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/Makefile @@ -0,0 +1,24 @@ +default: fmt vet errcheck test + +test: + go test -v -timeout 60s -race ./... + +vet: + go vet ./... + +errcheck: + errcheck github.com/Shopify/sarama/... + +fmt: + @if [ -n "$$(go fmt ./...)" ]; then echo 'Please run go fmt on your code.' && exit 1; fi + +install_dependencies: install_errcheck install_go_vet get + +install_errcheck: + go get github.com/kisielk/errcheck + +install_go_vet: + go get golang.org/x/tools/cmd/vet + +get: + go get -t diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/README.md b/Godeps/_workspace/src/github.com/Shopify/sarama/README.md new file mode 100644 index 0000000000000..486372730db11 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/README.md @@ -0,0 +1,31 @@ +sarama +====== + +[![GoDoc](https://godoc.org/github.com/Shopify/sarama?status.png)](https://godoc.org/github.com/Shopify/sarama) +[![Build Status](https://travis-ci.org/Shopify/sarama.svg?branch=master)](https://travis-ci.org/Shopify/sarama) + +Sarama is an MIT-licensed Go client library for Apache Kafka 0.8 (and later). + +### Getting started + +- API documentation and example are available via godoc at https://godoc.org/github.com/Shopify/sarama. +- Mocks for testing are available in the [mocks](./mocks) subpackage. +- The [examples](./examples) directory contains more elaborate example applications. +- The [tools](./tools) directory contains command line tools that can be useful for testing, diagnostics, and instrumentation. +- There is a google group for Kafka client users and authors at https://groups.google.com/forum/#!forum/kafka-clients + +### Compatibility and API stability + +Sarama provides a "2 releases + 2 months" compatibility guarantee: we support the two latest releases of Kafka +and Go, and we provide a two month grace period for older releases. This means we currently officially +support Go 1.3 and 1.4, and Kafka 0.8.1 and 0.8.2. + +Sarama follows semantic versioning and provides API stability via the gopkg.in service. +You can import a version with a guaranteed stable API via http://gopkg.in/Shopify/sarama.v1. +A changelog is available [here](CHANGELOG.md). + +### Other + +* [Sarama wiki](https://github.com/Shopify/sarama/wiki) to get started hacking on sarama itself. +* [Kafka Project Home](https://kafka.apache.org/) +* [Kafka Protocol Specification](https://cwiki.apache.org/confluence/display/KAFKA/A+Guide+To+The+Kafka+Protocol) diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/Vagrantfile b/Godeps/_workspace/src/github.com/Shopify/sarama/Vagrantfile new file mode 100644 index 0000000000000..4862dd93689a6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/Vagrantfile @@ -0,0 +1,22 @@ +# -*- mode: ruby -*- +# vi: set ft=ruby : + +# Vagrantfile API/syntax version. Don't touch unless you know what you're doing! +VAGRANTFILE_API_VERSION = "2" + +MEMORY = 3072 + +Vagrant.configure(VAGRANTFILE_API_VERSION) do |config| + config.vm.box = "hashicorp/precise64" + + config.vm.provision :shell, path: "vagrant/provision.sh" + + config.vm.network "private_network", ip: "192.168.100.67" + + config.vm.provider "vmware_fusion" do |v| + v.vmx["memsize"] = MEMORY.to_s + end + config.vm.provider "virtualbox" do |v| + v.memory = MEMORY + end +end diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/async_producer.go b/Godeps/_workspace/src/github.com/Shopify/sarama/async_producer.go new file mode 100644 index 0000000000000..8e229490f8640 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/async_producer.go @@ -0,0 +1,924 @@ +package sarama + +import ( + "fmt" + "sync" + "time" + + "github.com/eapache/go-resiliency/breaker" + "github.com/eapache/queue" +) + +// AsyncProducer publishes Kafka messages using a non-blocking API. It routes messages +// to the correct broker for the provided topic-partition, refreshing metadata as appropriate, +// and parses responses for errors. You must read from the Errors() channel or the +// producer will deadlock. You must call Close() or AsyncClose() on a producer to avoid +// leaks: it will not be garbage-collected automatically when it passes out of +// scope. +type AsyncProducer interface { + + // AsyncClose triggers a shutdown of the producer, flushing any messages it may have + // buffered. The shutdown has completed when both the Errors and Successes channels + // have been closed. When calling AsyncClose, you *must* continue to read from those + // channels in order to drain the results of any messages in flight. + AsyncClose() + + // Close shuts down the producer and flushes any messages it may have buffered. + // You must call this function before a producer object passes out of scope, as + // it may otherwise leak memory. You must call this before calling Close on the + // underlying client. + Close() error + + // Input is the input channel for the user to write messages to that they wish to send. + Input() chan<- *ProducerMessage + + // Successes is the success output channel back to the user when AckSuccesses is enabled. + // If Return.Successes is true, you MUST read from this channel or the Producer will deadlock. + // It is suggested that you send and read messages together in a single select statement. + Successes() <-chan *ProducerMessage + + // Errors is the error output channel back to the user. You MUST read from this channel + // or the Producer will deadlock when the channel is full. Alternatively, you can set + // Producer.Return.Errors in your config to false, which prevents errors to be returned. + Errors() <-chan *ProducerError +} + +type asyncProducer struct { + client Client + conf *Config + ownClient bool + + errors chan *ProducerError + input, successes, retries chan *ProducerMessage + inFlight sync.WaitGroup + + brokers map[*Broker]chan<- *ProducerMessage + brokerRefs map[chan<- *ProducerMessage]int + brokerLock sync.Mutex +} + +// NewAsyncProducer creates a new AsyncProducer using the given broker addresses and configuration. +func NewAsyncProducer(addrs []string, conf *Config) (AsyncProducer, error) { + client, err := NewClient(addrs, conf) + if err != nil { + return nil, err + } + + p, err := NewAsyncProducerFromClient(client) + if err != nil { + return nil, err + } + p.(*asyncProducer).ownClient = true + return p, nil +} + +// NewAsyncProducerFromClient creates a new Producer using the given client. It is still +// necessary to call Close() on the underlying client when shutting down this producer. +func NewAsyncProducerFromClient(client Client) (AsyncProducer, error) { + // Check that we are not dealing with a closed Client before processing any other arguments + if client.Closed() { + return nil, ErrClosedClient + } + + p := &asyncProducer{ + client: client, + conf: client.Config(), + errors: make(chan *ProducerError), + input: make(chan *ProducerMessage), + successes: make(chan *ProducerMessage), + retries: make(chan *ProducerMessage), + brokers: make(map[*Broker]chan<- *ProducerMessage), + brokerRefs: make(map[chan<- *ProducerMessage]int), + } + + // launch our singleton dispatchers + go withRecover(p.dispatcher) + go withRecover(p.retryHandler) + + return p, nil +} + +type flagSet int8 + +const ( + chaser flagSet = 1 << iota // message is last in a group that failed + shutdown // start the shutdown process +) + +// ProducerMessage is the collection of elements passed to the Producer in order to send a message. +type ProducerMessage struct { + Topic string // The Kafka topic for this message. + Key Encoder // The partitioning key for this message. It must implement the Encoder interface. Pre-existing Encoders include StringEncoder and ByteEncoder. + Value Encoder // The actual message to store in Kafka. It must implement the Encoder interface. Pre-existing Encoders include StringEncoder and ByteEncoder. + + // These are filled in by the producer as the message is processed + Offset int64 // Offset is the offset of the message stored on the broker. This is only guaranteed to be defined if the message was successfully delivered and RequiredAcks is not NoResponse. + Partition int32 // Partition is the partition that the message was sent to. This is only guaranteed to be defined if the message was successfully delivered. + + Metadata interface{} // This field is used to hold arbitrary data you wish to include so it will be available when receiving on the Successes and Errors channels. Sarama completely ignores this field and is only to be used for pass-through data. + + retries int + flags flagSet +} + +func (m *ProducerMessage) byteSize() int { + size := 26 // the metadata overhead of CRC, flags, etc. + if m.Key != nil { + size += m.Key.Length() + } + if m.Value != nil { + size += m.Value.Length() + } + return size +} + +func (m *ProducerMessage) clear() { + m.flags = 0 + m.retries = 0 +} + +// ProducerError is the type of error generated when the producer fails to deliver a message. +// It contains the original ProducerMessage as well as the actual error value. +type ProducerError struct { + Msg *ProducerMessage + Err error +} + +func (pe ProducerError) Error() string { + return fmt.Sprintf("kafka: Failed to produce message to topic %s: %s", pe.Msg.Topic, pe.Err) +} + +// ProducerErrors is a type that wraps a batch of "ProducerError"s and implements the Error interface. +// It can be returned from the Producer's Close method to avoid the need to manually drain the Errors channel +// when closing a producer. +type ProducerErrors []*ProducerError + +func (pe ProducerErrors) Error() string { + return fmt.Sprintf("kafka: Failed to deliver %d messages.", len(pe)) +} + +func (p *asyncProducer) Errors() <-chan *ProducerError { + return p.errors +} + +func (p *asyncProducer) Successes() <-chan *ProducerMessage { + return p.successes +} + +func (p *asyncProducer) Input() chan<- *ProducerMessage { + return p.input +} + +func (p *asyncProducer) Close() error { + p.AsyncClose() + + if p.conf.Producer.Return.Successes { + go withRecover(func() { + for _ = range p.successes { + } + }) + } + + var errors ProducerErrors + if p.conf.Producer.Return.Errors { + for event := range p.errors { + errors = append(errors, event) + } + } + + if len(errors) > 0 { + return errors + } + return nil +} + +func (p *asyncProducer) AsyncClose() { + go withRecover(p.shutdown) +} + +// singleton +// dispatches messages by topic +func (p *asyncProducer) dispatcher() { + handlers := make(map[string]chan<- *ProducerMessage) + shuttingDown := false + + for msg := range p.input { + if msg == nil { + Logger.Println("Something tried to send a nil message, it was ignored.") + continue + } + + if msg.flags&shutdown != 0 { + shuttingDown = true + p.inFlight.Done() + continue + } else if msg.retries == 0 { + if shuttingDown { + // we can't just call returnError here because that decrements the wait group, + // which hasn't been incremented yet for this message, and shouldn't be + pErr := &ProducerError{Msg: msg, Err: ErrShuttingDown} + if p.conf.Producer.Return.Errors { + p.errors <- pErr + } else { + Logger.Println(pErr) + } + continue + } + p.inFlight.Add(1) + } + + if (p.conf.Producer.Compression == CompressionNone && msg.Value != nil && msg.Value.Length() > p.conf.Producer.MaxMessageBytes) || + (msg.byteSize() > p.conf.Producer.MaxMessageBytes) { + + p.returnError(msg, ErrMessageSizeTooLarge) + continue + } + + handler := handlers[msg.Topic] + if handler == nil { + handler = p.newTopicProducer(msg.Topic) + handlers[msg.Topic] = handler + } + + handler <- msg + } + + for _, handler := range handlers { + close(handler) + } +} + +// one per topic +// partitions messages, then dispatches them by partition +type topicProducer struct { + parent *asyncProducer + topic string + input <-chan *ProducerMessage + + breaker *breaker.Breaker + handlers map[int32]chan<- *ProducerMessage + partitioner Partitioner +} + +func (p *asyncProducer) newTopicProducer(topic string) chan<- *ProducerMessage { + input := make(chan *ProducerMessage, p.conf.ChannelBufferSize) + tp := &topicProducer{ + parent: p, + topic: topic, + input: input, + breaker: breaker.New(3, 1, 10*time.Second), + handlers: make(map[int32]chan<- *ProducerMessage), + partitioner: p.conf.Producer.Partitioner(topic), + } + go withRecover(tp.dispatch) + return input +} + +func (tp *topicProducer) dispatch() { + for msg := range tp.input { + if msg.retries == 0 { + if err := tp.partitionMessage(msg); err != nil { + tp.parent.returnError(msg, err) + continue + } + } + + handler := tp.handlers[msg.Partition] + if handler == nil { + handler = tp.parent.newPartitionProducer(msg.Topic, msg.Partition) + tp.handlers[msg.Partition] = handler + } + + handler <- msg + } + + for _, handler := range tp.handlers { + close(handler) + } +} + +func (tp *topicProducer) partitionMessage(msg *ProducerMessage) error { + var partitions []int32 + + err := tp.breaker.Run(func() (err error) { + if tp.partitioner.RequiresConsistency() { + partitions, err = tp.parent.client.Partitions(msg.Topic) + } else { + partitions, err = tp.parent.client.WritablePartitions(msg.Topic) + } + return + }) + + if err != nil { + return err + } + + numPartitions := int32(len(partitions)) + + if numPartitions == 0 { + return ErrLeaderNotAvailable + } + + choice, err := tp.partitioner.Partition(msg, numPartitions) + + if err != nil { + return err + } else if choice < 0 || choice >= numPartitions { + return ErrInvalidPartition + } + + msg.Partition = partitions[choice] + + return nil +} + +// one per partition per topic +// dispatches messages to the appropriate broker +// also responsible for maintaining message order during retries +type partitionProducer struct { + parent *asyncProducer + topic string + partition int32 + input <-chan *ProducerMessage + + leader *Broker + breaker *breaker.Breaker + output chan<- *ProducerMessage + + // highWatermark tracks the "current" retry level, which is the only one where we actually let messages through, + // all other messages get buffered in retryState[msg.retries].buf to preserve ordering + // retryState[msg.retries].expectChaser simply tracks whether we've seen a chaser message for a given level (and + // therefore whether our buffer is complete and safe to flush) + highWatermark int + retryState []partitionRetryState +} + +type partitionRetryState struct { + buf []*ProducerMessage + expectChaser bool +} + +func (p *asyncProducer) newPartitionProducer(topic string, partition int32) chan<- *ProducerMessage { + input := make(chan *ProducerMessage, p.conf.ChannelBufferSize) + pp := &partitionProducer{ + parent: p, + topic: topic, + partition: partition, + input: input, + + breaker: breaker.New(3, 1, 10*time.Second), + retryState: make([]partitionRetryState, p.conf.Producer.Retry.Max+1), + } + go withRecover(pp.dispatch) + return input +} + +func (pp *partitionProducer) dispatch() { + // try to prefetch the leader; if this doesn't work, we'll do a proper call to `updateLeader` + // on the first message + pp.leader, _ = pp.parent.client.Leader(pp.topic, pp.partition) + if pp.leader != nil { + pp.output = pp.parent.getBrokerProducer(pp.leader) + } + + for msg := range pp.input { + if msg.retries > pp.highWatermark { + // a new, higher, retry level; handle it and then back off + pp.newHighWatermark(msg.retries) + time.Sleep(pp.parent.conf.Producer.Retry.Backoff) + } else if pp.highWatermark > 0 { + // we are retrying something (else highWatermark would be 0) but this message is not a *new* retry level + if msg.retries < pp.highWatermark { + // in fact this message is not even the current retry level, so buffer it for now (unless it's a just a chaser) + if msg.flags&chaser == chaser { + pp.retryState[msg.retries].expectChaser = false + pp.parent.inFlight.Done() // this chaser is now handled and will be garbage collected + } else { + pp.retryState[msg.retries].buf = append(pp.retryState[msg.retries].buf, msg) + } + continue + } else if msg.flags&chaser == chaser { + // this message is of the current retry level (msg.retries == highWatermark) and the chaser flag is set, + // meaning this retry level is done and we can go down (at least) one level and flush that + pp.retryState[pp.highWatermark].expectChaser = false + pp.flushRetryBuffers() + pp.parent.inFlight.Done() // this chaser is now handled and will be garbage collected + continue + } + } + + // if we made it this far then the current msg contains real data, and can be sent to the next goroutine + // without breaking any of our ordering guarantees + + if pp.output == nil { + if err := pp.updateLeader(); err != nil { + pp.parent.returnError(msg, err) + time.Sleep(pp.parent.conf.Producer.Retry.Backoff) + continue + } + Logger.Printf("producer/leader/%s/%d selected broker %d\n", pp.topic, pp.partition, pp.leader.ID()) + } + + pp.output <- msg + } + + if pp.output != nil { + pp.parent.unrefBrokerProducer(pp.leader, pp.output) + } +} + +func (pp *partitionProducer) newHighWatermark(hwm int) { + Logger.Printf("producer/leader/%s/%d state change to [retrying-%d]\n", pp.topic, pp.partition, hwm) + pp.highWatermark = hwm + + // send off a chaser so that we know when everything "in between" has made it + // back to us and we can safely flush the backlog (otherwise we risk re-ordering messages) + pp.retryState[pp.highWatermark].expectChaser = true + pp.parent.inFlight.Add(1) // we're generating a chaser message; track it so we don't shut down while it's still inflight + pp.output <- &ProducerMessage{Topic: pp.topic, Partition: pp.partition, flags: chaser, retries: pp.highWatermark - 1} + + // a new HWM means that our current broker selection is out of date + Logger.Printf("producer/leader/%s/%d abandoning broker %d\n", pp.topic, pp.partition, pp.leader.ID()) + pp.parent.unrefBrokerProducer(pp.leader, pp.output) + pp.output = nil +} + +func (pp *partitionProducer) flushRetryBuffers() { + Logger.Printf("producer/leader/%s/%d state change to [flushing-%d]\n", pp.topic, pp.partition, pp.highWatermark) + for { + pp.highWatermark-- + + if pp.output == nil { + if err := pp.updateLeader(); err != nil { + pp.parent.returnErrors(pp.retryState[pp.highWatermark].buf, err) + goto flushDone + } + Logger.Printf("producer/leader/%s/%d selected broker %d\n", pp.topic, pp.partition, pp.leader.ID()) + } + + for _, msg := range pp.retryState[pp.highWatermark].buf { + pp.output <- msg + } + + flushDone: + pp.retryState[pp.highWatermark].buf = nil + if pp.retryState[pp.highWatermark].expectChaser { + Logger.Printf("producer/leader/%s/%d state change to [retrying-%d]\n", pp.topic, pp.partition, pp.highWatermark) + break + } else if pp.highWatermark == 0 { + Logger.Printf("producer/leader/%s/%d state change to [normal]\n", pp.topic, pp.partition) + break + } + } +} + +func (pp *partitionProducer) updateLeader() error { + return pp.breaker.Run(func() (err error) { + if err = pp.parent.client.RefreshMetadata(pp.topic); err != nil { + return err + } + + if pp.leader, err = pp.parent.client.Leader(pp.topic, pp.partition); err != nil { + return err + } + + pp.output = pp.parent.getBrokerProducer(pp.leader) + return nil + }) +} + +// one per broker, constructs both an aggregator and a flusher +func (p *asyncProducer) newBrokerProducer(broker *Broker) chan<- *ProducerMessage { + input := make(chan *ProducerMessage) + bridge := make(chan []*ProducerMessage) + + a := &aggregator{ + parent: p, + broker: broker, + input: input, + output: bridge, + } + go withRecover(a.run) + + f := &flusher{ + parent: p, + broker: broker, + input: bridge, + currentRetries: make(map[string]map[int32]error), + } + go withRecover(f.run) + + return input +} + +// groups messages together into appropriately-sized batches for sending to the broker +// based on https://godoc.org/github.com/eapache/channels#BatchingChannel +type aggregator struct { + parent *asyncProducer + broker *Broker + input <-chan *ProducerMessage + output chan<- []*ProducerMessage + + buffer []*ProducerMessage + bufferBytes int + timer <-chan time.Time +} + +func (a *aggregator) run() { + var output chan<- []*ProducerMessage + + for { + select { + case msg := <-a.input: + if msg == nil { + goto shutdown + } + + if a.wouldOverflow(msg) { + Logger.Printf("producer/aggregator/%d maximum request accumulated, forcing blocking flush\n", a.broker.ID()) + a.output <- a.buffer + a.reset() + output = nil + } + + a.buffer = append(a.buffer, msg) + a.bufferBytes += msg.byteSize() + + if a.readyToFlush(msg) { + output = a.output + } else if a.parent.conf.Producer.Flush.Frequency > 0 && a.timer == nil { + a.timer = time.After(a.parent.conf.Producer.Flush.Frequency) + } + case <-a.timer: + output = a.output + case output <- a.buffer: + a.reset() + output = nil + } + } + +shutdown: + if len(a.buffer) > 0 { + a.output <- a.buffer + } + close(a.output) +} + +func (a *aggregator) wouldOverflow(msg *ProducerMessage) bool { + switch { + // Would we overflow our maximum possible size-on-the-wire? 10KiB is arbitrary overhead for safety. + case a.bufferBytes+msg.byteSize() >= int(MaxRequestSize-(10*1024)): + return true + // Would we overflow the size-limit of a compressed message-batch? + case a.parent.conf.Producer.Compression != CompressionNone && a.bufferBytes+msg.byteSize() >= a.parent.conf.Producer.MaxMessageBytes: + return true + // Would we overflow simply in number of messages? + case a.parent.conf.Producer.Flush.MaxMessages > 0 && len(a.buffer) >= a.parent.conf.Producer.Flush.MaxMessages: + return true + default: + return false + } +} + +func (a *aggregator) readyToFlush(msg *ProducerMessage) bool { + switch { + // If all three config values are 0, we always flush as-fast-as-possible + case a.parent.conf.Producer.Flush.Frequency == 0 && a.parent.conf.Producer.Flush.Bytes == 0 && a.parent.conf.Producer.Flush.Messages == 0: + return true + // If the messages is a chaser we must flush to maintain the state-machine + case msg.flags&chaser == chaser: + return true + // If we've passed the message trigger-point + case a.parent.conf.Producer.Flush.Messages > 0 && len(a.buffer) >= a.parent.conf.Producer.Flush.Messages: + return true + // If we've passed the byte trigger-point + case a.parent.conf.Producer.Flush.Bytes > 0 && a.bufferBytes >= a.parent.conf.Producer.Flush.Bytes: + return true + default: + return false + } +} + +func (a *aggregator) reset() { + a.timer = nil + a.buffer = nil + a.bufferBytes = 0 +} + +// takes a batch at a time from the aggregator and sends to the broker +type flusher struct { + parent *asyncProducer + broker *Broker + input <-chan []*ProducerMessage + + currentRetries map[string]map[int32]error +} + +func (f *flusher) run() { + var closing error + + Logger.Printf("producer/flusher/%d starting up\n", f.broker.ID()) + + for batch := range f.input { + if closing != nil { + f.parent.retryMessages(batch, closing) + continue + } + + msgSets := f.groupAndFilter(batch) + request := f.parent.buildRequest(msgSets) + if request == nil { + continue + } + + response, err := f.broker.Produce(request) + + switch err.(type) { + case nil: + break + case PacketEncodingError: + f.parent.returnErrors(batch, err) + continue + default: + Logger.Printf("producer/flusher/%d state change to [closing] because %s\n", f.broker.ID(), err) + f.parent.abandonBrokerConnection(f.broker) + _ = f.broker.Close() + closing = err + f.parent.retryMessages(batch, err) + continue + } + + if response == nil { + // this only happens when RequiredAcks is NoResponse, so we have to assume success + f.parent.returnSuccesses(batch) + continue + } + + f.parseResponse(msgSets, response) + } + Logger.Printf("producer/flusher/%d shut down\n", f.broker.ID()) +} + +func (f *flusher) groupAndFilter(batch []*ProducerMessage) map[string]map[int32][]*ProducerMessage { + msgSets := make(map[string]map[int32][]*ProducerMessage) + + for i, msg := range batch { + + if f.currentRetries[msg.Topic] != nil && f.currentRetries[msg.Topic][msg.Partition] != nil { + // we're currently retrying this partition so we need to filter out this message + f.parent.retryMessages([]*ProducerMessage{msg}, f.currentRetries[msg.Topic][msg.Partition]) + batch[i] = nil + + if msg.flags&chaser == chaser { + // ...but now we can start processing future messages again + Logger.Printf("producer/flusher/%d state change to [normal] on %s/%d\n", + f.broker.ID(), msg.Topic, msg.Partition) + delete(f.currentRetries[msg.Topic], msg.Partition) + } + + continue + } + + partitionSet := msgSets[msg.Topic] + if partitionSet == nil { + partitionSet = make(map[int32][]*ProducerMessage) + msgSets[msg.Topic] = partitionSet + } + + partitionSet[msg.Partition] = append(partitionSet[msg.Partition], msg) + } + + return msgSets +} + +func (f *flusher) parseResponse(msgSets map[string]map[int32][]*ProducerMessage, response *ProduceResponse) { + // we iterate through the blocks in the request set, not the response, so that we notice + // if the response is missing a block completely + for topic, partitionSet := range msgSets { + for partition, msgs := range partitionSet { + block := response.GetBlock(topic, partition) + if block == nil { + f.parent.returnErrors(msgs, ErrIncompleteResponse) + continue + } + + switch block.Err { + // Success + case ErrNoError: + for i := range msgs { + msgs[i].Offset = block.Offset + int64(i) + } + f.parent.returnSuccesses(msgs) + // Retriable errors + case ErrUnknownTopicOrPartition, ErrNotLeaderForPartition, ErrLeaderNotAvailable, + ErrRequestTimedOut, ErrNotEnoughReplicas, ErrNotEnoughReplicasAfterAppend: + Logger.Printf("producer/flusher/%d state change to [retrying] on %s/%d because %v\n", + f.broker.ID(), topic, partition, block.Err) + if f.currentRetries[topic] == nil { + f.currentRetries[topic] = make(map[int32]error) + } + f.currentRetries[topic][partition] = block.Err + f.parent.retryMessages(msgs, block.Err) + // Other non-retriable errors + default: + f.parent.returnErrors(msgs, block.Err) + } + } + } +} + +// singleton +// effectively a "bridge" between the flushers and the dispatcher in order to avoid deadlock +// based on https://godoc.org/github.com/eapache/channels#InfiniteChannel +func (p *asyncProducer) retryHandler() { + var msg *ProducerMessage + buf := queue.New() + + for { + if buf.Length() == 0 { + msg = <-p.retries + } else { + select { + case msg = <-p.retries: + case p.input <- buf.Peek().(*ProducerMessage): + buf.Remove() + continue + } + } + + if msg == nil { + return + } + + buf.Add(msg) + } +} + +// utility functions + +func (p *asyncProducer) shutdown() { + Logger.Println("Producer shutting down.") + p.inFlight.Add(1) + p.input <- &ProducerMessage{flags: shutdown} + + p.inFlight.Wait() + + if p.ownClient { + err := p.client.Close() + if err != nil { + Logger.Println("producer/shutdown failed to close the embedded client:", err) + } + } + + close(p.input) + close(p.retries) + close(p.errors) + close(p.successes) +} + +func (p *asyncProducer) buildRequest(batch map[string]map[int32][]*ProducerMessage) *ProduceRequest { + + req := &ProduceRequest{RequiredAcks: p.conf.Producer.RequiredAcks, Timeout: int32(p.conf.Producer.Timeout / time.Millisecond)} + empty := true + + for topic, partitionSet := range batch { + for partition, msgSet := range partitionSet { + setToSend := new(MessageSet) + setSize := 0 + for _, msg := range msgSet { + var keyBytes, valBytes []byte + var err error + if msg.Key != nil { + if keyBytes, err = msg.Key.Encode(); err != nil { + p.returnError(msg, err) + continue + } + } + if msg.Value != nil { + if valBytes, err = msg.Value.Encode(); err != nil { + p.returnError(msg, err) + continue + } + } + + if p.conf.Producer.Compression != CompressionNone && setSize+msg.byteSize() > p.conf.Producer.MaxMessageBytes { + // compression causes message-sets to be wrapped as single messages, which have tighter + // size requirements, so we have to respect those limits + valBytes, err := encode(setToSend) + if err != nil { + Logger.Println(err) // if this happens, it's basically our fault. + panic(err) + } + req.AddMessage(topic, partition, &Message{Codec: p.conf.Producer.Compression, Key: nil, Value: valBytes}) + setToSend = new(MessageSet) + setSize = 0 + } + setSize += msg.byteSize() + + setToSend.addMessage(&Message{Codec: CompressionNone, Key: keyBytes, Value: valBytes}) + empty = false + } + + if p.conf.Producer.Compression == CompressionNone { + req.AddSet(topic, partition, setToSend) + } else { + valBytes, err := encode(setToSend) + if err != nil { + Logger.Println(err) // if this happens, it's basically our fault. + panic(err) + } + req.AddMessage(topic, partition, &Message{Codec: p.conf.Producer.Compression, Key: nil, Value: valBytes}) + } + } + } + + if empty { + return nil + } + return req +} + +func (p *asyncProducer) returnError(msg *ProducerMessage, err error) { + msg.clear() + pErr := &ProducerError{Msg: msg, Err: err} + if p.conf.Producer.Return.Errors { + p.errors <- pErr + } else { + Logger.Println(pErr) + } + p.inFlight.Done() +} + +func (p *asyncProducer) returnErrors(batch []*ProducerMessage, err error) { + for _, msg := range batch { + if msg != nil { + p.returnError(msg, err) + } + } +} + +func (p *asyncProducer) returnSuccesses(batch []*ProducerMessage) { + for _, msg := range batch { + if msg == nil { + continue + } + if p.conf.Producer.Return.Successes { + msg.clear() + p.successes <- msg + } + p.inFlight.Done() + } +} + +func (p *asyncProducer) retryMessages(batch []*ProducerMessage, err error) { + for _, msg := range batch { + if msg == nil { + continue + } + if msg.retries >= p.conf.Producer.Retry.Max { + p.returnError(msg, err) + } else { + msg.retries++ + p.retries <- msg + } + } +} + +func (p *asyncProducer) getBrokerProducer(broker *Broker) chan<- *ProducerMessage { + p.brokerLock.Lock() + defer p.brokerLock.Unlock() + + bp := p.brokers[broker] + + if bp == nil { + bp = p.newBrokerProducer(broker) + p.brokers[broker] = bp + p.brokerRefs[bp] = 0 + } + + p.brokerRefs[bp]++ + + return bp +} + +func (p *asyncProducer) unrefBrokerProducer(broker *Broker, bp chan<- *ProducerMessage) { + p.brokerLock.Lock() + defer p.brokerLock.Unlock() + + p.brokerRefs[bp]-- + if p.brokerRefs[bp] == 0 { + close(bp) + delete(p.brokerRefs, bp) + + if p.brokers[broker] == bp { + delete(p.brokers, broker) + } + } +} + +func (p *asyncProducer) abandonBrokerConnection(broker *Broker) { + p.brokerLock.Lock() + defer p.brokerLock.Unlock() + + delete(p.brokers, broker) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/async_producer_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/async_producer_test.go new file mode 100644 index 0000000000000..403456839fc40 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/async_producer_test.go @@ -0,0 +1,743 @@ +package sarama + +import ( + "errors" + "log" + "os" + "os/signal" + "sync" + "testing" + "time" +) + +const TestMessage = "ABC THE MESSAGE" + +func closeProducer(t *testing.T, p AsyncProducer) { + var wg sync.WaitGroup + p.AsyncClose() + + wg.Add(2) + go func() { + for _ = range p.Successes() { + t.Error("Unexpected message on Successes()") + } + wg.Done() + }() + go func() { + for msg := range p.Errors() { + t.Error(msg.Err) + } + wg.Done() + }() + wg.Wait() +} + +func expectResults(t *testing.T, p AsyncProducer, successes, errors int) { + for successes > 0 || errors > 0 { + select { + case msg := <-p.Errors(): + if msg.Msg.flags != 0 { + t.Error("Message had flags set") + } + errors-- + if errors < 0 { + t.Error(msg.Err) + } + case msg := <-p.Successes(): + if msg.flags != 0 { + t.Error("Message had flags set") + } + successes-- + if successes < 0 { + t.Error("Too many successes") + } + } + } +} + +type testPartitioner chan *int32 + +func (p testPartitioner) Partition(msg *ProducerMessage, numPartitions int32) (int32, error) { + part := <-p + if part == nil { + return 0, errors.New("BOOM") + } + + return *part, nil +} + +func (p testPartitioner) RequiresConsistency() bool { + return true +} + +func (p testPartitioner) feed(partition int32) { + p <- &partition +} + +func TestAsyncProducer(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader := newMockBroker(t, 2) + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker(leader.Addr(), leader.BrokerID()) + metadataResponse.AddTopicPartition("my_topic", 0, leader.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataResponse) + + prodSuccess := new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", 0, ErrNoError) + leader.Returns(prodSuccess) + + config := NewConfig() + config.Producer.Flush.Messages = 10 + config.Producer.Return.Successes = true + producer, err := NewAsyncProducer([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + for i := 0; i < 10; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage), Metadata: i} + } + for i := 0; i < 10; i++ { + select { + case msg := <-producer.Errors(): + t.Error(msg.Err) + if msg.Msg.flags != 0 { + t.Error("Message had flags set") + } + case msg := <-producer.Successes(): + if msg.flags != 0 { + t.Error("Message had flags set") + } + if msg.Metadata.(int) != i { + t.Error("Message metadata did not match") + } + } + } + + closeProducer(t, producer) + leader.Close() + seedBroker.Close() +} + +func TestAsyncProducerMultipleFlushes(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader := newMockBroker(t, 2) + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker(leader.Addr(), leader.BrokerID()) + metadataResponse.AddTopicPartition("my_topic", 0, leader.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataResponse) + + prodSuccess := new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", 0, ErrNoError) + leader.Returns(prodSuccess) + leader.Returns(prodSuccess) + leader.Returns(prodSuccess) + + config := NewConfig() + config.Producer.Flush.Messages = 5 + config.Producer.Return.Successes = true + producer, err := NewAsyncProducer([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + for flush := 0; flush < 3; flush++ { + for i := 0; i < 5; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + } + expectResults(t, producer, 5, 0) + } + + closeProducer(t, producer) + leader.Close() + seedBroker.Close() +} + +func TestAsyncProducerMultipleBrokers(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader0 := newMockBroker(t, 2) + leader1 := newMockBroker(t, 3) + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker(leader0.Addr(), leader0.BrokerID()) + metadataResponse.AddBroker(leader1.Addr(), leader1.BrokerID()) + metadataResponse.AddTopicPartition("my_topic", 0, leader0.BrokerID(), nil, nil, ErrNoError) + metadataResponse.AddTopicPartition("my_topic", 1, leader1.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataResponse) + + prodResponse0 := new(ProduceResponse) + prodResponse0.AddTopicPartition("my_topic", 0, ErrNoError) + leader0.Returns(prodResponse0) + + prodResponse1 := new(ProduceResponse) + prodResponse1.AddTopicPartition("my_topic", 1, ErrNoError) + leader1.Returns(prodResponse1) + + config := NewConfig() + config.Producer.Flush.Messages = 5 + config.Producer.Return.Successes = true + config.Producer.Partitioner = NewRoundRobinPartitioner + producer, err := NewAsyncProducer([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + for i := 0; i < 10; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + } + expectResults(t, producer, 10, 0) + + closeProducer(t, producer) + leader1.Close() + leader0.Close() + seedBroker.Close() +} + +func TestAsyncProducerCustomPartitioner(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader := newMockBroker(t, 2) + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker(leader.Addr(), leader.BrokerID()) + metadataResponse.AddTopicPartition("my_topic", 0, leader.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataResponse) + + prodResponse := new(ProduceResponse) + prodResponse.AddTopicPartition("my_topic", 0, ErrNoError) + leader.Returns(prodResponse) + + config := NewConfig() + config.Producer.Flush.Messages = 2 + config.Producer.Return.Successes = true + config.Producer.Partitioner = func(topic string) Partitioner { + p := make(testPartitioner) + go func() { + p.feed(0) + p <- nil + p <- nil + p <- nil + p.feed(0) + }() + return p + } + producer, err := NewAsyncProducer([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + for i := 0; i < 5; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + } + expectResults(t, producer, 2, 3) + + closeProducer(t, producer) + leader.Close() + seedBroker.Close() +} + +func TestAsyncProducerFailureRetry(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader1 := newMockBroker(t, 2) + leader2 := newMockBroker(t, 3) + + metadataLeader1 := new(MetadataResponse) + metadataLeader1.AddBroker(leader1.Addr(), leader1.BrokerID()) + metadataLeader1.AddTopicPartition("my_topic", 0, leader1.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataLeader1) + + config := NewConfig() + config.Producer.Flush.Messages = 10 + config.Producer.Return.Successes = true + config.Producer.Retry.Backoff = 0 + producer, err := NewAsyncProducer([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + seedBroker.Close() + + for i := 0; i < 10; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + } + prodNotLeader := new(ProduceResponse) + prodNotLeader.AddTopicPartition("my_topic", 0, ErrNotLeaderForPartition) + leader1.Returns(prodNotLeader) + + metadataLeader2 := new(MetadataResponse) + metadataLeader2.AddBroker(leader2.Addr(), leader2.BrokerID()) + metadataLeader2.AddTopicPartition("my_topic", 0, leader2.BrokerID(), nil, nil, ErrNoError) + leader1.Returns(metadataLeader2) + + prodSuccess := new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", 0, ErrNoError) + leader2.Returns(prodSuccess) + expectResults(t, producer, 10, 0) + leader1.Close() + + for i := 0; i < 10; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + } + leader2.Returns(prodSuccess) + expectResults(t, producer, 10, 0) + + leader2.Close() + closeProducer(t, producer) +} + +// If a Kafka broker becomes unavailable and then returns back in service, then +// producer reconnects to it and continues sending messages. +func TestAsyncProducerBrokerBounce(t *testing.T) { + // Given + seedBroker := newMockBroker(t, 1) + leader := newMockBroker(t, 2) + leaderAddr := leader.Addr() + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker(leaderAddr, leader.BrokerID()) + metadataResponse.AddTopicPartition("my_topic", 0, leader.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataResponse) + + prodSuccess := new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", 0, ErrNoError) + + config := NewConfig() + config.Producer.Flush.Messages = 1 + config.Producer.Return.Successes = true + config.Producer.Retry.Backoff = 0 + producer, err := NewAsyncProducer([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + leader.Returns(prodSuccess) + expectResults(t, producer, 1, 0) + + // When: a broker connection gets reset by a broker (network glitch, restart, you name it). + leader.Close() // producer should get EOF + leader = newMockBrokerAddr(t, 2, leaderAddr) // start it up again right away for giggles + seedBroker.Returns(metadataResponse) // tell it to go to broker 2 again + + // Then: a produced message goes through the new broker connection. + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + leader.Returns(prodSuccess) + expectResults(t, producer, 1, 0) + + closeProducer(t, producer) + seedBroker.Close() + leader.Close() +} + +func TestAsyncProducerBrokerBounceWithStaleMetadata(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader1 := newMockBroker(t, 2) + leader2 := newMockBroker(t, 3) + + metadataLeader1 := new(MetadataResponse) + metadataLeader1.AddBroker(leader1.Addr(), leader1.BrokerID()) + metadataLeader1.AddTopicPartition("my_topic", 0, leader1.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataLeader1) + + config := NewConfig() + config.Producer.Flush.Messages = 10 + config.Producer.Return.Successes = true + config.Producer.Retry.Max = 3 + config.Producer.Retry.Backoff = 0 + producer, err := NewAsyncProducer([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + for i := 0; i < 10; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + } + leader1.Close() // producer should get EOF + seedBroker.Returns(metadataLeader1) // tell it to go to leader1 again even though it's still down + seedBroker.Returns(metadataLeader1) // tell it to go to leader1 again even though it's still down + + // ok fine, tell it to go to leader2 finally + metadataLeader2 := new(MetadataResponse) + metadataLeader2.AddBroker(leader2.Addr(), leader2.BrokerID()) + metadataLeader2.AddTopicPartition("my_topic", 0, leader2.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataLeader2) + + prodSuccess := new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", 0, ErrNoError) + leader2.Returns(prodSuccess) + expectResults(t, producer, 10, 0) + seedBroker.Close() + leader2.Close() + + closeProducer(t, producer) +} + +func TestAsyncProducerMultipleRetries(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader1 := newMockBroker(t, 2) + leader2 := newMockBroker(t, 3) + + metadataLeader1 := new(MetadataResponse) + metadataLeader1.AddBroker(leader1.Addr(), leader1.BrokerID()) + metadataLeader1.AddTopicPartition("my_topic", 0, leader1.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataLeader1) + + config := NewConfig() + config.Producer.Flush.Messages = 10 + config.Producer.Return.Successes = true + config.Producer.Retry.Max = 4 + config.Producer.Retry.Backoff = 0 + producer, err := NewAsyncProducer([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + for i := 0; i < 10; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + } + prodNotLeader := new(ProduceResponse) + prodNotLeader.AddTopicPartition("my_topic", 0, ErrNotLeaderForPartition) + leader1.Returns(prodNotLeader) + + metadataLeader2 := new(MetadataResponse) + metadataLeader2.AddBroker(leader2.Addr(), leader2.BrokerID()) + metadataLeader2.AddTopicPartition("my_topic", 0, leader2.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataLeader2) + leader2.Returns(prodNotLeader) + seedBroker.Returns(metadataLeader1) + leader1.Returns(prodNotLeader) + seedBroker.Returns(metadataLeader1) + leader1.Returns(prodNotLeader) + seedBroker.Returns(metadataLeader2) + + prodSuccess := new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", 0, ErrNoError) + leader2.Returns(prodSuccess) + expectResults(t, producer, 10, 0) + + for i := 0; i < 10; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + } + leader2.Returns(prodSuccess) + expectResults(t, producer, 10, 0) + + seedBroker.Close() + leader1.Close() + leader2.Close() + closeProducer(t, producer) +} + +func TestAsyncProducerOutOfRetries(t *testing.T) { + t.Skip("Enable once bug #294 is fixed.") + + seedBroker := newMockBroker(t, 1) + leader := newMockBroker(t, 2) + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker(leader.Addr(), leader.BrokerID()) + metadataResponse.AddTopicPartition("my_topic", 0, leader.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataResponse) + + config := NewConfig() + config.Producer.Flush.Messages = 10 + config.Producer.Return.Successes = true + config.Producer.Retry.Backoff = 0 + config.Producer.Retry.Max = 0 + producer, err := NewAsyncProducer([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + for i := 0; i < 10; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + } + + prodNotLeader := new(ProduceResponse) + prodNotLeader.AddTopicPartition("my_topic", 0, ErrNotLeaderForPartition) + leader.Returns(prodNotLeader) + + for i := 0; i < 10; i++ { + select { + case msg := <-producer.Errors(): + if msg.Err != ErrNotLeaderForPartition { + t.Error(msg.Err) + } + case <-producer.Successes(): + t.Error("Unexpected success") + } + } + + seedBroker.Returns(metadataResponse) + + for i := 0; i < 10; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + } + + prodSuccess := new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", 0, ErrNoError) + leader.Returns(prodSuccess) + + expectResults(t, producer, 10, 0) + + leader.Close() + seedBroker.Close() + safeClose(t, producer) +} + +func TestAsyncProducerRetryWithReferenceOpen(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader := newMockBroker(t, 2) + leaderAddr := leader.Addr() + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker(leaderAddr, leader.BrokerID()) + metadataResponse.AddTopicPartition("my_topic", 0, leader.BrokerID(), nil, nil, ErrNoError) + metadataResponse.AddTopicPartition("my_topic", 1, leader.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataResponse) + + config := NewConfig() + config.Producer.Return.Successes = true + config.Producer.Retry.Backoff = 0 + config.Producer.Retry.Max = 1 + config.Producer.Partitioner = NewRoundRobinPartitioner + producer, err := NewAsyncProducer([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + // prime partition 0 + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + prodSuccess := new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", 0, ErrNoError) + leader.Returns(prodSuccess) + expectResults(t, producer, 1, 0) + + // prime partition 1 + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + prodSuccess = new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", 1, ErrNoError) + leader.Returns(prodSuccess) + expectResults(t, producer, 1, 0) + + // reboot the broker (the producer will get EOF on its existing connection) + leader.Close() + leader = newMockBrokerAddr(t, 2, leaderAddr) + + // send another message on partition 0 to trigger the EOF and retry + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + + // tell partition 0 to go to that broker again + seedBroker.Returns(metadataResponse) + + // succeed this time + prodSuccess = new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", 0, ErrNoError) + leader.Returns(prodSuccess) + expectResults(t, producer, 1, 0) + + // shutdown + closeProducer(t, producer) + seedBroker.Close() + leader.Close() +} + +func TestAsyncProducerFlusherRetryCondition(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader := newMockBroker(t, 2) + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker(leader.Addr(), leader.BrokerID()) + metadataResponse.AddTopicPartition("my_topic", 0, leader.BrokerID(), nil, nil, ErrNoError) + metadataResponse.AddTopicPartition("my_topic", 1, leader.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataResponse) + + config := NewConfig() + config.Producer.Flush.Messages = 5 + config.Producer.Return.Successes = true + config.Producer.Retry.Backoff = 0 + config.Producer.Retry.Max = 1 + config.Producer.Partitioner = NewManualPartitioner + producer, err := NewAsyncProducer([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + // prime partitions + for p := int32(0); p < 2; p++ { + for i := 0; i < 5; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage), Partition: p} + } + prodSuccess := new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", p, ErrNoError) + leader.Returns(prodSuccess) + expectResults(t, producer, 5, 0) + } + + // send more messages on partition 0 + for i := 0; i < 5; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage), Partition: 0} + } + prodNotLeader := new(ProduceResponse) + prodNotLeader.AddTopicPartition("my_topic", 0, ErrNotLeaderForPartition) + leader.Returns(prodNotLeader) + + // tell partition 0 to go to that broker again + seedBroker.Returns(metadataResponse) + + // succeed this time + prodSuccess := new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", 0, ErrNoError) + leader.Returns(prodSuccess) + expectResults(t, producer, 5, 0) + + // put five more through + for i := 0; i < 5; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage), Partition: 0} + } + leader.Returns(prodSuccess) + expectResults(t, producer, 5, 0) + + // shutdown + closeProducer(t, producer) + seedBroker.Close() + leader.Close() +} + +func TestAsyncProducerRetryShutdown(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader := newMockBroker(t, 2) + + metadataLeader := new(MetadataResponse) + metadataLeader.AddBroker(leader.Addr(), leader.BrokerID()) + metadataLeader.AddTopicPartition("my_topic", 0, leader.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataLeader) + + config := NewConfig() + config.Producer.Flush.Messages = 10 + config.Producer.Return.Successes = true + config.Producer.Retry.Backoff = 0 + producer, err := NewAsyncProducer([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + for i := 0; i < 10; i++ { + producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder(TestMessage)} + } + producer.AsyncClose() + time.Sleep(5 * time.Millisecond) // let the shutdown goroutine kick in + + producer.Input() <- &ProducerMessage{Topic: "FOO"} + if err := <-producer.Errors(); err.Err != ErrShuttingDown { + t.Error(err) + } + + prodNotLeader := new(ProduceResponse) + prodNotLeader.AddTopicPartition("my_topic", 0, ErrNotLeaderForPartition) + leader.Returns(prodNotLeader) + + seedBroker.Returns(metadataLeader) + + prodSuccess := new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", 0, ErrNoError) + leader.Returns(prodSuccess) + expectResults(t, producer, 10, 0) + + seedBroker.Close() + leader.Close() + + // wait for the async-closed producer to shut down fully + for err := range producer.Errors() { + t.Error(err) + } +} + +// This example shows how to use the producer while simultaneously +// reading the Errors channel to know about any failures. +func ExampleAsyncProducer_select() { + producer, err := NewAsyncProducer([]string{"localhost:9092"}, nil) + if err != nil { + panic(err) + } + + defer func() { + if err := producer.Close(); err != nil { + log.Fatalln(err) + } + }() + + // Trap SIGINT to trigger a shutdown. + signals := make(chan os.Signal, 1) + signal.Notify(signals, os.Interrupt) + + var enqueued, errors int +ProducerLoop: + for { + select { + case producer.Input() <- &ProducerMessage{Topic: "my_topic", Key: nil, Value: StringEncoder("testing 123")}: + enqueued++ + case err := <-producer.Errors(): + log.Println("Failed to produce message", err) + errors++ + case <-signals: + break ProducerLoop + } + } + + log.Printf("Enqueued: %d; errors: %d\n", enqueued, errors) +} + +// This example shows how to use the producer with separate goroutines +// reading from the Successes and Errors channels. Note that in order +// for the Successes channel to be populated, you have to set +// config.Producer.Return.Successes to true. +func ExampleAsyncProducer_goroutines() { + config := NewConfig() + config.Producer.Return.Successes = true + producer, err := NewAsyncProducer([]string{"localhost:9092"}, config) + if err != nil { + panic(err) + } + + // Trap SIGINT to trigger a graceful shutdown. + signals := make(chan os.Signal, 1) + signal.Notify(signals, os.Interrupt) + + var ( + wg sync.WaitGroup + enqueued, successes, errors int + ) + + wg.Add(1) + go func() { + defer wg.Done() + for _ = range producer.Successes() { + successes++ + } + }() + + wg.Add(1) + go func() { + defer wg.Done() + for err := range producer.Errors() { + log.Println(err) + errors++ + } + }() + +ProducerLoop: + for { + message := &ProducerMessage{Topic: "my_topic", Value: StringEncoder("testing 123")} + select { + case producer.Input() <- message: + enqueued++ + + case <-signals: + producer.AsyncClose() // Trigger a shutdown of the producer. + break ProducerLoop + } + } + + wg.Wait() + + log.Printf("Successfully produced: %d; errors: %d\n", successes, errors) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/broker.go b/Godeps/_workspace/src/github.com/Shopify/sarama/broker.go new file mode 100644 index 0000000000000..eb5bc0bf81605 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/broker.go @@ -0,0 +1,385 @@ +package sarama + +import ( + "crypto/tls" + "fmt" + "io" + "net" + "strconv" + "sync" + "sync/atomic" + "time" +) + +// Broker represents a single Kafka broker connection. All operations on this object are entirely concurrency-safe. +type Broker struct { + id int32 + addr string + + conf *Config + correlationID int32 + conn net.Conn + connErr error + lock sync.Mutex + opened int32 + + responses chan responsePromise + done chan bool +} + +type responsePromise struct { + correlationID int32 + packets chan []byte + errors chan error +} + +// NewBroker creates and returns a Broker targetting the given host:port address. +// This does not attempt to actually connect, you have to call Open() for that. +func NewBroker(addr string) *Broker { + return &Broker{id: -1, addr: addr} +} + +// Open tries to connect to the Broker if it is not already connected or connecting, but does not block +// waiting for the connection to complete. This means that any subsequent operations on the broker will +// block waiting for the connection to succeed or fail. To get the effect of a fully synchronous Open call, +// follow it by a call to Connected(). The only errors Open will return directly are ConfigurationError or +// AlreadyConnected. If conf is nil, the result of NewConfig() is used. +func (b *Broker) Open(conf *Config) error { + if conf == nil { + conf = NewConfig() + } + + err := conf.Validate() + if err != nil { + return err + } + + if !atomic.CompareAndSwapInt32(&b.opened, 0, 1) { + return ErrAlreadyConnected + } + + b.lock.Lock() + + if b.conn != nil { + b.lock.Unlock() + Logger.Printf("Failed to connect to broker %s: %s\n", b.addr, ErrAlreadyConnected) + return ErrAlreadyConnected + } + + go withRecover(func() { + defer b.lock.Unlock() + + dialer := net.Dialer{ + Timeout: conf.Net.DialTimeout, + KeepAlive: conf.Net.KeepAlive, + } + + if conf.Net.TLS.Enable { + b.conn, b.connErr = tls.DialWithDialer(&dialer, "tcp", b.addr, conf.Net.TLS.Config) + } else { + b.conn, b.connErr = dialer.Dial("tcp", b.addr) + } + if b.connErr != nil { + b.conn = nil + atomic.StoreInt32(&b.opened, 0) + Logger.Printf("Failed to connect to broker %s: %s\n", b.addr, b.connErr) + return + } + + b.conf = conf + b.done = make(chan bool) + b.responses = make(chan responsePromise, b.conf.Net.MaxOpenRequests-1) + + if b.id >= 0 { + Logger.Printf("Connected to broker at %s (registered as #%d)\n", b.addr, b.id) + } else { + Logger.Printf("Connected to broker at %s (unregistered)\n", b.addr) + } + go withRecover(b.responseReceiver) + }) + + return nil +} + +// Connected returns true if the broker is connected and false otherwise. If the broker is not +// connected but it had tried to connect, the error from that connection attempt is also returned. +func (b *Broker) Connected() (bool, error) { + b.lock.Lock() + defer b.lock.Unlock() + + return b.conn != nil, b.connErr +} + +func (b *Broker) Close() error { + b.lock.Lock() + defer b.lock.Unlock() + + if b.conn == nil { + return ErrNotConnected + } + + close(b.responses) + <-b.done + + err := b.conn.Close() + + b.conn = nil + b.connErr = nil + b.done = nil + b.responses = nil + + atomic.StoreInt32(&b.opened, 0) + + if err == nil { + Logger.Printf("Closed connection to broker %s\n", b.addr) + } else { + Logger.Printf("Error while closing connection to broker %s: %s\n", b.addr, err) + } + + return err +} + +// ID returns the broker ID retrieved from Kafka's metadata, or -1 if that is not known. +func (b *Broker) ID() int32 { + return b.id +} + +// Addr returns the broker address as either retrieved from Kafka's metadata or passed to NewBroker. +func (b *Broker) Addr() string { + return b.addr +} + +func (b *Broker) GetMetadata(request *MetadataRequest) (*MetadataResponse, error) { + response := new(MetadataResponse) + + err := b.sendAndReceive(request, response) + + if err != nil { + return nil, err + } + + return response, nil +} + +func (b *Broker) GetConsumerMetadata(request *ConsumerMetadataRequest) (*ConsumerMetadataResponse, error) { + response := new(ConsumerMetadataResponse) + + err := b.sendAndReceive(request, response) + + if err != nil { + return nil, err + } + + return response, nil +} + +func (b *Broker) GetAvailableOffsets(request *OffsetRequest) (*OffsetResponse, error) { + response := new(OffsetResponse) + + err := b.sendAndReceive(request, response) + + if err != nil { + return nil, err + } + + return response, nil +} + +func (b *Broker) Produce(request *ProduceRequest) (*ProduceResponse, error) { + var response *ProduceResponse + var err error + + if request.RequiredAcks == NoResponse { + err = b.sendAndReceive(request, nil) + } else { + response = new(ProduceResponse) + err = b.sendAndReceive(request, response) + } + + if err != nil { + return nil, err + } + + return response, nil +} + +func (b *Broker) Fetch(request *FetchRequest) (*FetchResponse, error) { + response := new(FetchResponse) + + err := b.sendAndReceive(request, response) + + if err != nil { + return nil, err + } + + return response, nil +} + +func (b *Broker) CommitOffset(request *OffsetCommitRequest) (*OffsetCommitResponse, error) { + response := new(OffsetCommitResponse) + + err := b.sendAndReceive(request, response) + + if err != nil { + return nil, err + } + + return response, nil +} + +func (b *Broker) FetchOffset(request *OffsetFetchRequest) (*OffsetFetchResponse, error) { + response := new(OffsetFetchResponse) + + err := b.sendAndReceive(request, response) + + if err != nil { + return nil, err + } + + return response, nil +} + +func (b *Broker) send(rb requestBody, promiseResponse bool) (*responsePromise, error) { + b.lock.Lock() + defer b.lock.Unlock() + + if b.conn == nil { + if b.connErr != nil { + return nil, b.connErr + } + return nil, ErrNotConnected + } + + req := &request{correlationID: b.correlationID, clientID: b.conf.ClientID, body: rb} + buf, err := encode(req) + if err != nil { + return nil, err + } + + err = b.conn.SetWriteDeadline(time.Now().Add(b.conf.Net.WriteTimeout)) + if err != nil { + return nil, err + } + + _, err = b.conn.Write(buf) + if err != nil { + return nil, err + } + b.correlationID++ + + if !promiseResponse { + return nil, nil + } + + promise := responsePromise{req.correlationID, make(chan []byte), make(chan error)} + b.responses <- promise + + return &promise, nil +} + +func (b *Broker) sendAndReceive(req requestBody, res decoder) error { + promise, err := b.send(req, res != nil) + + if err != nil { + return err + } + + if promise == nil { + return nil + } + + select { + case buf := <-promise.packets: + return decode(buf, res) + case err = <-promise.errors: + return err + } +} + +func (b *Broker) decode(pd packetDecoder) (err error) { + b.id, err = pd.getInt32() + if err != nil { + return err + } + + host, err := pd.getString() + if err != nil { + return err + } + + port, err := pd.getInt32() + if err != nil { + return err + } + + b.addr = fmt.Sprint(host, ":", port) + + return nil +} + +func (b *Broker) encode(pe packetEncoder) (err error) { + + host, portstr, err := net.SplitHostPort(b.addr) + if err != nil { + return err + } + port, err := strconv.Atoi(portstr) + if err != nil { + return err + } + + pe.putInt32(b.id) + + err = pe.putString(host) + if err != nil { + return err + } + + pe.putInt32(int32(port)) + + return nil +} + +func (b *Broker) responseReceiver() { + header := make([]byte, 8) + for response := range b.responses { + err := b.conn.SetReadDeadline(time.Now().Add(b.conf.Net.ReadTimeout)) + if err != nil { + response.errors <- err + continue + } + + _, err = io.ReadFull(b.conn, header) + if err != nil { + response.errors <- err + continue + } + + decodedHeader := responseHeader{} + err = decode(header, &decodedHeader) + if err != nil { + response.errors <- err + continue + } + if decodedHeader.correlationID != response.correlationID { + // TODO if decoded ID < cur ID, discard until we catch up + // TODO if decoded ID > cur ID, save it so when cur ID catches up we have a response + response.errors <- PacketDecodingError{fmt.Sprintf("correlation ID didn't match, wanted %d, got %d", response.correlationID, decodedHeader.correlationID)} + continue + } + + buf := make([]byte, decodedHeader.length-4) + _, err = io.ReadFull(b.conn, buf) + if err != nil { + // XXX: the above ReadFull call inherits the same ReadDeadline set at the top of this loop, so it may + // fail with a timeout error. If this happens, our connection is permanently toast since we will no longer + // be aligned correctly on the stream (we'll be reading garbage Kafka headers from the middle of data). + // Can we/should we fail harder in that case? + response.errors <- err + continue + } + + response.packets <- buf + } + close(b.done) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/broker_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/broker_test.go new file mode 100644 index 0000000000000..df3499e49aeb7 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/broker_test.go @@ -0,0 +1,177 @@ +package sarama + +import ( + "fmt" + "testing" +) + +func ExampleBroker() error { + broker := NewBroker("localhost:9092") + err := broker.Open(nil) + if err != nil { + return err + } + + request := MetadataRequest{Topics: []string{"myTopic"}} + response, err := broker.GetMetadata(&request) + if err != nil { + _ = broker.Close() + return err + } + + fmt.Println("There are", len(response.Topics), "topics active in the cluster.") + + return broker.Close() +} + +type mockEncoder struct { + bytes []byte +} + +func (m mockEncoder) encode(pe packetEncoder) error { + return pe.putRawBytes(m.bytes) +} + +func TestBrokerAccessors(t *testing.T) { + broker := NewBroker("abc:123") + + if broker.ID() != -1 { + t.Error("New broker didn't have an ID of -1.") + } + + if broker.Addr() != "abc:123" { + t.Error("New broker didn't have the correct address") + } + + broker.id = 34 + if broker.ID() != 34 { + t.Error("Manually setting broker ID did not take effect.") + } +} + +func TestSimpleBrokerCommunication(t *testing.T) { + mb := newMockBroker(t, 0) + defer mb.Close() + + broker := NewBroker(mb.Addr()) + err := broker.Open(nil) + if err != nil { + t.Fatal(err) + } + + for _, tt := range brokerTestTable { + mb.Returns(&mockEncoder{tt.response}) + } + for _, tt := range brokerTestTable { + tt.runner(t, broker) + } + + err = broker.Close() + if err != nil { + t.Error(err) + } +} + +// We're not testing encoding/decoding here, so most of the requests/responses will be empty for simplicity's sake +var brokerTestTable = []struct { + response []byte + runner func(*testing.T, *Broker) +}{ + {[]byte{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, + func(t *testing.T, broker *Broker) { + request := MetadataRequest{} + response, err := broker.GetMetadata(&request) + if err != nil { + t.Error(err) + } + if response == nil { + t.Error("Metadata request got no response!") + } + }}, + + {[]byte{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 't', 0x00, 0x00, 0x00, 0x00}, + func(t *testing.T, broker *Broker) { + request := ConsumerMetadataRequest{} + response, err := broker.GetConsumerMetadata(&request) + if err != nil { + t.Error(err) + } + if response == nil { + t.Error("Consumer Metadata request got no response!") + } + }}, + + {[]byte{}, + func(t *testing.T, broker *Broker) { + request := ProduceRequest{} + request.RequiredAcks = NoResponse + response, err := broker.Produce(&request) + if err != nil { + t.Error(err) + } + if response != nil { + t.Error("Produce request with NoResponse got a response!") + } + }}, + + {[]byte{0x00, 0x00, 0x00, 0x00}, + func(t *testing.T, broker *Broker) { + request := ProduceRequest{} + request.RequiredAcks = WaitForLocal + response, err := broker.Produce(&request) + if err != nil { + t.Error(err) + } + if response == nil { + t.Error("Produce request without NoResponse got no response!") + } + }}, + + {[]byte{0x00, 0x00, 0x00, 0x00}, + func(t *testing.T, broker *Broker) { + request := FetchRequest{} + response, err := broker.Fetch(&request) + if err != nil { + t.Error(err) + } + if response == nil { + t.Error("Fetch request got no response!") + } + }}, + + {[]byte{0x00, 0x00, 0x00, 0x00}, + func(t *testing.T, broker *Broker) { + request := OffsetFetchRequest{} + response, err := broker.FetchOffset(&request) + if err != nil { + t.Error(err) + } + if response == nil { + t.Error("OffsetFetch request got no response!") + } + }}, + + {[]byte{0x00, 0x00, 0x00, 0x00}, + func(t *testing.T, broker *Broker) { + request := OffsetCommitRequest{} + response, err := broker.CommitOffset(&request) + if err != nil { + t.Error(err) + } + if response == nil { + t.Error("OffsetCommit request got no response!") + } + }}, + + {[]byte{0x00, 0x00, 0x00, 0x00}, + func(t *testing.T, broker *Broker) { + request := OffsetRequest{} + response, err := broker.GetAvailableOffsets(&request) + if err != nil { + t.Error(err) + } + if response == nil { + t.Error("Offset request got no response!") + } + }}, +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/client.go b/Godeps/_workspace/src/github.com/Shopify/sarama/client.go new file mode 100644 index 0000000000000..974d223e25288 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/client.go @@ -0,0 +1,727 @@ +package sarama + +import ( + "math/rand" + "sort" + "sync" + "time" +) + +// Client is a generic Kafka client. It manages connections to one or more Kafka brokers. +// You MUST call Close() on a client to avoid leaks, it will not be garbage-collected +// automatically when it passes out of scope. A single client can be safely shared by +// multiple concurrent Producers and Consumers. +type Client interface { + // Config returns the Config struct of the client. This struct should not be altered after it + // has been created. + Config() *Config + + // Topics returns the set of available topics as retrieved from the cluster metadata. + Topics() ([]string, error) + + // Partitions returns the sorted list of all partition IDs for the given topic. + Partitions(topic string) ([]int32, error) + + // WritablePartitions returns the sorted list of all writable partition IDs for the given topic, + // where "writable" means "having a valid leader accepting writes". + WritablePartitions(topic string) ([]int32, error) + + // Leader returns the broker object that is the leader of the current topic/partition, as + // determined by querying the cluster metadata. + Leader(topic string, partitionID int32) (*Broker, error) + + // Replicas returns the set of all replica IDs for the given partition. + Replicas(topic string, partitionID int32) ([]int32, error) + + // RefreshMetadata takes a list of topics and queries the cluster to refresh the + // available metadata for those topics. If no topics are provided, it will refresh metadata + // for all topics. + RefreshMetadata(topics ...string) error + + // GetOffset queries the cluster to get the most recent available offset at the given + // time on the topic/partition combination. Time should be OffsetOldest for the earliest available + // offset, OffsetNewest for the offset of the message that will be produced next, or a time. + GetOffset(topic string, partitionID int32, time int64) (int64, error) + + // Coordinator returns the coordinating broker for a consumer group. It will return a locally cached + // value if it's available. You can call RefreshCoordinator to update the cached value. + // This function only works on Kafka 0.8.2 and higher. + Coordinator(consumerGroup string) (*Broker, error) + + // RefreshCoordinator retrieves the coordinator for a consumer group and stores it in local cache. + // This function only works on Kafka 0.8.2 and higher. + RefreshCoordinator(consumerGroup string) error + + // Close shuts down all broker connections managed by this client. It is required to call this function before + // a client object passes out of scope, as it will otherwise leak memory. You must close any Producers or Consumers + // using a client before you close the client. + Close() error + + // Closed returns true if the client has already had Close called on it + Closed() bool +} + +const ( + // OffsetNewest stands for the log head offset, i.e. the offset that will be assigned to the next message + // that will be produced to the partition. You can send this to a client's GetOffset method to get this + // offset, or when calling ConsumePartition to start consuming new messages. + OffsetNewest int64 = -1 + // OffsetOldest stands for the oldest offset available on the broker for a partition. You can send this + // to a client's GetOffset method to get this offset, or when calling ConsumePartition to start consuming + // from the oldest offset that is still available on the broker. + OffsetOldest int64 = -2 +) + +type client struct { + conf *Config + closer, closed chan none // for shutting down background metadata updater + + // the broker addresses given to us through the constructor are not guaranteed to be returned in + // the cluster metadata (I *think* it only returns brokers who are currently leading partitions?) + // so we store them separately + seedBrokers []*Broker + deadSeeds []*Broker + + brokers map[int32]*Broker // maps broker ids to brokers + metadata map[string]map[int32]*PartitionMetadata // maps topics to partition ids to metadata + coordinators map[string]int32 // Maps consumer group names to coordinating broker IDs + + // If the number of partitions is large, we can get some churn calling cachedPartitions, + // so the result is cached. It is important to update this value whenever metadata is changed + cachedPartitionsResults map[string][maxPartitionIndex][]int32 + + lock sync.RWMutex // protects access to the maps that hold cluster state. +} + +// NewClient creates a new Client. It connects to one of the given broker addresses +// and uses that broker to automatically fetch metadata on the rest of the kafka cluster. If metadata cannot +// be retrieved from any of the given broker addresses, the client is not created. +func NewClient(addrs []string, conf *Config) (Client, error) { + Logger.Println("Initializing new client") + + if conf == nil { + conf = NewConfig() + } + + if err := conf.Validate(); err != nil { + return nil, err + } + + if len(addrs) < 1 { + return nil, ConfigurationError("You must provide at least one broker address") + } + + client := &client{ + conf: conf, + closer: make(chan none), + closed: make(chan none), + brokers: make(map[int32]*Broker), + metadata: make(map[string]map[int32]*PartitionMetadata), + cachedPartitionsResults: make(map[string][maxPartitionIndex][]int32), + coordinators: make(map[string]int32), + } + + random := rand.New(rand.NewSource(time.Now().UnixNano())) + for _, index := range random.Perm(len(addrs)) { + client.seedBrokers = append(client.seedBrokers, NewBroker(addrs[index])) + } + + // do an initial fetch of all cluster metadata by specifing an empty list of topics + err := client.RefreshMetadata() + switch err { + case nil: + break + case ErrLeaderNotAvailable, ErrReplicaNotAvailable: + // indicates that maybe part of the cluster is down, but is not fatal to creating the client + Logger.Println(err) + default: + close(client.closed) // we haven't started the background updater yet, so we have to do this manually + _ = client.Close() + return nil, err + } + go withRecover(client.backgroundMetadataUpdater) + + Logger.Println("Successfully initialized new client") + + return client, nil +} + +func (client *client) Config() *Config { + return client.conf +} + +func (client *client) Close() error { + if client.Closed() { + // Chances are this is being called from a defer() and the error will go unobserved + // so we go ahead and log the event in this case. + Logger.Printf("Close() called on already closed client") + return ErrClosedClient + } + + // shutdown and wait for the background thread before we take the lock, to avoid races + close(client.closer) + <-client.closed + + client.lock.Lock() + defer client.lock.Unlock() + Logger.Println("Closing Client") + + for _, broker := range client.brokers { + safeAsyncClose(broker) + } + + for _, broker := range client.seedBrokers { + safeAsyncClose(broker) + } + + client.brokers = nil + client.metadata = nil + + return nil +} + +func (client *client) Closed() bool { + return client.brokers == nil +} + +func (client *client) Topics() ([]string, error) { + if client.Closed() { + return nil, ErrClosedClient + } + + client.lock.RLock() + defer client.lock.RUnlock() + + ret := make([]string, 0, len(client.metadata)) + for topic := range client.metadata { + ret = append(ret, topic) + } + + return ret, nil +} + +func (client *client) Partitions(topic string) ([]int32, error) { + if client.Closed() { + return nil, ErrClosedClient + } + + partitions := client.cachedPartitions(topic, allPartitions) + + if len(partitions) == 0 { + err := client.RefreshMetadata(topic) + if err != nil { + return nil, err + } + partitions = client.cachedPartitions(topic, allPartitions) + } + + if partitions == nil { + return nil, ErrUnknownTopicOrPartition + } + + return partitions, nil +} + +func (client *client) WritablePartitions(topic string) ([]int32, error) { + if client.Closed() { + return nil, ErrClosedClient + } + + partitions := client.cachedPartitions(topic, writablePartitions) + + // len==0 catches when it's nil (no such topic) and the odd case when every single + // partition is undergoing leader election simultaneously. Callers have to be able to handle + // this function returning an empty slice (which is a valid return value) but catching it + // here the first time (note we *don't* catch it below where we return ErrUnknownTopicOrPartition) triggers + // a metadata refresh as a nicety so callers can just try again and don't have to manually + // trigger a refresh (otherwise they'd just keep getting a stale cached copy). + if len(partitions) == 0 { + err := client.RefreshMetadata(topic) + if err != nil { + return nil, err + } + partitions = client.cachedPartitions(topic, writablePartitions) + } + + if partitions == nil { + return nil, ErrUnknownTopicOrPartition + } + + return partitions, nil +} + +func (client *client) Replicas(topic string, partitionID int32) ([]int32, error) { + if client.Closed() { + return nil, ErrClosedClient + } + + metadata := client.cachedMetadata(topic, partitionID) + + if metadata == nil { + err := client.RefreshMetadata(topic) + if err != nil { + return nil, err + } + metadata = client.cachedMetadata(topic, partitionID) + } + + if metadata == nil { + return nil, ErrUnknownTopicOrPartition + } + + if metadata.Err == ErrReplicaNotAvailable { + return nil, metadata.Err + } + return dupeAndSort(metadata.Replicas), nil +} + +func (client *client) Leader(topic string, partitionID int32) (*Broker, error) { + if client.Closed() { + return nil, ErrClosedClient + } + + leader, err := client.cachedLeader(topic, partitionID) + + if leader == nil { + err := client.RefreshMetadata(topic) + if err != nil { + return nil, err + } + leader, err = client.cachedLeader(topic, partitionID) + } + + return leader, err +} + +func (client *client) RefreshMetadata(topics ...string) error { + if client.Closed() { + return ErrClosedClient + } + + // Prior to 0.8.2, Kafka will throw exceptions on an empty topic and not return a proper + // error. This handles the case by returning an error instead of sending it + // off to Kafka. See: https://github.com/Shopify/sarama/pull/38#issuecomment-26362310 + for _, topic := range topics { + if len(topic) == 0 { + return ErrInvalidTopic // this is the error that 0.8.2 and later correctly return + } + } + + return client.tryRefreshMetadata(topics, client.conf.Metadata.Retry.Max) +} + +func (client *client) GetOffset(topic string, partitionID int32, time int64) (int64, error) { + if client.Closed() { + return -1, ErrClosedClient + } + + offset, err := client.getOffset(topic, partitionID, time) + + if err != nil { + if err := client.RefreshMetadata(topic); err != nil { + return -1, err + } + return client.getOffset(topic, partitionID, time) + } + + return offset, err +} + +func (client *client) Coordinator(consumerGroup string) (*Broker, error) { + if client.Closed() { + return nil, ErrClosedClient + } + + coordinator := client.cachedCoordinator(consumerGroup) + + if coordinator == nil { + if err := client.RefreshCoordinator(consumerGroup); err != nil { + return nil, err + } + coordinator = client.cachedCoordinator(consumerGroup) + } + + if coordinator == nil { + return nil, ErrConsumerCoordinatorNotAvailable + } + + _ = coordinator.Open(client.conf) + return coordinator, nil +} + +func (client *client) RefreshCoordinator(consumerGroup string) error { + if client.Closed() { + return ErrClosedClient + } + + response, err := client.getConsumerMetadata(consumerGroup, client.conf.Metadata.Retry.Max) + if err != nil { + return err + } + + client.lock.Lock() + defer client.lock.Unlock() + client.registerBroker(response.Coordinator) + client.coordinators[consumerGroup] = response.Coordinator.ID() + return nil +} + +// private broker management helpers + +// registerBroker makes sure a broker received by a Metadata or Coordinator request is registered +// in the brokers map. It returns the broker that is registered, which may be the provided broker, +// or a previously registered Broker instance. You must hold the write lock before calling this function. +func (client *client) registerBroker(broker *Broker) { + if client.brokers[broker.ID()] == nil { + client.brokers[broker.ID()] = broker + Logger.Printf("client/brokers registered new broker #%d at %s", broker.ID(), broker.Addr()) + } else if broker.Addr() != client.brokers[broker.ID()].Addr() { + safeAsyncClose(client.brokers[broker.ID()]) + client.brokers[broker.ID()] = broker + Logger.Printf("client/brokers replaced registered broker #%d with %s", broker.ID(), broker.Addr()) + } +} + +// deregisterBroker removes a broker from the seedsBroker list, and if it's +// not the seedbroker, removes it from brokers map completely. +func (client *client) deregisterBroker(broker *Broker) { + client.lock.Lock() + defer client.lock.Unlock() + + if len(client.seedBrokers) > 0 && broker == client.seedBrokers[0] { + client.deadSeeds = append(client.deadSeeds, broker) + client.seedBrokers = client.seedBrokers[1:] + } else { + // we do this so that our loop in `tryRefreshMetadata` doesn't go on forever, + // but we really shouldn't have to; once that loop is made better this case can be + // removed, and the function generally can be renamed from `deregisterBroker` to + // `nextSeedBroker` or something + Logger.Printf("client/brokers deregistered broker #%d at %s", broker.ID(), broker.Addr()) + delete(client.brokers, broker.ID()) + } +} + +func (client *client) resurrectDeadBrokers() { + client.lock.Lock() + defer client.lock.Unlock() + + Logger.Printf("client/brokers resurrecting %d dead seed brokers", len(client.deadSeeds)) + client.seedBrokers = append(client.seedBrokers, client.deadSeeds...) + client.deadSeeds = nil +} + +func (client *client) any() *Broker { + client.lock.RLock() + defer client.lock.RUnlock() + + if len(client.seedBrokers) > 0 { + _ = client.seedBrokers[0].Open(client.conf) + return client.seedBrokers[0] + } + + // not guaranteed to be random *or* deterministic + for _, broker := range client.brokers { + _ = broker.Open(client.conf) + return broker + } + + return nil +} + +// private caching/lazy metadata helpers + +type partitionType int + +const ( + allPartitions partitionType = iota + writablePartitions + // If you add any more types, update the partition cache in update() + + // Ensure this is the last partition type value + maxPartitionIndex +) + +func (client *client) cachedMetadata(topic string, partitionID int32) *PartitionMetadata { + client.lock.RLock() + defer client.lock.RUnlock() + + partitions := client.metadata[topic] + if partitions != nil { + return partitions[partitionID] + } + + return nil +} + +func (client *client) cachedPartitions(topic string, partitionSet partitionType) []int32 { + client.lock.RLock() + defer client.lock.RUnlock() + + partitions, exists := client.cachedPartitionsResults[topic] + + if !exists { + return nil + } + return partitions[partitionSet] +} + +func (client *client) setPartitionCache(topic string, partitionSet partitionType) []int32 { + partitions := client.metadata[topic] + + if partitions == nil { + return nil + } + + ret := make([]int32, 0, len(partitions)) + for _, partition := range partitions { + if partitionSet == writablePartitions && partition.Err == ErrLeaderNotAvailable { + continue + } + ret = append(ret, partition.ID) + } + + sort.Sort(int32Slice(ret)) + return ret +} + +func (client *client) cachedLeader(topic string, partitionID int32) (*Broker, error) { + client.lock.RLock() + defer client.lock.RUnlock() + + partitions := client.metadata[topic] + if partitions != nil { + metadata, ok := partitions[partitionID] + if ok { + if metadata.Err == ErrLeaderNotAvailable { + return nil, ErrLeaderNotAvailable + } + b := client.brokers[metadata.Leader] + if b == nil { + return nil, ErrLeaderNotAvailable + } + _ = b.Open(client.conf) + return b, nil + } + } + + return nil, ErrUnknownTopicOrPartition +} + +func (client *client) getOffset(topic string, partitionID int32, time int64) (int64, error) { + broker, err := client.Leader(topic, partitionID) + if err != nil { + return -1, err + } + + request := &OffsetRequest{} + request.AddBlock(topic, partitionID, time, 1) + + response, err := broker.GetAvailableOffsets(request) + if err != nil { + _ = broker.Close() + return -1, err + } + + block := response.GetBlock(topic, partitionID) + if block == nil { + _ = broker.Close() + return -1, ErrIncompleteResponse + } + if block.Err != ErrNoError { + return -1, block.Err + } + if len(block.Offsets) != 1 { + return -1, ErrOffsetOutOfRange + } + + return block.Offsets[0], nil +} + +// core metadata update logic + +func (client *client) backgroundMetadataUpdater() { + defer close(client.closed) + + if client.conf.Metadata.RefreshFrequency == time.Duration(0) { + return + } + + ticker := time.NewTicker(client.conf.Metadata.RefreshFrequency) + defer ticker.Stop() + + for { + select { + case <-ticker.C: + if err := client.RefreshMetadata(); err != nil { + Logger.Println("Client background metadata update:", err) + } + case <-client.closer: + return + } + } +} + +func (client *client) tryRefreshMetadata(topics []string, attemptsRemaining int) error { + retry := func(err error) error { + if attemptsRemaining > 0 { + Logger.Printf("client/metadata retrying after %dms... (%d attempts remaining)\n", client.conf.Metadata.Retry.Backoff/time.Millisecond, attemptsRemaining) + time.Sleep(client.conf.Metadata.Retry.Backoff) + return client.tryRefreshMetadata(topics, attemptsRemaining-1) + } + return err + } + + for broker := client.any(); broker != nil; broker = client.any() { + if len(topics) > 0 { + Logger.Printf("client/metadata fetching metadata for %v from broker %s\n", topics, broker.addr) + } else { + Logger.Printf("client/metadata fetching metadata for all topics from broker %s\n", broker.addr) + } + response, err := broker.GetMetadata(&MetadataRequest{Topics: topics}) + + switch err.(type) { + case nil: + // valid response, use it + if shouldRetry, err := client.updateMetadata(response); shouldRetry { + Logger.Println("client/metadata found some partitions to be leaderless") + return retry(err) // note: err can be nil + } else { + return err + } + + case PacketEncodingError: + // didn't even send, return the error + return err + default: + // some other error, remove that broker and try again + Logger.Println("client/metadata got error from broker while fetching metadata:", err) + _ = broker.Close() + client.deregisterBroker(broker) + } + } + + Logger.Println("client/metadata no available broker to send metadata request to") + client.resurrectDeadBrokers() + return retry(ErrOutOfBrokers) +} + +// if no fatal error, returns a list of topics that need retrying due to ErrLeaderNotAvailable +func (client *client) updateMetadata(data *MetadataResponse) (retry bool, err error) { + client.lock.Lock() + defer client.lock.Unlock() + + // For all the brokers we received: + // - if it is a new ID, save it + // - if it is an existing ID, but the address we have is stale, discard the old one and save it + // - otherwise ignore it, replacing our existing one would just bounce the connection + for _, broker := range data.Brokers { + client.registerBroker(broker) + } + + for _, topic := range data.Topics { + delete(client.metadata, topic.Name) + delete(client.cachedPartitionsResults, topic.Name) + + switch topic.Err { + case ErrNoError: + break + case ErrInvalidTopic: // don't retry, don't store partial results + err = topic.Err + continue + case ErrUnknownTopicOrPartition: // retry, do not store partial partition results + err = topic.Err + retry = true + continue + case ErrLeaderNotAvailable: // retry, but store partial partition results + retry = true + break + default: // don't retry, don't store partial results + Logger.Printf("Unexpected topic-level metadata error: %s", topic.Err) + err = topic.Err + continue + } + + client.metadata[topic.Name] = make(map[int32]*PartitionMetadata, len(topic.Partitions)) + for _, partition := range topic.Partitions { + client.metadata[topic.Name][partition.ID] = partition + if partition.Err == ErrLeaderNotAvailable { + retry = true + } + } + + var partitionCache [maxPartitionIndex][]int32 + partitionCache[allPartitions] = client.setPartitionCache(topic.Name, allPartitions) + partitionCache[writablePartitions] = client.setPartitionCache(topic.Name, writablePartitions) + client.cachedPartitionsResults[topic.Name] = partitionCache + } + + return +} + +func (client *client) cachedCoordinator(consumerGroup string) *Broker { + client.lock.RLock() + defer client.lock.RUnlock() + if coordinatorID, ok := client.coordinators[consumerGroup]; !ok { + return nil + } else { + return client.brokers[coordinatorID] + } +} + +func (client *client) getConsumerMetadata(consumerGroup string, attemptsRemaining int) (*ConsumerMetadataResponse, error) { + retry := func(err error) (*ConsumerMetadataResponse, error) { + if attemptsRemaining > 0 { + Logger.Printf("client/coordinator retrying after %dms... (%d attempts remaining)\n", client.conf.Metadata.Retry.Backoff/time.Millisecond, attemptsRemaining) + time.Sleep(client.conf.Metadata.Retry.Backoff) + return client.getConsumerMetadata(consumerGroup, attemptsRemaining-1) + } + return nil, err + } + + for broker := client.any(); broker != nil; broker = client.any() { + Logger.Printf("client/coordinator requesting coordinator for consumergoup %s from %s\n", consumerGroup, broker.Addr()) + + request := new(ConsumerMetadataRequest) + request.ConsumerGroup = consumerGroup + + response, err := broker.GetConsumerMetadata(request) + + if err != nil { + Logger.Printf("client/coordinator request to broker %s failed: %s\n", broker.Addr(), err) + + switch err.(type) { + case PacketEncodingError: + return nil, err + default: + _ = broker.Close() + client.deregisterBroker(broker) + continue + } + } + + switch response.Err { + case ErrNoError: + Logger.Printf("client/coordinator coordinator for consumergoup %s is #%d (%s)\n", consumerGroup, response.Coordinator.ID(), response.Coordinator.Addr()) + return response, nil + + case ErrConsumerCoordinatorNotAvailable: + Logger.Printf("client/coordinator coordinator for consumer group %s is not available\n", consumerGroup) + + // This is very ugly, but this scenario will only happen once per cluster. + // The __consumer_offsets topic only has to be created one time. + // The number of partitions not configurable, but partition 0 should always exist. + if _, err := client.Leader("__consumer_offsets", 0); err != nil { + Logger.Printf("client/coordinator the __consumer_offsets topic is not initialized completely yet. Waiting 2 seconds...\n") + time.Sleep(2 * time.Second) + } + + return retry(ErrConsumerCoordinatorNotAvailable) + default: + return nil, response.Err + } + } + + Logger.Println("client/coordinator no available broker to send consumer metadata request to") + client.resurrectDeadBrokers() + return retry(ErrOutOfBrokers) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/client_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/client_test.go new file mode 100644 index 0000000000000..f84b9af31a68c --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/client_test.go @@ -0,0 +1,608 @@ +package sarama + +import ( + "io" + "sync" + "testing" + "time" +) + +func safeClose(t testing.TB, c io.Closer) { + err := c.Close() + if err != nil { + t.Error(err) + } +} + +func TestSimpleClient(t *testing.T) { + seedBroker := newMockBroker(t, 1) + + seedBroker.Returns(new(MetadataResponse)) + + client, err := NewClient([]string{seedBroker.Addr()}, nil) + if err != nil { + t.Fatal(err) + } + + seedBroker.Close() + safeClose(t, client) +} + +func TestCachedPartitions(t *testing.T) { + seedBroker := newMockBroker(t, 1) + + replicas := []int32{3, 1, 5} + isr := []int32{5, 1} + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker("localhost:12345", 2) + metadataResponse.AddTopicPartition("my_topic", 0, 2, replicas, isr, ErrNoError) + metadataResponse.AddTopicPartition("my_topic", 1, 2, replicas, isr, ErrLeaderNotAvailable) + seedBroker.Returns(metadataResponse) + + config := NewConfig() + config.Metadata.Retry.Max = 0 + c, err := NewClient([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + client := c.(*client) + + // Verify they aren't cached the same + allP := client.cachedPartitionsResults["my_topic"][allPartitions] + writeP := client.cachedPartitionsResults["my_topic"][writablePartitions] + if len(allP) == len(writeP) { + t.Fatal("Invalid lengths!") + } + + tmp := client.cachedPartitionsResults["my_topic"] + // Verify we actually use the cache at all! + tmp[allPartitions] = []int32{1, 2, 3, 4} + client.cachedPartitionsResults["my_topic"] = tmp + if 4 != len(client.cachedPartitions("my_topic", allPartitions)) { + t.Fatal("Not using the cache!") + } + + seedBroker.Close() + safeClose(t, client) +} + +func TestClientDoesntCachePartitionsForTopicsWithErrors(t *testing.T) { + seedBroker := newMockBroker(t, 1) + + replicas := []int32{seedBroker.BrokerID()} + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker(seedBroker.Addr(), seedBroker.BrokerID()) + metadataResponse.AddTopicPartition("my_topic", 1, replicas[0], replicas, replicas, ErrNoError) + metadataResponse.AddTopicPartition("my_topic", 2, replicas[0], replicas, replicas, ErrNoError) + seedBroker.Returns(metadataResponse) + + config := NewConfig() + config.Metadata.Retry.Max = 0 + client, err := NewClient([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + metadataResponse = new(MetadataResponse) + metadataResponse.AddTopic("unknown", ErrUnknownTopicOrPartition) + seedBroker.Returns(metadataResponse) + + partitions, err := client.Partitions("unknown") + + if err != ErrUnknownTopicOrPartition { + t.Error("Expected ErrUnknownTopicOrPartition, found", err) + } + if partitions != nil { + t.Errorf("Should return nil as partition list, found %v", partitions) + } + + // Should still use the cache of a known topic + partitions, err = client.Partitions("my_topic") + if err != nil { + t.Errorf("Expected no error, found %v", err) + } + + metadataResponse = new(MetadataResponse) + metadataResponse.AddTopic("unknown", ErrUnknownTopicOrPartition) + seedBroker.Returns(metadataResponse) + + // Should not use cache for unknown topic + partitions, err = client.Partitions("unknown") + if err != ErrUnknownTopicOrPartition { + t.Error("Expected ErrUnknownTopicOrPartition, found", err) + } + if partitions != nil { + t.Errorf("Should return nil as partition list, found %v", partitions) + } + + seedBroker.Close() + safeClose(t, client) +} + +func TestClientSeedBrokers(t *testing.T) { + seedBroker := newMockBroker(t, 1) + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker("localhost:12345", 2) + seedBroker.Returns(metadataResponse) + + client, err := NewClient([]string{seedBroker.Addr()}, nil) + if err != nil { + t.Fatal(err) + } + + seedBroker.Close() + safeClose(t, client) +} + +func TestClientMetadata(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader := newMockBroker(t, 5) + + replicas := []int32{3, 1, 5} + isr := []int32{5, 1} + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker(leader.Addr(), leader.BrokerID()) + metadataResponse.AddTopicPartition("my_topic", 0, leader.BrokerID(), replicas, isr, ErrNoError) + metadataResponse.AddTopicPartition("my_topic", 1, leader.BrokerID(), replicas, isr, ErrLeaderNotAvailable) + seedBroker.Returns(metadataResponse) + + config := NewConfig() + config.Metadata.Retry.Max = 0 + client, err := NewClient([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + topics, err := client.Topics() + if err != nil { + t.Error(err) + } else if len(topics) != 1 || topics[0] != "my_topic" { + t.Error("Client returned incorrect topics:", topics) + } + + parts, err := client.Partitions("my_topic") + if err != nil { + t.Error(err) + } else if len(parts) != 2 || parts[0] != 0 || parts[1] != 1 { + t.Error("Client returned incorrect partitions for my_topic:", parts) + } + + parts, err = client.WritablePartitions("my_topic") + if err != nil { + t.Error(err) + } else if len(parts) != 1 || parts[0] != 0 { + t.Error("Client returned incorrect writable partitions for my_topic:", parts) + } + + tst, err := client.Leader("my_topic", 0) + if err != nil { + t.Error(err) + } else if tst.ID() != 5 { + t.Error("Leader for my_topic had incorrect ID.") + } + + replicas, err = client.Replicas("my_topic", 0) + if err != nil { + t.Error(err) + } else if replicas[0] != 1 { + t.Error("Incorrect (or unsorted) replica") + } else if replicas[1] != 3 { + t.Error("Incorrect (or unsorted) replica") + } else if replicas[2] != 5 { + t.Error("Incorrect (or unsorted) replica") + } + + leader.Close() + seedBroker.Close() + safeClose(t, client) +} + +func TestClientGetOffset(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader := newMockBroker(t, 2) + leaderAddr := leader.Addr() + + metadata := new(MetadataResponse) + metadata.AddTopicPartition("foo", 0, leader.BrokerID(), nil, nil, ErrNoError) + metadata.AddBroker(leaderAddr, leader.BrokerID()) + seedBroker.Returns(metadata) + + client, err := NewClient([]string{seedBroker.Addr()}, nil) + if err != nil { + t.Fatal(err) + } + + offsetResponse := new(OffsetResponse) + offsetResponse.AddTopicPartition("foo", 0, 123) + leader.Returns(offsetResponse) + + offset, err := client.GetOffset("foo", 0, OffsetNewest) + if err != nil { + t.Error(err) + } + if offset != 123 { + t.Error("Unexpected offset, got ", offset) + } + + leader.Close() + seedBroker.Returns(metadata) + + leader = newMockBrokerAddr(t, 2, leaderAddr) + offsetResponse = new(OffsetResponse) + offsetResponse.AddTopicPartition("foo", 0, 456) + leader.Returns(offsetResponse) + + offset, err = client.GetOffset("foo", 0, OffsetNewest) + if err != nil { + t.Error(err) + } + if offset != 456 { + t.Error("Unexpected offset, got ", offset) + } + + seedBroker.Close() + leader.Close() + safeClose(t, client) +} + +func TestClientReceivingUnknownTopic(t *testing.T) { + seedBroker := newMockBroker(t, 1) + + metadataResponse1 := new(MetadataResponse) + seedBroker.Returns(metadataResponse1) + + config := NewConfig() + config.Metadata.Retry.Max = 1 + config.Metadata.Retry.Backoff = 0 + client, err := NewClient([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + metadataUnknownTopic := new(MetadataResponse) + metadataUnknownTopic.AddTopic("new_topic", ErrUnknownTopicOrPartition) + seedBroker.Returns(metadataUnknownTopic) + seedBroker.Returns(metadataUnknownTopic) + + if err := client.RefreshMetadata("new_topic"); err != ErrUnknownTopicOrPartition { + t.Error("ErrUnknownTopicOrPartition expected, got", err) + } + + // If we are asking for the leader of a partition of the non-existing topic. + // we will request metadata again. + seedBroker.Returns(metadataUnknownTopic) + seedBroker.Returns(metadataUnknownTopic) + + if _, err = client.Leader("new_topic", 1); err != ErrUnknownTopicOrPartition { + t.Error("Expected ErrUnknownTopicOrPartition, got", err) + } + + safeClose(t, client) + seedBroker.Close() +} + +func TestClientReceivingPartialMetadata(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader := newMockBroker(t, 5) + + metadataResponse1 := new(MetadataResponse) + metadataResponse1.AddBroker(leader.Addr(), leader.BrokerID()) + seedBroker.Returns(metadataResponse1) + + config := NewConfig() + config.Metadata.Retry.Max = 0 + client, err := NewClient([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + replicas := []int32{leader.BrokerID(), seedBroker.BrokerID()} + + metadataPartial := new(MetadataResponse) + metadataPartial.AddTopic("new_topic", ErrLeaderNotAvailable) + metadataPartial.AddTopicPartition("new_topic", 0, leader.BrokerID(), replicas, replicas, ErrNoError) + metadataPartial.AddTopicPartition("new_topic", 1, -1, replicas, []int32{}, ErrLeaderNotAvailable) + seedBroker.Returns(metadataPartial) + + if err := client.RefreshMetadata("new_topic"); err != nil { + t.Error("ErrLeaderNotAvailable should not make RefreshMetadata respond with an error") + } + + // Even though the metadata was incomplete, we should be able to get the leader of a partition + // for which we did get a useful response, without doing additional requests. + + partition0Leader, err := client.Leader("new_topic", 0) + if err != nil { + t.Error(err) + } else if partition0Leader.Addr() != leader.Addr() { + t.Error("Unexpected leader returned", partition0Leader.Addr()) + } + + // If we are asking for the leader of a partition that didn't have a leader before, + // we will do another metadata request. + + seedBroker.Returns(metadataPartial) + + // Still no leader for the partition, so asking for it should return an error. + _, err = client.Leader("new_topic", 1) + if err != ErrLeaderNotAvailable { + t.Error("Expected ErrLeaderNotAvailable, got", err) + } + + safeClose(t, client) + seedBroker.Close() + leader.Close() +} + +func TestClientRefreshBehaviour(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader := newMockBroker(t, 5) + + metadataResponse1 := new(MetadataResponse) + metadataResponse1.AddBroker(leader.Addr(), leader.BrokerID()) + seedBroker.Returns(metadataResponse1) + + metadataResponse2 := new(MetadataResponse) + metadataResponse2.AddTopicPartition("my_topic", 0xb, leader.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataResponse2) + + client, err := NewClient([]string{seedBroker.Addr()}, nil) + if err != nil { + t.Fatal(err) + } + + parts, err := client.Partitions("my_topic") + if err != nil { + t.Error(err) + } else if len(parts) != 1 || parts[0] != 0xb { + t.Error("Client returned incorrect partitions for my_topic:", parts) + } + + tst, err := client.Leader("my_topic", 0xb) + if err != nil { + t.Error(err) + } else if tst.ID() != 5 { + t.Error("Leader for my_topic had incorrect ID.") + } + + leader.Close() + seedBroker.Close() + safeClose(t, client) +} + +func TestClientResurrectDeadSeeds(t *testing.T) { + initialSeed := newMockBroker(t, 0) + emptyMetadata := new(MetadataResponse) + initialSeed.Returns(emptyMetadata) + + conf := NewConfig() + conf.Metadata.Retry.Backoff = 0 + conf.Metadata.RefreshFrequency = 0 + c, err := NewClient([]string{initialSeed.Addr()}, conf) + if err != nil { + t.Fatal(err) + } + initialSeed.Close() + + client := c.(*client) + + seed1 := newMockBroker(t, 1) + seed2 := newMockBroker(t, 2) + seed3 := newMockBroker(t, 3) + addr1 := seed1.Addr() + addr2 := seed2.Addr() + addr3 := seed3.Addr() + + // Overwrite the seed brokers with a fixed ordering to make this test deterministic. + safeClose(t, client.seedBrokers[0]) + client.seedBrokers = []*Broker{NewBroker(addr1), NewBroker(addr2), NewBroker(addr3)} + client.deadSeeds = []*Broker{} + + wg := sync.WaitGroup{} + wg.Add(1) + go func() { + if err := client.RefreshMetadata(); err != nil { + t.Error(err) + } + wg.Done() + }() + seed1.Close() + seed2.Close() + + seed1 = newMockBrokerAddr(t, 1, addr1) + seed2 = newMockBrokerAddr(t, 2, addr2) + + seed3.Close() + + seed1.Close() + seed2.Returns(emptyMetadata) + + wg.Wait() + + if len(client.seedBrokers) != 2 { + t.Error("incorrect number of live seeds") + } + if len(client.deadSeeds) != 1 { + t.Error("incorrect number of dead seeds") + } + + safeClose(t, c) +} + +func TestClientCoordinatorWithConsumerOffsetsTopic(t *testing.T) { + seedBroker := newMockBroker(t, 1) + staleCoordinator := newMockBroker(t, 2) + freshCoordinator := newMockBroker(t, 3) + + replicas := []int32{staleCoordinator.BrokerID(), freshCoordinator.BrokerID()} + metadataResponse1 := new(MetadataResponse) + metadataResponse1.AddBroker(staleCoordinator.Addr(), staleCoordinator.BrokerID()) + metadataResponse1.AddBroker(freshCoordinator.Addr(), freshCoordinator.BrokerID()) + metadataResponse1.AddTopicPartition("__consumer_offsets", 0, replicas[0], replicas, replicas, ErrNoError) + seedBroker.Returns(metadataResponse1) + + client, err := NewClient([]string{seedBroker.Addr()}, nil) + if err != nil { + t.Fatal(err) + } + + coordinatorResponse1 := new(ConsumerMetadataResponse) + coordinatorResponse1.Err = ErrConsumerCoordinatorNotAvailable + seedBroker.Returns(coordinatorResponse1) + + coordinatorResponse2 := new(ConsumerMetadataResponse) + coordinatorResponse2.CoordinatorID = staleCoordinator.BrokerID() + coordinatorResponse2.CoordinatorHost = "127.0.0.1" + coordinatorResponse2.CoordinatorPort = staleCoordinator.Port() + + seedBroker.Returns(coordinatorResponse2) + + broker, err := client.Coordinator("my_group") + if err != nil { + t.Error(err) + } + + if staleCoordinator.Addr() != broker.Addr() { + t.Errorf("Expected coordinator to have address %s, found %s", staleCoordinator.Addr(), broker.Addr()) + } + + if staleCoordinator.BrokerID() != broker.ID() { + t.Errorf("Expected coordinator to have ID %d, found %d", staleCoordinator.BrokerID(), broker.ID()) + } + + // Grab the cached value + broker2, err := client.Coordinator("my_group") + if err != nil { + t.Error(err) + } + + if broker2.Addr() != broker.Addr() { + t.Errorf("Expected the coordinator to be the same, but found %s vs. %s", broker2.Addr(), broker.Addr()) + } + + coordinatorResponse3 := new(ConsumerMetadataResponse) + coordinatorResponse3.CoordinatorID = freshCoordinator.BrokerID() + coordinatorResponse3.CoordinatorHost = "127.0.0.1" + coordinatorResponse3.CoordinatorPort = freshCoordinator.Port() + + seedBroker.Returns(coordinatorResponse3) + + // Refresh the locally cahced value because it's stale + if err := client.RefreshCoordinator("my_group"); err != nil { + t.Error(err) + } + + // Grab the fresh value + broker3, err := client.Coordinator("my_group") + if err != nil { + t.Error(err) + } + + if broker3.Addr() != freshCoordinator.Addr() { + t.Errorf("Expected the freshCoordinator to be returned, but found %s.", broker3.Addr()) + } + + freshCoordinator.Close() + staleCoordinator.Close() + seedBroker.Close() + safeClose(t, client) +} + +func TestClientCoordinatorWithoutConsumerOffsetsTopic(t *testing.T) { + seedBroker := newMockBroker(t, 1) + coordinator := newMockBroker(t, 2) + + metadataResponse1 := new(MetadataResponse) + seedBroker.Returns(metadataResponse1) + + config := NewConfig() + config.Metadata.Retry.Max = 1 + config.Metadata.Retry.Backoff = 0 + client, err := NewClient([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + coordinatorResponse1 := new(ConsumerMetadataResponse) + coordinatorResponse1.Err = ErrConsumerCoordinatorNotAvailable + seedBroker.Returns(coordinatorResponse1) + + metadataResponse2 := new(MetadataResponse) + metadataResponse2.AddTopic("__consumer_offsets", ErrUnknownTopicOrPartition) + seedBroker.Returns(metadataResponse2) + + replicas := []int32{coordinator.BrokerID()} + metadataResponse3 := new(MetadataResponse) + metadataResponse3.AddTopicPartition("__consumer_offsets", 0, replicas[0], replicas, replicas, ErrNoError) + seedBroker.Returns(metadataResponse3) + + coordinatorResponse2 := new(ConsumerMetadataResponse) + coordinatorResponse2.CoordinatorID = coordinator.BrokerID() + coordinatorResponse2.CoordinatorHost = "127.0.0.1" + coordinatorResponse2.CoordinatorPort = coordinator.Port() + + seedBroker.Returns(coordinatorResponse2) + + broker, err := client.Coordinator("my_group") + if err != nil { + t.Error(err) + } + + if coordinator.Addr() != broker.Addr() { + t.Errorf("Expected coordinator to have address %s, found %s", coordinator.Addr(), broker.Addr()) + } + + if coordinator.BrokerID() != broker.ID() { + t.Errorf("Expected coordinator to have ID %d, found %d", coordinator.BrokerID(), broker.ID()) + } + + coordinator.Close() + seedBroker.Close() + safeClose(t, client) +} + +func TestClientAutorefreshShutdownRace(t *testing.T) { + seedBroker := newMockBroker(t, 1) + + metadataResponse := new(MetadataResponse) + seedBroker.Returns(metadataResponse) + + conf := NewConfig() + conf.Metadata.RefreshFrequency = 100 * time.Millisecond + client, err := NewClient([]string{seedBroker.Addr()}, conf) + if err != nil { + t.Fatal(err) + } + + // Wait for the background refresh to kick in + time.Sleep(110 * time.Millisecond) + + done := make(chan none) + go func() { + // Close the client + if err := client.Close(); err != nil { + t.Fatal(err) + } + close(done) + }() + + // Wait for the Close to kick in + time.Sleep(10 * time.Millisecond) + + // Then return some metadata to the still-running background thread + leader := newMockBroker(t, 2) + metadataResponse.AddBroker(leader.Addr(), leader.BrokerID()) + metadataResponse.AddTopicPartition("foo", 0, leader.BrokerID(), []int32{2}, []int32{2}, ErrNoError) + seedBroker.Returns(metadataResponse) + + <-done + + seedBroker.Close() + + // give the update time to happen so we get a panic if it's still running (which it shouldn't) + time.Sleep(10 * time.Millisecond) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/config.go b/Godeps/_workspace/src/github.com/Shopify/sarama/config.go new file mode 100644 index 0000000000000..0fae111e9231b --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/config.go @@ -0,0 +1,275 @@ +package sarama + +import ( + "crypto/tls" + "time" +) + +// Config is used to pass multiple configuration options to Sarama's constructors. +type Config struct { + // Net is the namespace for network-level properties used by the Broker, and shared by the Client/Producer/Consumer. + Net struct { + MaxOpenRequests int // How many outstanding requests a connection is allowed to have before sending on it blocks (default 5). + + // All three of the below configurations are similar to the `socket.timeout.ms` setting in JVM kafka. + DialTimeout time.Duration // How long to wait for the initial connection to succeed before timing out and returning an error (default 30s). + ReadTimeout time.Duration // How long to wait for a response before timing out and returning an error (default 30s). + WriteTimeout time.Duration // How long to wait for a transmit to succeed before timing out and returning an error (default 30s). + + // NOTE: these config values have no compatibility guarantees; they may change when Kafka releases its + // official TLS support in version 0.9. + TLS struct { + Enable bool // Whether or not to use TLS when connecting to the broker (defaults to false). + Config *tls.Config // The TLS configuration to use for secure connections if enabled (defaults to nil). + } + + // KeepAlive specifies the keep-alive period for an active network connection. + // If zero, keep-alives are disabled. (default is 0: disabled). + KeepAlive time.Duration + } + + // Metadata is the namespace for metadata management properties used by the Client, and shared by the Producer/Consumer. + Metadata struct { + Retry struct { + Max int // The total number of times to retry a metadata request when the cluster is in the middle of a leader election (default 3). + Backoff time.Duration // How long to wait for leader election to occur before retrying (default 250ms). Similar to the JVM's `retry.backoff.ms`. + } + // How frequently to refresh the cluster metadata in the background. Defaults to 10 minutes. + // Set to 0 to disable. Similar to `topic.metadata.refresh.interval.ms` in the JVM version. + RefreshFrequency time.Duration + } + + // Producer is the namespace for configuration related to producing messages, used by the Producer. + Producer struct { + // The maximum permitted size of a message (defaults to 1000000). Should be set equal to or smaller than the broker's `message.max.bytes`. + MaxMessageBytes int + // The level of acknowledgement reliability needed from the broker (defaults to WaitForLocal). + // Equivalent to the `request.required.acks` setting of the JVM producer. + RequiredAcks RequiredAcks + // The maximum duration the broker will wait the receipt of the number of RequiredAcks (defaults to 10 seconds). + // This is only relevant when RequiredAcks is set to WaitForAll or a number > 1. Only supports millisecond resolution, + // nanoseconds will be truncated. Equivalent to the JVM producer's `request.timeout.ms` setting. + Timeout time.Duration + // The type of compression to use on messages (defaults to no compression). Similar to `compression.codec` setting of the JVM producer. + Compression CompressionCodec + // Generates partitioners for choosing the partition to send messages to (defaults to hashing the message key). + // Similar to the `partitioner.class` setting for the JVM producer. + Partitioner PartitionerConstructor + + // Return specifies what channels will be populated. If they are set to true, you must read from + // the respective channels to prevent deadlock. + Return struct { + // If enabled, successfully delivered messages will be returned on the Successes channel (default disabled). + Successes bool + + // If enabled, messages that failed to deliver will be returned on the Errors channel, including error (default enabled). + Errors bool + } + + // The following config options control how often messages are batched up and sent to the broker. By default, + // messages are sent as fast as possible, and all messages received while the current batch is in-flight are placed + // into the subsequent batch. + Flush struct { + Bytes int // The best-effort number of bytes needed to trigger a flush. Use the global sarama.MaxRequestSize to set a hard upper limit. + Messages int // The best-effort number of messages needed to trigger a flush. Use `MaxMessages` to set a hard upper limit. + Frequency time.Duration // The best-effort frequency of flushes. Equivalent to `queue.buffering.max.ms` setting of JVM producer. + // The maximum number of messages the producer will send in a single broker request. + // Defaults to 0 for unlimited. Similar to `queue.buffering.max.messages` in the JVM producer. + MaxMessages int + } + + Retry struct { + // The total number of times to retry sending a message (default 3). + // Similar to the `message.send.max.retries` setting of the JVM producer. + Max int + // How long to wait for the cluster to settle between retries (default 100ms). + // Similar to the `retry.backoff.ms` setting of the JVM producer. + Backoff time.Duration + } + } + + // Consumer is the namespace for configuration related to consuming messages, used by the Consumer. + Consumer struct { + Retry struct { + // How long to wait after a failing to read from a partition before trying again (default 2s). + Backoff time.Duration + } + + // Fetch is the namespace for controlling how many bytes are retrieved by any given request. + Fetch struct { + // The minimum number of message bytes to fetch in a request - the broker will wait until at least this many are available. + // The default is 1, as 0 causes the consumer to spin when no messages are available. Equivalent to the JVM's `fetch.min.bytes`. + Min int32 + // The default number of message bytes to fetch from the broker in each request (default 32768). This should be larger than the + // majority of your messages, or else the consumer will spend a lot of time negotiating sizes and not actually consuming. Similar + // to the JVM's `fetch.message.max.bytes`. + Default int32 + // The maximum number of message bytes to fetch from the broker in a single request. Messages larger than this will return + // ErrMessageTooLarge and will not be consumable, so you must be sure this is at least as large as your largest message. + // Defaults to 0 (no limit). Similar to the JVM's `fetch.message.max.bytes`. The global `sarama.MaxResponseSize` still applies. + Max int32 + } + // The maximum amount of time the broker will wait for Consumer.Fetch.Min bytes to become available before it + // returns fewer than that anyways. The default is 250ms, since 0 causes the consumer to spin when no events are available. + // 100-500ms is a reasonable range for most cases. Kafka only supports precision up to milliseconds; nanoseconds will be truncated. + // Equivalent to the JVM's `fetch.wait.max.ms`. + MaxWaitTime time.Duration + + // The maximum amount of time the consumer expects a message takes to process for the user. If writing to the Messages channel + // takes longer than this, that partition will stop fetching more messages until it can proceed again. Note that, since the + // Messages channel is buffered, the actual grace time is (MaxProcessingTime * ChanneBufferSize). Defaults to 100ms. + MaxProcessingTime time.Duration + + // Return specifies what channels will be populated. If they are set to true, you must read from + // them to prevent deadlock. + Return struct { + // If enabled, any errors that occured while consuming are returned on the Errors channel (default disabled). + Errors bool + } + } + + // A user-provided string sent with every request to the brokers for logging, debugging, and auditing purposes. + // Defaults to "sarama", but you should probably set it to something specific to your application. + ClientID string + // The number of events to buffer in internal and external channels. This permits the producer and consumer to + // continue processing some messages in the background while user code is working, greatly improving throughput. + // Defaults to 256. + ChannelBufferSize int +} + +// NewConfig returns a new configuration instance with sane defaults. +func NewConfig() *Config { + c := &Config{} + + c.Net.MaxOpenRequests = 5 + c.Net.DialTimeout = 30 * time.Second + c.Net.ReadTimeout = 30 * time.Second + c.Net.WriteTimeout = 30 * time.Second + + c.Metadata.Retry.Max = 3 + c.Metadata.Retry.Backoff = 250 * time.Millisecond + c.Metadata.RefreshFrequency = 10 * time.Minute + + c.Producer.MaxMessageBytes = 1000000 + c.Producer.RequiredAcks = WaitForLocal + c.Producer.Timeout = 10 * time.Second + c.Producer.Partitioner = NewHashPartitioner + c.Producer.Retry.Max = 3 + c.Producer.Retry.Backoff = 100 * time.Millisecond + c.Producer.Return.Errors = true + + c.Consumer.Fetch.Min = 1 + c.Consumer.Fetch.Default = 32768 + c.Consumer.Retry.Backoff = 2 * time.Second + c.Consumer.MaxWaitTime = 250 * time.Millisecond + c.Consumer.MaxProcessingTime = 100 * time.Millisecond + c.Consumer.Return.Errors = false + + c.ChannelBufferSize = 256 + + return c +} + +// Validate checks a Config instance. It will return a +// ConfigurationError if the specified values don't make sense. +func (c *Config) Validate() error { + // some configuration values should be warned on but not fail completely, do those first + if c.Net.TLS.Enable == false && c.Net.TLS.Config != nil { + Logger.Println("Net.TLS is disabled but a non-nil configuration was provided.") + } + if c.Producer.RequiredAcks > 1 { + Logger.Println("Producer.RequiredAcks > 1 is deprecated and will raise an exception with kafka >= 0.8.2.0.") + } + if c.Producer.MaxMessageBytes >= int(MaxRequestSize) { + Logger.Println("Producer.MaxMessageBytes is larger than MaxRequestSize; it will be ignored.") + } + if c.Producer.Flush.Bytes >= int(MaxRequestSize) { + Logger.Println("Producer.Flush.Bytes is larger than MaxRequestSize; it will be ignored.") + } + if c.Producer.Timeout%time.Millisecond != 0 { + Logger.Println("Producer.Timeout only supports millisecond resolution; nanoseconds will be truncated.") + } + if c.Consumer.MaxWaitTime < 100*time.Millisecond { + Logger.Println("Consumer.MaxWaitTime is very low, which can cause high CPU and network usage. See documentation for details.") + } + if c.Consumer.MaxWaitTime%time.Millisecond != 0 { + Logger.Println("Consumer.MaxWaitTime only supports millisecond precision; nanoseconds will be truncated.") + } + if c.ClientID == "sarama" { + Logger.Println("ClientID is the default of 'sarama', you should consider setting it to something application-specific.") + } + + // validate Net values + switch { + case c.Net.MaxOpenRequests <= 0: + return ConfigurationError("Net.MaxOpenRequests must be > 0") + case c.Net.DialTimeout <= 0: + return ConfigurationError("Net.DialTimeout must be > 0") + case c.Net.ReadTimeout <= 0: + return ConfigurationError("Net.ReadTimeout must be > 0") + case c.Net.WriteTimeout <= 0: + return ConfigurationError("Net.WriteTimeout must be > 0") + case c.Net.KeepAlive < 0: + return ConfigurationError("Net.KeepAlive must be >= 0") + } + + // validate the Metadata values + switch { + case c.Metadata.Retry.Max < 0: + return ConfigurationError("Metadata.Retry.Max must be >= 0") + case c.Metadata.Retry.Backoff < 0: + return ConfigurationError("Metadata.Retry.Backoff must be >= 0") + case c.Metadata.RefreshFrequency < 0: + return ConfigurationError("Metadata.RefreshFrequency must be >= 0") + } + + // validate the Producer values + switch { + case c.Producer.MaxMessageBytes <= 0: + return ConfigurationError("Producer.MaxMessageBytes must be > 0") + case c.Producer.RequiredAcks < -1: + return ConfigurationError("Producer.RequiredAcks must be >= -1") + case c.Producer.Timeout <= 0: + return ConfigurationError("Producer.Timeout must be > 0") + case c.Producer.Partitioner == nil: + return ConfigurationError("Producer.Partitioner must not be nil") + case c.Producer.Flush.Bytes < 0: + return ConfigurationError("Producer.Flush.Bytes must be >= 0") + case c.Producer.Flush.Messages < 0: + return ConfigurationError("Producer.Flush.Messages must be >= 0") + case c.Producer.Flush.Frequency < 0: + return ConfigurationError("Producer.Flush.Frequency must be >= 0") + case c.Producer.Flush.MaxMessages < 0: + return ConfigurationError("Producer.Flush.MaxMessages must be >= 0") + case c.Producer.Flush.MaxMessages > 0 && c.Producer.Flush.MaxMessages < c.Producer.Flush.Messages: + return ConfigurationError("Producer.Flush.MaxMessages must be >= Producer.Flush.Messages when set") + case c.Producer.Retry.Max < 0: + return ConfigurationError("Producer.Retry.Max must be >= 0") + case c.Producer.Retry.Backoff < 0: + return ConfigurationError("Producer.Retry.Backoff must be >= 0") + } + + // validate the Consumer values + switch { + case c.Consumer.Fetch.Min <= 0: + return ConfigurationError("Consumer.Fetch.Min must be > 0") + case c.Consumer.Fetch.Default <= 0: + return ConfigurationError("Consumer.Fetch.Default must be > 0") + case c.Consumer.Fetch.Max < 0: + return ConfigurationError("Consumer.Fetch.Max must be >= 0") + case c.Consumer.MaxWaitTime < 1*time.Millisecond: + return ConfigurationError("Consumer.MaxWaitTime must be >= 1ms") + case c.Consumer.MaxProcessingTime <= 0: + return ConfigurationError("Consumer.MaxProcessingTime must be > 0") + case c.Consumer.Retry.Backoff < 0: + return ConfigurationError("Consumer.Retry.Backoff must be >= 0") + } + + // validate misc shared values + switch { + case c.ChannelBufferSize < 0: + return ConfigurationError("ChannelBufferSize must be >= 0") + } + + return nil +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/config_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/config_test.go new file mode 100644 index 0000000000000..255281a6554ce --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/config_test.go @@ -0,0 +1,10 @@ +package sarama + +import "testing" + +func TestDefaultConfigValidates(t *testing.T) { + config := NewConfig() + if err := config.Validate(); err != nil { + t.Error(err) + } +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/consumer.go b/Godeps/_workspace/src/github.com/Shopify/sarama/consumer.go new file mode 100644 index 0000000000000..43ce3b21bca32 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/consumer.go @@ -0,0 +1,676 @@ +package sarama + +import ( + "errors" + "fmt" + "sync" + "sync/atomic" + "time" +) + +// ConsumerMessage encapsulates a Kafka message returned by the consumer. +type ConsumerMessage struct { + Key, Value []byte + Topic string + Partition int32 + Offset int64 +} + +// ConsumerError is what is provided to the user when an error occurs. +// It wraps an error and includes the topic and partition. +type ConsumerError struct { + Topic string + Partition int32 + Err error +} + +func (ce ConsumerError) Error() string { + return fmt.Sprintf("kafka: error while consuming %s/%d: %s", ce.Topic, ce.Partition, ce.Err) +} + +// ConsumerErrors is a type that wraps a batch of errors and implements the Error interface. +// It can be returned from the PartitionConsumer's Close methods to avoid the need to manually drain errors +// when stopping. +type ConsumerErrors []*ConsumerError + +func (ce ConsumerErrors) Error() string { + return fmt.Sprintf("kafka: %d errors while consuming", len(ce)) +} + +// Consumer manages PartitionConsumers which process Kafka messages from brokers. You MUST call Close() +// on a consumer to avoid leaks, it will not be garbage-collected automatically when it passes out of +// scope. +// +// Sarama's Consumer type does not currently support automatic consumer group rebalancing and offset tracking, +// however the https://github.com/wvanbergen/kafka library builds on Sarama to add this support. We plan +// to properly integrate this functionality at a later date. +type Consumer interface { + + // Topics returns the set of available topics as retrieved from the cluster metadata. + // This method is the same as Client.Topics(), and is provided for convenience. + Topics() ([]string, error) + + // Partitions returns the sorted list of all partition IDs for the given topic. + // This method is the same as Client.Pertitions(), and is provided for convenience. + Partitions(topic string) ([]int32, error) + + // ConsumePartition creates a PartitionConsumer on the given topic/partition with the given offset. It will + // return an error if this Consumer is already consuming on the given topic/partition. Offset can be a + // literal offset, or OffsetNewest or OffsetOldest + ConsumePartition(topic string, partition int32, offset int64) (PartitionConsumer, error) + + // Close shuts down the consumer. It must be called after all child PartitionConsumers have already been closed. + Close() error +} + +type consumer struct { + client Client + conf *Config + ownClient bool + + lock sync.Mutex + children map[string]map[int32]*partitionConsumer + brokerConsumers map[*Broker]*brokerConsumer +} + +// NewConsumer creates a new consumer using the given broker addresses and configuration. +func NewConsumer(addrs []string, config *Config) (Consumer, error) { + client, err := NewClient(addrs, config) + if err != nil { + return nil, err + } + + c, err := NewConsumerFromClient(client) + if err != nil { + return nil, err + } + c.(*consumer).ownClient = true + return c, nil +} + +// NewConsumerFromClient creates a new consumer using the given client. It is still +// necessary to call Close() on the underlying client when shutting down this consumer. +func NewConsumerFromClient(client Client) (Consumer, error) { + // Check that we are not dealing with a closed Client before processing any other arguments + if client.Closed() { + return nil, ErrClosedClient + } + + c := &consumer{ + client: client, + conf: client.Config(), + children: make(map[string]map[int32]*partitionConsumer), + brokerConsumers: make(map[*Broker]*brokerConsumer), + } + + return c, nil +} + +func (c *consumer) Close() error { + if c.ownClient { + return c.client.Close() + } + return nil +} + +func (c *consumer) Topics() ([]string, error) { + return c.client.Topics() +} + +func (c *consumer) Partitions(topic string) ([]int32, error) { + return c.client.Partitions(topic) +} + +func (c *consumer) ConsumePartition(topic string, partition int32, offset int64) (PartitionConsumer, error) { + child := &partitionConsumer{ + consumer: c, + conf: c.conf, + topic: topic, + partition: partition, + messages: make(chan *ConsumerMessage, c.conf.ChannelBufferSize), + errors: make(chan *ConsumerError, c.conf.ChannelBufferSize), + feeder: make(chan *FetchResponse, 1), + trigger: make(chan none, 1), + dying: make(chan none), + fetchSize: c.conf.Consumer.Fetch.Default, + } + + if err := child.chooseStartingOffset(offset); err != nil { + return nil, err + } + + var leader *Broker + var err error + if leader, err = c.client.Leader(child.topic, child.partition); err != nil { + return nil, err + } + + if err := c.addChild(child); err != nil { + return nil, err + } + + go withRecover(child.dispatcher) + go withRecover(child.responseFeeder) + + child.broker = c.refBrokerConsumer(leader) + child.broker.input <- child + + return child, nil +} + +func (c *consumer) addChild(child *partitionConsumer) error { + c.lock.Lock() + defer c.lock.Unlock() + + topicChildren := c.children[child.topic] + if topicChildren == nil { + topicChildren = make(map[int32]*partitionConsumer) + c.children[child.topic] = topicChildren + } + + if topicChildren[child.partition] != nil { + return ConfigurationError("That topic/partition is already being consumed") + } + + topicChildren[child.partition] = child + return nil +} + +func (c *consumer) removeChild(child *partitionConsumer) { + c.lock.Lock() + defer c.lock.Unlock() + + delete(c.children[child.topic], child.partition) +} + +func (c *consumer) refBrokerConsumer(broker *Broker) *brokerConsumer { + c.lock.Lock() + defer c.lock.Unlock() + + bc := c.brokerConsumers[broker] + if bc == nil { + bc = c.newBrokerConsumer(broker) + c.brokerConsumers[broker] = bc + } + + bc.refs++ + + return bc +} + +func (c *consumer) unrefBrokerConsumer(brokerWorker *brokerConsumer) { + c.lock.Lock() + defer c.lock.Unlock() + + brokerWorker.refs-- + + if brokerWorker.refs == 0 { + close(brokerWorker.input) + if c.brokerConsumers[brokerWorker.broker] == brokerWorker { + delete(c.brokerConsumers, brokerWorker.broker) + } + } +} + +func (c *consumer) abandonBrokerConsumer(brokerWorker *brokerConsumer) { + c.lock.Lock() + defer c.lock.Unlock() + + delete(c.brokerConsumers, brokerWorker.broker) +} + +// PartitionConsumer + +// PartitionConsumer processes Kafka messages from a given topic and partition. You MUST call Close() +// or AsyncClose() on a PartitionConsumer to avoid leaks, it will not be garbage-collected automatically +// when it passes out of scope. +// +// The simplest way of using a PartitionConsumer is to loop over its Messages channel using a for/range +// loop. The PartitionConsumer will only stop itself in one case: when the offset being consumed is reported +// as out of range by the brokers. In this case you should decide what you want to do (try a different offset, +// notify a human, etc) and handle it appropriately. For all other error cases, it will just keep retrying. +// By default, it logs these errors to sarama.Logger; if you want to be notified directly of all errors, set +// your config's Consumer.Return.Errors to true and read from the Errors channel, using a select statement +// or a separate goroutine. Check out the Consumer examples to see implementations of these different approaches. +type PartitionConsumer interface { + + // AsyncClose initiates a shutdown of the PartitionConsumer. This method will return immediately, + // after which you should wait until the 'messages' and 'errors' channel are drained. + // It is required to call this function, or Close before a consumer object passes out of scope, + // as it will otherwise leak memory. You must call this before calling Close on the underlying + // client. + AsyncClose() + + // Close stops the PartitionConsumer from fetching messages. It is required to call this function + // (or AsyncClose) before a consumer object passes out of scope, as it will otherwise leak memory. You must + // call this before calling Close on the underlying client. + Close() error + + // Messages returns the read channel for the messages that are returned by the broker. + Messages() <-chan *ConsumerMessage + + // Errors returns a read channel of errors that occured during consuming, if enabled. By default, + // errors are logged and not returned over this channel. If you want to implement any custom errpr + // handling, set your config's Consumer.Return.Errors setting to true, and read from this channel. + Errors() <-chan *ConsumerError + + // HighWaterMarkOffset returns the high water mark offset of the partition, i.e. the offset that will + // be used for the next message that will be produced. You can use this to determine how far behind + // the processing is. + HighWaterMarkOffset() int64 +} + +type partitionConsumer struct { + consumer *consumer + conf *Config + topic string + partition int32 + + broker *brokerConsumer + messages chan *ConsumerMessage + errors chan *ConsumerError + feeder chan *FetchResponse + + trigger, dying chan none + responseResult error + + fetchSize int32 + offset int64 + highWaterMarkOffset int64 +} + +var errTimedOut = errors.New("timed out feeding messages to the user") // not user-facing + +func (child *partitionConsumer) sendError(err error) { + cErr := &ConsumerError{ + Topic: child.topic, + Partition: child.partition, + Err: err, + } + + if child.conf.Consumer.Return.Errors { + child.errors <- cErr + } else { + Logger.Println(cErr) + } +} + +func (child *partitionConsumer) dispatcher() { + for _ = range child.trigger { + select { + case <-child.dying: + close(child.trigger) + case <-time.After(child.conf.Consumer.Retry.Backoff): + if child.broker != nil { + child.consumer.unrefBrokerConsumer(child.broker) + child.broker = nil + } + + Logger.Printf("consumer/%s/%d finding new broker\n", child.topic, child.partition) + if err := child.dispatch(); err != nil { + child.sendError(err) + child.trigger <- none{} + } + } + } + + if child.broker != nil { + child.consumer.unrefBrokerConsumer(child.broker) + } + child.consumer.removeChild(child) + close(child.feeder) +} + +func (child *partitionConsumer) dispatch() error { + if err := child.consumer.client.RefreshMetadata(child.topic); err != nil { + return err + } + + var leader *Broker + var err error + if leader, err = child.consumer.client.Leader(child.topic, child.partition); err != nil { + return err + } + + child.broker = child.consumer.refBrokerConsumer(leader) + + child.broker.input <- child + + return nil +} + +func (child *partitionConsumer) chooseStartingOffset(offset int64) error { + newestOffset, err := child.consumer.client.GetOffset(child.topic, child.partition, OffsetNewest) + if err != nil { + return err + } + oldestOffset, err := child.consumer.client.GetOffset(child.topic, child.partition, OffsetOldest) + if err != nil { + return err + } + + switch { + case offset == OffsetNewest: + child.offset = newestOffset + case offset == OffsetOldest: + child.offset = oldestOffset + case offset >= oldestOffset && offset <= newestOffset: + child.offset = offset + default: + return ErrOffsetOutOfRange + } + + return nil +} + +func (child *partitionConsumer) Messages() <-chan *ConsumerMessage { + return child.messages +} + +func (child *partitionConsumer) Errors() <-chan *ConsumerError { + return child.errors +} + +func (child *partitionConsumer) AsyncClose() { + // this triggers whatever broker owns this child to abandon it and close its trigger channel, which causes + // the dispatcher to exit its loop, which removes it from the consumer then closes its 'messages' and + // 'errors' channel (alternatively, if the child is already at the dispatcher for some reason, that will + // also just close itself) + close(child.dying) +} + +func (child *partitionConsumer) Close() error { + child.AsyncClose() + + go withRecover(func() { + for _ = range child.messages { + // drain + } + }) + + var errors ConsumerErrors + for err := range child.errors { + errors = append(errors, err) + } + + if len(errors) > 0 { + return errors + } + return nil +} + +func (child *partitionConsumer) HighWaterMarkOffset() int64 { + return atomic.LoadInt64(&child.highWaterMarkOffset) +} + +func (child *partitionConsumer) responseFeeder() { + var msgs []*ConsumerMessage + +feederLoop: + for response := range child.feeder { + msgs, child.responseResult = child.parseResponse(response) + + for i, msg := range msgs { + select { + case child.messages <- msg: + case <-time.After(child.conf.Consumer.MaxProcessingTime): + child.responseResult = errTimedOut + child.broker.acks.Done() + for _, msg = range msgs[i:] { + child.messages <- msg + } + child.broker.input <- child + continue feederLoop + } + } + + child.broker.acks.Done() + } + + close(child.messages) + close(child.errors) +} + +func (child *partitionConsumer) parseResponse(response *FetchResponse) ([]*ConsumerMessage, error) { + block := response.GetBlock(child.topic, child.partition) + if block == nil { + return nil, ErrIncompleteResponse + } + + if block.Err != ErrNoError { + return nil, block.Err + } + + if len(block.MsgSet.Messages) == 0 { + // We got no messages. If we got a trailing one then we need to ask for more data. + // Otherwise we just poll again and wait for one to be produced... + if block.MsgSet.PartialTrailingMessage { + if child.conf.Consumer.Fetch.Max > 0 && child.fetchSize == child.conf.Consumer.Fetch.Max { + // we can't ask for more data, we've hit the configured limit + child.sendError(ErrMessageTooLarge) + child.offset++ // skip this one so we can keep processing future messages + } else { + child.fetchSize *= 2 + if child.conf.Consumer.Fetch.Max > 0 && child.fetchSize > child.conf.Consumer.Fetch.Max { + child.fetchSize = child.conf.Consumer.Fetch.Max + } + } + } + + return nil, nil + } + + // we got messages, reset our fetch size in case it was increased for a previous request + child.fetchSize = child.conf.Consumer.Fetch.Default + atomic.StoreInt64(&child.highWaterMarkOffset, block.HighWaterMarkOffset) + + incomplete := false + prelude := true + var messages []*ConsumerMessage + for _, msgBlock := range block.MsgSet.Messages { + + for _, msg := range msgBlock.Messages() { + if prelude && msg.Offset < child.offset { + continue + } + prelude = false + + if msg.Offset >= child.offset { + messages = append(messages, &ConsumerMessage{ + Topic: child.topic, + Partition: child.partition, + Key: msg.Msg.Key, + Value: msg.Msg.Value, + Offset: msg.Offset, + }) + child.offset = msg.Offset + 1 + } else { + incomplete = true + } + } + + } + + if incomplete || len(messages) == 0 { + return nil, ErrIncompleteResponse + } + return messages, nil +} + +// brokerConsumer + +type brokerConsumer struct { + consumer *consumer + broker *Broker + input chan *partitionConsumer + newSubscriptions chan []*partitionConsumer + wait chan none + subscriptions map[*partitionConsumer]none + acks sync.WaitGroup + refs int +} + +func (c *consumer) newBrokerConsumer(broker *Broker) *brokerConsumer { + bc := &brokerConsumer{ + consumer: c, + broker: broker, + input: make(chan *partitionConsumer), + newSubscriptions: make(chan []*partitionConsumer), + wait: make(chan none), + subscriptions: make(map[*partitionConsumer]none), + refs: 0, + } + + go withRecover(bc.subscriptionManager) + go withRecover(bc.subscriptionConsumer) + + return bc +} + +func (bc *brokerConsumer) subscriptionManager() { + var buffer []*partitionConsumer + + // The subscriptionManager constantly accepts new subscriptions on `input` (even when the main subscriptionConsumer + // goroutine is in the middle of a network request) and batches it up. The main worker goroutine picks + // up a batch of new subscriptions between every network request by reading from `newSubscriptions`, so we give + // it nil if no new subscriptions are available. We also write to `wait` only when new subscriptions is available, + // so the main goroutine can block waiting for work if it has none. + for { + if len(buffer) > 0 { + select { + case event, ok := <-bc.input: + if !ok { + goto done + } + buffer = append(buffer, event) + case bc.newSubscriptions <- buffer: + buffer = nil + case bc.wait <- none{}: + } + } else { + select { + case event, ok := <-bc.input: + if !ok { + goto done + } + buffer = append(buffer, event) + case bc.newSubscriptions <- nil: + } + } + } + +done: + close(bc.wait) + if len(buffer) > 0 { + bc.newSubscriptions <- buffer + } + close(bc.newSubscriptions) +} + +func (bc *brokerConsumer) subscriptionConsumer() { + <-bc.wait // wait for our first piece of work + + // the subscriptionConsumer ensures we will get nil right away if no new subscriptions is available + for newSubscriptions := range bc.newSubscriptions { + for _, child := range newSubscriptions { + bc.subscriptions[child] = none{} + Logger.Printf("consumer/broker/%d added subscription to %s/%d\n", bc.broker.ID(), child.topic, child.partition) + } + + if len(bc.subscriptions) == 0 { + // We're about to be shut down or we're about to receive more subscriptions. + // Either way, the signal just hasn't propagated to our goroutine yet. + <-bc.wait + continue + } + + response, err := bc.fetchNewMessages() + + if err != nil { + Logger.Printf("consumer/broker/%d disconnecting due to error processing FetchRequest: %s\n", bc.broker.ID(), err) + bc.abort(err) + return + } + + bc.acks.Add(len(bc.subscriptions)) + for child := range bc.subscriptions { + child.feeder <- response + } + bc.acks.Wait() + bc.handleResponses() + } +} + +func (bc *brokerConsumer) handleResponses() { + // handles the response codes left for us by our subscriptions, and abandons ones that have been closed + for child := range bc.subscriptions { + select { + case <-child.dying: + Logger.Printf("consumer/broker/%d closed dead subscription to %s/%d\n", bc.broker.ID(), child.topic, child.partition) + close(child.trigger) + delete(bc.subscriptions, child) + default: + result := child.responseResult + child.responseResult = nil + + switch result { + case nil: + break + case errTimedOut: + Logger.Printf("consumer/broker/%d abandoned subscription to %s/%d because consuming was taking too long\n", + bc.broker.ID(), child.topic, child.partition) + delete(bc.subscriptions, child) + case ErrOffsetOutOfRange: + // there's no point in retrying this it will just fail the same way again + // shut it down and force the user to choose what to do + child.sendError(result) + Logger.Printf("consumer/%s/%d shutting down because %s\n", child.topic, child.partition, result) + close(child.trigger) + delete(bc.subscriptions, child) + case ErrUnknownTopicOrPartition, ErrNotLeaderForPartition, ErrLeaderNotAvailable: + // not an error, but does need redispatching + Logger.Printf("consumer/broker/%d abandoned subscription to %s/%d because %s\n", + bc.broker.ID(), child.topic, child.partition, result) + child.trigger <- none{} + delete(bc.subscriptions, child) + default: + // dunno, tell the user and try redispatching + child.sendError(result) + Logger.Printf("consumer/broker/%d abandoned subscription to %s/%d because %s\n", + bc.broker.ID(), child.topic, child.partition, result) + child.trigger <- none{} + delete(bc.subscriptions, child) + } + } + } +} + +func (bc *brokerConsumer) abort(err error) { + bc.consumer.abandonBrokerConsumer(bc) + _ = bc.broker.Close() // we don't care about the error this might return, we already have one + + for child := range bc.subscriptions { + child.sendError(err) + child.trigger <- none{} + } + + for newSubscription := range bc.newSubscriptions { + for _, child := range newSubscription { + child.sendError(err) + child.trigger <- none{} + } + } +} + +func (bc *brokerConsumer) fetchNewMessages() (*FetchResponse, error) { + request := &FetchRequest{ + MinBytes: bc.consumer.conf.Consumer.Fetch.Min, + MaxWaitTime: int32(bc.consumer.conf.Consumer.MaxWaitTime / time.Millisecond), + } + + for child := range bc.subscriptions { + request.AddBlock(child.topic, child.partition, child.offset, child.fetchSize) + } + + return bc.broker.Fetch(request) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_metadata_request.go b/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_metadata_request.go new file mode 100644 index 0000000000000..9b8fcd74e864b --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_metadata_request.go @@ -0,0 +1,22 @@ +package sarama + +type ConsumerMetadataRequest struct { + ConsumerGroup string +} + +func (r *ConsumerMetadataRequest) encode(pe packetEncoder) error { + return pe.putString(r.ConsumerGroup) +} + +func (r *ConsumerMetadataRequest) decode(pd packetDecoder) (err error) { + r.ConsumerGroup, err = pd.getString() + return err +} + +func (r *ConsumerMetadataRequest) key() int16 { + return 10 +} + +func (r *ConsumerMetadataRequest) version() int16 { + return 0 +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_metadata_request_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_metadata_request_test.go new file mode 100644 index 0000000000000..4509631a0402b --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_metadata_request_test.go @@ -0,0 +1,19 @@ +package sarama + +import "testing" + +var ( + consumerMetadataRequestEmpty = []byte{ + 0x00, 0x00} + + consumerMetadataRequestString = []byte{ + 0x00, 0x06, 'f', 'o', 'o', 'b', 'a', 'r'} +) + +func TestConsumerMetadataRequest(t *testing.T) { + request := new(ConsumerMetadataRequest) + testRequest(t, "empty string", request, consumerMetadataRequestEmpty) + + request.ConsumerGroup = "foobar" + testRequest(t, "with string", request, consumerMetadataRequestString) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_metadata_response.go b/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_metadata_response.go new file mode 100644 index 0000000000000..d6b5614b432f0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_metadata_response.go @@ -0,0 +1,73 @@ +package sarama + +import ( + "net" + "strconv" +) + +type ConsumerMetadataResponse struct { + Err KError + Coordinator *Broker + CoordinatorID int32 // deprecated: use Coordinator.ID() + CoordinatorHost string // deprecated: use Coordinator.Addr() + CoordinatorPort int32 // deprecated: use Coordinator.Addr() +} + +func (r *ConsumerMetadataResponse) decode(pd packetDecoder) (err error) { + tmp, err := pd.getInt16() + if err != nil { + return err + } + r.Err = KError(tmp) + + coordinator := new(Broker) + if err := coordinator.decode(pd); err != nil { + return err + } + if coordinator.addr == ":0" { + return nil + } + r.Coordinator = coordinator + + // this can all go away in 2.0, but we have to fill in deprecated fields to maintain + // backwards compatibility + host, portstr, err := net.SplitHostPort(r.Coordinator.Addr()) + if err != nil { + return err + } + port, err := strconv.ParseInt(portstr, 10, 32) + if err != nil { + return err + } + r.CoordinatorID = r.Coordinator.ID() + r.CoordinatorHost = host + r.CoordinatorPort = int32(port) + + return nil +} + +func (r *ConsumerMetadataResponse) encode(pe packetEncoder) error { + pe.putInt16(int16(r.Err)) + if r.Coordinator != nil { + host, portstr, err := net.SplitHostPort(r.Coordinator.Addr()) + if err != nil { + return err + } + port, err := strconv.ParseInt(portstr, 10, 32) + if err != nil { + return err + } + pe.putInt32(r.Coordinator.ID()) + if err := pe.putString(host); err != nil { + return err + } + pe.putInt32(int32(port)) + return nil + } + pe.putInt32(r.CoordinatorID) + if err := pe.putString(r.CoordinatorHost); err != nil { + return err + } + pe.putInt32(r.CoordinatorPort) + return nil +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_metadata_response_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_metadata_response_test.go new file mode 100644 index 0000000000000..b748784d7efa7 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_metadata_response_test.go @@ -0,0 +1,35 @@ +package sarama + +import "testing" + +var ( + consumerMetadataResponseError = []byte{ + 0x00, 0x0E, + 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00} + + consumerMetadataResponseSuccess = []byte{ + 0x00, 0x00, + 0x00, 0x00, 0x00, 0xAB, + 0x00, 0x03, 'f', 'o', 'o', + 0x00, 0x00, 0xCC, 0xDD} +) + +func TestConsumerMetadataResponseError(t *testing.T) { + response := ConsumerMetadataResponse{Err: ErrOffsetsLoadInProgress} + testResponse(t, "error", &response, consumerMetadataResponseError) +} + +func TestConsumerMetadataResponseSuccess(t *testing.T) { + broker := NewBroker("foo:52445") + broker.id = 0xAB + response := ConsumerMetadataResponse{ + Coordinator: broker, + CoordinatorID: 0xAB, + CoordinatorHost: "foo", + CoordinatorPort: 0xCCDD, + Err: ErrNoError, + } + testResponse(t, "success", &response, consumerMetadataResponseSuccess) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_test.go new file mode 100644 index 0000000000000..df3af07ffe2e0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/consumer_test.go @@ -0,0 +1,844 @@ +package sarama + +import ( + "sync" + "testing" + "time" +) + +var testMsg = StringEncoder("Foo") + +// If a particular offset is provided then messages are consumed starting from +// that offset. +func TestConsumerOffsetManual(t *testing.T) { + // Given + broker0 := newMockBroker(t, 0) + + mockFetchResponse := newMockFetchResponse(t, 1) + for i := 0; i < 10; i++ { + mockFetchResponse.SetMessage("my_topic", 0, int64(i+1234), testMsg) + } + + broker0.SetHandlerByMap(map[string]MockResponse{ + "MetadataRequest": newMockMetadataResponse(t). + SetBroker(broker0.Addr(), broker0.BrokerID()). + SetLeader("my_topic", 0, broker0.BrokerID()), + "OffsetRequest": newMockOffsetResponse(t). + SetOffset("my_topic", 0, OffsetOldest, 0). + SetOffset("my_topic", 0, OffsetNewest, 2345), + "FetchRequest": mockFetchResponse, + }) + + // When + master, err := NewConsumer([]string{broker0.Addr()}, nil) + if err != nil { + t.Fatal(err) + } + + consumer, err := master.ConsumePartition("my_topic", 0, 1234) + if err != nil { + t.Fatal(err) + } + + // Then: messages starting from offset 1234 are consumed. + for i := 0; i < 10; i++ { + select { + case message := <-consumer.Messages(): + assertMessageOffset(t, message, int64(i+1234)) + case err := <-consumer.Errors(): + t.Error(err) + } + } + + safeClose(t, consumer) + safeClose(t, master) + broker0.Close() +} + +// If `OffsetNewest` is passed as the initial offset then the first consumed +// message is indeed corresponds to the offset that broker claims to be the +// newest in its metadata response. +func TestConsumerOffsetNewest(t *testing.T) { + // Given + broker0 := newMockBroker(t, 0) + broker0.SetHandlerByMap(map[string]MockResponse{ + "MetadataRequest": newMockMetadataResponse(t). + SetBroker(broker0.Addr(), broker0.BrokerID()). + SetLeader("my_topic", 0, broker0.BrokerID()), + "OffsetRequest": newMockOffsetResponse(t). + SetOffset("my_topic", 0, OffsetNewest, 10). + SetOffset("my_topic", 0, OffsetOldest, 7), + "FetchRequest": newMockFetchResponse(t, 1). + SetMessage("my_topic", 0, 9, testMsg). + SetMessage("my_topic", 0, 10, testMsg). + SetMessage("my_topic", 0, 11, testMsg). + SetHighWaterMark("my_topic", 0, 14), + }) + + master, err := NewConsumer([]string{broker0.Addr()}, nil) + if err != nil { + t.Fatal(err) + } + + // When + consumer, err := master.ConsumePartition("my_topic", 0, OffsetNewest) + if err != nil { + t.Fatal(err) + } + + // Then + assertMessageOffset(t, <-consumer.Messages(), 10) + if hwmo := consumer.HighWaterMarkOffset(); hwmo != 14 { + t.Errorf("Expected high water mark offset 14, found %d", hwmo) + } + + safeClose(t, consumer) + safeClose(t, master) + broker0.Close() +} + +// It is possible to close a partition consumer and create the same anew. +func TestConsumerRecreate(t *testing.T) { + // Given + broker0 := newMockBroker(t, 0) + broker0.SetHandlerByMap(map[string]MockResponse{ + "MetadataRequest": newMockMetadataResponse(t). + SetBroker(broker0.Addr(), broker0.BrokerID()). + SetLeader("my_topic", 0, broker0.BrokerID()), + "OffsetRequest": newMockOffsetResponse(t). + SetOffset("my_topic", 0, OffsetOldest, 0). + SetOffset("my_topic", 0, OffsetNewest, 1000), + "FetchRequest": newMockFetchResponse(t, 1). + SetMessage("my_topic", 0, 10, testMsg), + }) + + c, err := NewConsumer([]string{broker0.Addr()}, nil) + if err != nil { + t.Fatal(err) + } + + pc, err := c.ConsumePartition("my_topic", 0, 10) + if err != nil { + t.Fatal(err) + } + assertMessageOffset(t, <-pc.Messages(), 10) + + // When + safeClose(t, pc) + pc, err = c.ConsumePartition("my_topic", 0, 10) + if err != nil { + t.Fatal(err) + } + + // Then + assertMessageOffset(t, <-pc.Messages(), 10) + + safeClose(t, pc) + safeClose(t, c) + broker0.Close() +} + +// An attempt to consume the same partition twice should fail. +func TestConsumerDuplicate(t *testing.T) { + // Given + broker0 := newMockBroker(t, 0) + broker0.SetHandlerByMap(map[string]MockResponse{ + "MetadataRequest": newMockMetadataResponse(t). + SetBroker(broker0.Addr(), broker0.BrokerID()). + SetLeader("my_topic", 0, broker0.BrokerID()), + "OffsetRequest": newMockOffsetResponse(t). + SetOffset("my_topic", 0, OffsetOldest, 0). + SetOffset("my_topic", 0, OffsetNewest, 1000), + "FetchRequest": newMockFetchResponse(t, 1), + }) + + config := NewConfig() + config.ChannelBufferSize = 0 + c, err := NewConsumer([]string{broker0.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + pc1, err := c.ConsumePartition("my_topic", 0, 0) + if err != nil { + t.Fatal(err) + } + + // When + pc2, err := c.ConsumePartition("my_topic", 0, 0) + + // Then + if pc2 != nil || err != ConfigurationError("That topic/partition is already being consumed") { + t.Fatal("A partition cannot be consumed twice at the same time") + } + + safeClose(t, pc1) + safeClose(t, c) + broker0.Close() +} + +// If consumer fails to refresh metadata it keeps retrying with frequency +// specified by `Config.Consumer.Retry.Backoff`. +func TestConsumerLeaderRefreshError(t *testing.T) { + // Given + broker0 := newMockBroker(t, 100) + + // Stage 1: my_topic/0 served by broker0 + Logger.Printf(" STAGE 1") + + broker0.SetHandlerByMap(map[string]MockResponse{ + "MetadataRequest": newMockMetadataResponse(t). + SetBroker(broker0.Addr(), broker0.BrokerID()). + SetLeader("my_topic", 0, broker0.BrokerID()), + "OffsetRequest": newMockOffsetResponse(t). + SetOffset("my_topic", 0, OffsetOldest, 123). + SetOffset("my_topic", 0, OffsetNewest, 1000), + "FetchRequest": newMockFetchResponse(t, 1). + SetMessage("my_topic", 0, 123, testMsg), + }) + + config := NewConfig() + config.Net.ReadTimeout = 100 * time.Millisecond + config.Consumer.Retry.Backoff = 200 * time.Millisecond + config.Consumer.Return.Errors = true + config.Metadata.Retry.Max = 0 + c, err := NewConsumer([]string{broker0.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + pc, err := c.ConsumePartition("my_topic", 0, OffsetOldest) + if err != nil { + t.Fatal(err) + } + + assertMessageOffset(t, <-pc.Messages(), 123) + + // Stage 2: broker0 says that it is no longer the leader for my_topic/0, + // but the requests to retrieve metadata fail with network timeout. + Logger.Printf(" STAGE 2") + + fetchResponse2 := &FetchResponse{} + fetchResponse2.AddError("my_topic", 0, ErrNotLeaderForPartition) + + broker0.SetHandlerByMap(map[string]MockResponse{ + "FetchRequest": newMockWrapper(fetchResponse2), + }) + + if consErr := <-pc.Errors(); consErr.Err != ErrOutOfBrokers { + t.Errorf("Unexpected error: %v", consErr.Err) + } + + // Stage 3: finally the metadata returned by broker0 tells that broker1 is + // a new leader for my_topic/0. Consumption resumes. + + Logger.Printf(" STAGE 3") + + broker1 := newMockBroker(t, 101) + + broker1.SetHandlerByMap(map[string]MockResponse{ + "FetchRequest": newMockFetchResponse(t, 1). + SetMessage("my_topic", 0, 124, testMsg), + }) + broker0.SetHandlerByMap(map[string]MockResponse{ + "MetadataRequest": newMockMetadataResponse(t). + SetBroker(broker0.Addr(), broker0.BrokerID()). + SetBroker(broker1.Addr(), broker1.BrokerID()). + SetLeader("my_topic", 0, broker1.BrokerID()), + }) + + assertMessageOffset(t, <-pc.Messages(), 124) + + safeClose(t, pc) + safeClose(t, c) + broker1.Close() + broker0.Close() +} + +func TestConsumerInvalidTopic(t *testing.T) { + // Given + broker0 := newMockBroker(t, 100) + broker0.SetHandlerByMap(map[string]MockResponse{ + "MetadataRequest": newMockMetadataResponse(t). + SetBroker(broker0.Addr(), broker0.BrokerID()), + }) + + c, err := NewConsumer([]string{broker0.Addr()}, nil) + if err != nil { + t.Fatal(err) + } + + // When + pc, err := c.ConsumePartition("my_topic", 0, OffsetOldest) + + // Then + if pc != nil || err != ErrUnknownTopicOrPartition { + t.Errorf("Should fail with, err=%v", err) + } + + safeClose(t, c) + broker0.Close() +} + +// Nothing bad happens if a partition consumer that has no leader assigned at +// the moment is closed. +func TestConsumerClosePartitionWithoutLeader(t *testing.T) { + // Given + broker0 := newMockBroker(t, 100) + broker0.SetHandlerByMap(map[string]MockResponse{ + "MetadataRequest": newMockMetadataResponse(t). + SetBroker(broker0.Addr(), broker0.BrokerID()). + SetLeader("my_topic", 0, broker0.BrokerID()), + "OffsetRequest": newMockOffsetResponse(t). + SetOffset("my_topic", 0, OffsetOldest, 123). + SetOffset("my_topic", 0, OffsetNewest, 1000), + "FetchRequest": newMockFetchResponse(t, 1). + SetMessage("my_topic", 0, 123, testMsg), + }) + + config := NewConfig() + config.Net.ReadTimeout = 100 * time.Millisecond + config.Consumer.Retry.Backoff = 100 * time.Millisecond + config.Consumer.Return.Errors = true + config.Metadata.Retry.Max = 0 + c, err := NewConsumer([]string{broker0.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + pc, err := c.ConsumePartition("my_topic", 0, OffsetOldest) + if err != nil { + t.Fatal(err) + } + + assertMessageOffset(t, <-pc.Messages(), 123) + + // broker0 says that it is no longer the leader for my_topic/0, but the + // requests to retrieve metadata fail with network timeout. + fetchResponse2 := &FetchResponse{} + fetchResponse2.AddError("my_topic", 0, ErrNotLeaderForPartition) + + broker0.SetHandlerByMap(map[string]MockResponse{ + "FetchRequest": newMockWrapper(fetchResponse2), + }) + + // When + if consErr := <-pc.Errors(); consErr.Err != ErrOutOfBrokers { + t.Errorf("Unexpected error: %v", consErr.Err) + } + + // Then: the partition consumer can be closed without any problem. + safeClose(t, pc) + safeClose(t, c) + broker0.Close() +} + +// If the initial offset passed on partition consumer creation is out of the +// actual offset range for the partition, then the partition consumer stops +// immediately closing its output channels. +func TestConsumerShutsDownOutOfRange(t *testing.T) { + // Given + broker0 := newMockBroker(t, 0) + broker0.SetHandler(func(req *request) (res encoder) { + switch reqBody := req.body.(type) { + case *MetadataRequest: + return newMockMetadataResponse(t). + SetBroker(broker0.Addr(), broker0.BrokerID()). + SetLeader("my_topic", 0, broker0.BrokerID()). + For(reqBody) + case *OffsetRequest: + return newMockOffsetResponse(t). + SetOffset("my_topic", 0, OffsetNewest, 1234). + SetOffset("my_topic", 0, OffsetOldest, 7). + For(reqBody) + case *FetchRequest: + fetchResponse := new(FetchResponse) + fetchResponse.AddError("my_topic", 0, ErrOffsetOutOfRange) + return fetchResponse + } + return nil + }) + + master, err := NewConsumer([]string{broker0.Addr()}, nil) + if err != nil { + t.Fatal(err) + } + + // When + consumer, err := master.ConsumePartition("my_topic", 0, 101) + if err != nil { + t.Fatal(err) + } + + // Then: consumer should shut down closing its messages and errors channels. + if _, ok := <-consumer.Messages(); ok { + t.Error("Expected the consumer to shut down") + } + safeClose(t, consumer) + + safeClose(t, master) + broker0.Close() +} + +// If a fetch response contains messages with offsets that are smaller then +// requested, then such messages are ignored. +func TestConsumerExtraOffsets(t *testing.T) { + // Given + broker0 := newMockBroker(t, 0) + called := 0 + broker0.SetHandler(func(req *request) (res encoder) { + switch req.body.(type) { + case *MetadataRequest: + return newMockMetadataResponse(t). + SetBroker(broker0.Addr(), broker0.BrokerID()). + SetLeader("my_topic", 0, broker0.BrokerID()).For(req.body) + case *OffsetRequest: + return newMockOffsetResponse(t). + SetOffset("my_topic", 0, OffsetNewest, 1234). + SetOffset("my_topic", 0, OffsetOldest, 0).For(req.body) + case *FetchRequest: + fetchResponse := &FetchResponse{} + called++ + if called > 1 { + fetchResponse.AddError("my_topic", 0, ErrNoError) + return fetchResponse + } + fetchResponse.AddMessage("my_topic", 0, nil, testMsg, 1) + fetchResponse.AddMessage("my_topic", 0, nil, testMsg, 2) + fetchResponse.AddMessage("my_topic", 0, nil, testMsg, 3) + fetchResponse.AddMessage("my_topic", 0, nil, testMsg, 4) + return fetchResponse + } + return nil + }) + + master, err := NewConsumer([]string{broker0.Addr()}, nil) + if err != nil { + t.Fatal(err) + } + + // When + consumer, err := master.ConsumePartition("my_topic", 0, 3) + if err != nil { + t.Fatal(err) + } + + // Then: messages with offsets 1 and 2 are not returned even though they + // are present in the response. + assertMessageOffset(t, <-consumer.Messages(), 3) + assertMessageOffset(t, <-consumer.Messages(), 4) + + safeClose(t, consumer) + safeClose(t, master) + broker0.Close() +} + +// It is fine if offsets of fetched messages are not sequential (although +// strictly increasing!). +func TestConsumerNonSequentialOffsets(t *testing.T) { + // Given + broker0 := newMockBroker(t, 0) + called := 0 + broker0.SetHandler(func(req *request) (res encoder) { + switch req.body.(type) { + case *MetadataRequest: + return newMockMetadataResponse(t). + SetBroker(broker0.Addr(), broker0.BrokerID()). + SetLeader("my_topic", 0, broker0.BrokerID()).For(req.body) + case *OffsetRequest: + return newMockOffsetResponse(t). + SetOffset("my_topic", 0, OffsetNewest, 1234). + SetOffset("my_topic", 0, OffsetOldest, 0).For(req.body) + case *FetchRequest: + called++ + fetchResponse := &FetchResponse{} + if called > 1 { + fetchResponse.AddError("my_topic", 0, ErrNoError) + return fetchResponse + } + fetchResponse.AddMessage("my_topic", 0, nil, testMsg, 5) + fetchResponse.AddMessage("my_topic", 0, nil, testMsg, 7) + fetchResponse.AddMessage("my_topic", 0, nil, testMsg, 11) + return fetchResponse + } + return nil + }) + + master, err := NewConsumer([]string{broker0.Addr()}, nil) + if err != nil { + t.Fatal(err) + } + + // When + consumer, err := master.ConsumePartition("my_topic", 0, 3) + if err != nil { + t.Fatal(err) + } + + // Then: messages with offsets 1 and 2 are not returned even though they + // are present in the response. + assertMessageOffset(t, <-consumer.Messages(), 5) + assertMessageOffset(t, <-consumer.Messages(), 7) + assertMessageOffset(t, <-consumer.Messages(), 11) + + safeClose(t, consumer) + safeClose(t, master) + broker0.Close() +} + +// If leadership for a partition is changing then consumer resolves the new +// leader and switches to it. +func TestConsumerRebalancingMultiplePartitions(t *testing.T) { + // initial setup + seedBroker := newMockBroker(t, 10) + leader0 := newMockBroker(t, 0) + leader1 := newMockBroker(t, 1) + + seedBroker.SetHandlerByMap(map[string]MockResponse{ + "MetadataRequest": newMockMetadataResponse(t). + SetBroker(leader0.Addr(), leader0.BrokerID()). + SetBroker(leader1.Addr(), leader1.BrokerID()). + SetLeader("my_topic", 0, leader0.BrokerID()). + SetLeader("my_topic", 1, leader1.BrokerID()), + }) + + mockOffsetResponse1 := newMockOffsetResponse(t). + SetOffset("my_topic", 0, OffsetOldest, 0). + SetOffset("my_topic", 0, OffsetNewest, 1000). + SetOffset("my_topic", 1, OffsetOldest, 0). + SetOffset("my_topic", 1, OffsetNewest, 1000) + leader0.SetHandlerByMap(map[string]MockResponse{ + "OffsetRequest": mockOffsetResponse1, + "FetchRequest": newMockFetchResponse(t, 1), + }) + leader1.SetHandlerByMap(map[string]MockResponse{ + "OffsetRequest": mockOffsetResponse1, + "FetchRequest": newMockFetchResponse(t, 1), + }) + + // launch test goroutines + config := NewConfig() + config.Consumer.Retry.Backoff = 50 + master, err := NewConsumer([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + // we expect to end up (eventually) consuming exactly ten messages on each partition + var wg sync.WaitGroup + for i := int32(0); i < 2; i++ { + consumer, err := master.ConsumePartition("my_topic", i, 0) + if err != nil { + t.Error(err) + } + + go func(c PartitionConsumer) { + for err := range c.Errors() { + t.Error(err) + } + }(consumer) + + wg.Add(1) + go func(partition int32, c PartitionConsumer) { + for i := 0; i < 10; i++ { + message := <-consumer.Messages() + if message.Offset != int64(i) { + t.Error("Incorrect message offset!", i, partition, message.Offset) + } + if message.Partition != partition { + t.Error("Incorrect message partition!") + } + } + safeClose(t, consumer) + wg.Done() + }(i, consumer) + } + + time.Sleep(50 * time.Millisecond) + Logger.Printf(" STAGE 1") + // Stage 1: + // * my_topic/0 -> leader0 serves 4 messages + // * my_topic/1 -> leader1 serves 0 messages + + mockFetchResponse := newMockFetchResponse(t, 1) + for i := 0; i < 4; i++ { + mockFetchResponse.SetMessage("my_topic", 0, int64(i), testMsg) + } + leader0.SetHandlerByMap(map[string]MockResponse{ + "FetchRequest": mockFetchResponse, + }) + + time.Sleep(50 * time.Millisecond) + Logger.Printf(" STAGE 2") + // Stage 2: + // * leader0 says that it is no longer serving my_topic/0 + // * seedBroker tells that leader1 is serving my_topic/0 now + + // seed broker tells that the new partition 0 leader is leader1 + seedBroker.SetHandlerByMap(map[string]MockResponse{ + "MetadataRequest": newMockMetadataResponse(t). + SetLeader("my_topic", 0, leader1.BrokerID()). + SetLeader("my_topic", 1, leader1.BrokerID()), + }) + + // leader0 says no longer leader of partition 0 + leader0.SetHandler(func(req *request) (res encoder) { + switch req.body.(type) { + case *FetchRequest: + fetchResponse := new(FetchResponse) + fetchResponse.AddError("my_topic", 0, ErrNotLeaderForPartition) + return fetchResponse + } + return nil + }) + + time.Sleep(50 * time.Millisecond) + Logger.Printf(" STAGE 3") + // Stage 3: + // * my_topic/0 -> leader1 serves 3 messages + // * my_topic/1 -> leader1 server 8 messages + + // leader1 provides 3 message on partition 0, and 8 messages on partition 1 + mockFetchResponse2 := newMockFetchResponse(t, 2) + for i := 4; i < 7; i++ { + mockFetchResponse2.SetMessage("my_topic", 0, int64(i), testMsg) + } + for i := 0; i < 8; i++ { + mockFetchResponse2.SetMessage("my_topic", 1, int64(i), testMsg) + } + leader1.SetHandlerByMap(map[string]MockResponse{ + "FetchRequest": mockFetchResponse2, + }) + + time.Sleep(50 * time.Millisecond) + Logger.Printf(" STAGE 4") + // Stage 4: + // * my_topic/0 -> leader1 serves 3 messages + // * my_topic/1 -> leader1 tells that it is no longer the leader + // * seedBroker tells that leader0 is a new leader for my_topic/1 + + // metadata assigns 0 to leader1 and 1 to leader0 + seedBroker.SetHandlerByMap(map[string]MockResponse{ + "MetadataRequest": newMockMetadataResponse(t). + SetLeader("my_topic", 0, leader1.BrokerID()). + SetLeader("my_topic", 1, leader0.BrokerID()), + }) + + // leader1 provides three more messages on partition0, says no longer leader of partition1 + mockFetchResponse3 := newMockFetchResponse(t, 3). + SetMessage("my_topic", 0, int64(7), testMsg). + SetMessage("my_topic", 0, int64(8), testMsg). + SetMessage("my_topic", 0, int64(9), testMsg) + leader1.SetHandler(func(req *request) (res encoder) { + switch reqBody := req.body.(type) { + case *FetchRequest: + res := mockFetchResponse3.For(reqBody).(*FetchResponse) + res.AddError("my_topic", 1, ErrNotLeaderForPartition) + return res + + } + return nil + }) + + // leader0 provides two messages on partition 1 + mockFetchResponse4 := newMockFetchResponse(t, 2) + for i := 8; i < 10; i++ { + mockFetchResponse4.SetMessage("my_topic", 1, int64(i), testMsg) + } + leader0.SetHandlerByMap(map[string]MockResponse{ + "FetchRequest": mockFetchResponse4, + }) + + wg.Wait() + safeClose(t, master) + leader1.Close() + leader0.Close() + seedBroker.Close() +} + +// When two partitions have the same broker as the leader, if one partition +// consumer channel buffer is full then that does not affect the ability to +// read messages by the other consumer. +func TestConsumerInterleavedClose(t *testing.T) { + // Given + broker0 := newMockBroker(t, 0) + broker0.SetHandlerByMap(map[string]MockResponse{ + "MetadataRequest": newMockMetadataResponse(t). + SetBroker(broker0.Addr(), broker0.BrokerID()). + SetLeader("my_topic", 0, broker0.BrokerID()). + SetLeader("my_topic", 1, broker0.BrokerID()), + "OffsetRequest": newMockOffsetResponse(t). + SetOffset("my_topic", 0, OffsetOldest, 1000). + SetOffset("my_topic", 0, OffsetNewest, 1100). + SetOffset("my_topic", 1, OffsetOldest, 2000). + SetOffset("my_topic", 1, OffsetNewest, 2100), + "FetchRequest": newMockFetchResponse(t, 1). + SetMessage("my_topic", 0, 1000, testMsg). + SetMessage("my_topic", 0, 1001, testMsg). + SetMessage("my_topic", 0, 1002, testMsg). + SetMessage("my_topic", 1, 2000, testMsg), + }) + + config := NewConfig() + config.ChannelBufferSize = 0 + master, err := NewConsumer([]string{broker0.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + c0, err := master.ConsumePartition("my_topic", 0, 1000) + if err != nil { + t.Fatal(err) + } + + c1, err := master.ConsumePartition("my_topic", 1, 2000) + if err != nil { + t.Fatal(err) + } + + // When/Then: we can read from partition 0 even if nobody reads from partition 1 + assertMessageOffset(t, <-c0.Messages(), 1000) + assertMessageOffset(t, <-c0.Messages(), 1001) + assertMessageOffset(t, <-c0.Messages(), 1002) + + safeClose(t, c1) + safeClose(t, c0) + safeClose(t, master) + broker0.Close() +} + +func TestConsumerBounceWithReferenceOpen(t *testing.T) { + broker0 := newMockBroker(t, 0) + broker0Addr := broker0.Addr() + broker1 := newMockBroker(t, 1) + + mockMetadataResponse := newMockMetadataResponse(t). + SetBroker(broker0.Addr(), broker0.BrokerID()). + SetBroker(broker1.Addr(), broker1.BrokerID()). + SetLeader("my_topic", 0, broker0.BrokerID()). + SetLeader("my_topic", 1, broker1.BrokerID()) + + mockOffsetResponse := newMockOffsetResponse(t). + SetOffset("my_topic", 0, OffsetOldest, 1000). + SetOffset("my_topic", 0, OffsetNewest, 1100). + SetOffset("my_topic", 1, OffsetOldest, 2000). + SetOffset("my_topic", 1, OffsetNewest, 2100) + + mockFetchResponse := newMockFetchResponse(t, 1) + for i := 0; i < 10; i++ { + mockFetchResponse.SetMessage("my_topic", 0, int64(1000+i), testMsg) + mockFetchResponse.SetMessage("my_topic", 1, int64(2000+i), testMsg) + } + + broker0.SetHandlerByMap(map[string]MockResponse{ + "OffsetRequest": mockOffsetResponse, + "FetchRequest": mockFetchResponse, + }) + broker1.SetHandlerByMap(map[string]MockResponse{ + "MetadataRequest": mockMetadataResponse, + "OffsetRequest": mockOffsetResponse, + "FetchRequest": mockFetchResponse, + }) + + config := NewConfig() + config.Consumer.Return.Errors = true + config.Consumer.Retry.Backoff = 100 * time.Millisecond + config.ChannelBufferSize = 1 + master, err := NewConsumer([]string{broker1.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + c0, err := master.ConsumePartition("my_topic", 0, 1000) + if err != nil { + t.Fatal(err) + } + + c1, err := master.ConsumePartition("my_topic", 1, 2000) + if err != nil { + t.Fatal(err) + } + + // read messages from both partition to make sure that both brokers operate + // normally. + assertMessageOffset(t, <-c0.Messages(), 1000) + assertMessageOffset(t, <-c1.Messages(), 2000) + + // Simulate broker shutdown. Note that metadata response does not change, + // that is the leadership does not move to another broker. So partition + // consumer will keep retrying to restore the connection with the broker. + broker0.Close() + + // Make sure that while the partition/0 leader is down, consumer/partition/1 + // is capable of pulling messages from broker1. + for i := 1; i < 7; i++ { + offset := (<-c1.Messages()).Offset + if offset != int64(2000+i) { + t.Errorf("Expected offset %d from consumer/partition/1", int64(2000+i)) + } + } + + // Bring broker0 back to service. + broker0 = newMockBrokerAddr(t, 0, broker0Addr) + broker0.SetHandlerByMap(map[string]MockResponse{ + "FetchRequest": mockFetchResponse, + }) + + // Read the rest of messages from both partitions. + for i := 7; i < 10; i++ { + assertMessageOffset(t, <-c1.Messages(), int64(2000+i)) + } + for i := 1; i < 10; i++ { + assertMessageOffset(t, <-c0.Messages(), int64(1000+i)) + } + + select { + case <-c0.Errors(): + default: + t.Errorf("Partition consumer should have detected broker restart") + } + + safeClose(t, c1) + safeClose(t, c0) + safeClose(t, master) + broker0.Close() + broker1.Close() +} + +func TestConsumerOffsetOutOfRange(t *testing.T) { + // Given + broker0 := newMockBroker(t, 2) + broker0.SetHandlerByMap(map[string]MockResponse{ + "MetadataRequest": newMockMetadataResponse(t). + SetBroker(broker0.Addr(), broker0.BrokerID()). + SetLeader("my_topic", 0, broker0.BrokerID()), + "OffsetRequest": newMockOffsetResponse(t). + SetOffset("my_topic", 0, OffsetNewest, 1234). + SetOffset("my_topic", 0, OffsetOldest, 2345), + }) + + master, err := NewConsumer([]string{broker0.Addr()}, nil) + if err != nil { + t.Fatal(err) + } + + // When/Then + if _, err := master.ConsumePartition("my_topic", 0, 0); err != ErrOffsetOutOfRange { + t.Fatal("Should return ErrOffsetOutOfRange, got:", err) + } + if _, err := master.ConsumePartition("my_topic", 0, 3456); err != ErrOffsetOutOfRange { + t.Fatal("Should return ErrOffsetOutOfRange, got:", err) + } + if _, err := master.ConsumePartition("my_topic", 0, -3); err != ErrOffsetOutOfRange { + t.Fatal("Should return ErrOffsetOutOfRange, got:", err) + } + + safeClose(t, master) + broker0.Close() +} + +func assertMessageOffset(t *testing.T, msg *ConsumerMessage, expectedOffset int64) { + if msg.Offset != expectedOffset { + t.Errorf("Incorrect message offset: expected=%d, actual=%d", expectedOffset, msg.Offset) + } +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/crc32_field.go b/Godeps/_workspace/src/github.com/Shopify/sarama/crc32_field.go new file mode 100644 index 0000000000000..f4fde18ad2529 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/crc32_field.go @@ -0,0 +1,35 @@ +package sarama + +import ( + "encoding/binary" + "hash/crc32" +) + +// crc32Field implements the pushEncoder and pushDecoder interfaces for calculating CRC32s. +type crc32Field struct { + startOffset int +} + +func (c *crc32Field) saveOffset(in int) { + c.startOffset = in +} + +func (c *crc32Field) reserveLength() int { + return 4 +} + +func (c *crc32Field) run(curOffset int, buf []byte) error { + crc := crc32.ChecksumIEEE(buf[c.startOffset+4 : curOffset]) + binary.BigEndian.PutUint32(buf[c.startOffset:], crc) + return nil +} + +func (c *crc32Field) check(curOffset int, buf []byte) error { + crc := crc32.ChecksumIEEE(buf[c.startOffset+4 : curOffset]) + + if crc != binary.BigEndian.Uint32(buf[c.startOffset:]) { + return PacketDecodingError{"CRC didn't match"} + } + + return nil +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/encoder_decoder.go b/Godeps/_workspace/src/github.com/Shopify/sarama/encoder_decoder.go new file mode 100644 index 0000000000000..b91efaa0ee60b --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/encoder_decoder.go @@ -0,0 +1,62 @@ +package sarama + +import "fmt" + +// Encoder is the interface that wraps the basic Encode method. +// Anything implementing Encoder can be turned into bytes using Kafka's encoding rules. +type encoder interface { + encode(pe packetEncoder) error +} + +// Encode takes an Encoder and turns it into bytes. +func encode(e encoder) ([]byte, error) { + if e == nil { + return nil, nil + } + + var prepEnc prepEncoder + var realEnc realEncoder + + err := e.encode(&prepEnc) + if err != nil { + return nil, err + } + + if prepEnc.length < 0 || prepEnc.length > int(MaxRequestSize) { + return nil, PacketEncodingError{fmt.Sprintf("invalid request size (%d)", prepEnc.length)} + } + + realEnc.raw = make([]byte, prepEnc.length) + err = e.encode(&realEnc) + if err != nil { + return nil, err + } + + return realEnc.raw, nil +} + +// Decoder is the interface that wraps the basic Decode method. +// Anything implementing Decoder can be extracted from bytes using Kafka's encoding rules. +type decoder interface { + decode(pd packetDecoder) error +} + +// Decode takes bytes and a Decoder and fills the fields of the decoder from the bytes, +// interpreted using Kafka's encoding rules. +func decode(buf []byte, in decoder) error { + if buf == nil { + return nil + } + + helper := realDecoder{raw: buf} + err := in.decode(&helper) + if err != nil { + return err + } + + if helper.off != len(buf) { + return PacketDecodingError{"invalid length"} + } + + return nil +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/errors.go b/Godeps/_workspace/src/github.com/Shopify/sarama/errors.go new file mode 100644 index 0000000000000..70f2b9bfdf914 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/errors.go @@ -0,0 +1,146 @@ +package sarama + +import ( + "errors" + "fmt" +) + +// ErrOutOfBrokers is the error returned when the client has run out of brokers to talk to because all of them errored +// or otherwise failed to respond. +var ErrOutOfBrokers = errors.New("kafka: client has run out of available brokers to talk to (Is your cluster reachable?)") + +// ErrClosedClient is the error returned when a method is called on a client that has been closed. +var ErrClosedClient = errors.New("kafka: tried to use a client that was closed") + +// ErrIncompleteResponse is the error returned when the server returns a syntactically valid response, but it does +// not contain the expected information. +var ErrIncompleteResponse = errors.New("kafka: response did not contain all the expected topic/partition blocks") + +// ErrInvalidPartition is the error returned when a partitioner returns an invalid partition index +// (meaning one outside of the range [0...numPartitions-1]). +var ErrInvalidPartition = errors.New("kafka: partitioner returned an invalid partition index") + +// ErrAlreadyConnected is the error returned when calling Open() on a Broker that is already connected or connecting. +var ErrAlreadyConnected = errors.New("kafka: broker connection already initiated") + +// ErrNotConnected is the error returned when trying to send or call Close() on a Broker that is not connected. +var ErrNotConnected = errors.New("kafka: broker not connected") + +// ErrInsufficientData is returned when decoding and the packet is truncated. This can be expected +// when requesting messages, since as an optimization the server is allowed to return a partial message at the end +// of the message set. +var ErrInsufficientData = errors.New("kafka: insufficient data to decode packet, more bytes expected") + +// ErrShuttingDown is returned when a producer receives a message during shutdown. +var ErrShuttingDown = errors.New("kafka: message received by producer in process of shutting down") + +// ErrMessageTooLarge is returned when the next message to consume is larger than the configured Consumer.Fetch.Max +var ErrMessageTooLarge = errors.New("kafka: message is larger than Consumer.Fetch.Max") + +// PacketEncodingError is returned from a failure while encoding a Kafka packet. This can happen, for example, +// if you try to encode a string over 2^15 characters in length, since Kafka's encoding rules do not permit that. +type PacketEncodingError struct { + Info string +} + +func (err PacketEncodingError) Error() string { + return fmt.Sprintf("kafka: error encoding packet: %s", err.Info) +} + +// PacketDecodingError is returned when there was an error (other than truncated data) decoding the Kafka broker's response. +// This can be a bad CRC or length field, or any other invalid value. +type PacketDecodingError struct { + Info string +} + +func (err PacketDecodingError) Error() string { + return fmt.Sprintf("kafka: error decoding packet: %s", err.Info) +} + +// ConfigurationError is the type of error returned from a constructor (e.g. NewClient, or NewConsumer) +// when the specified configuration is invalid. +type ConfigurationError string + +func (err ConfigurationError) Error() string { + return "kafka: invalid configuration (" + string(err) + ")" +} + +// KError is the type of error that can be returned directly by the Kafka broker. +// See https://cwiki.apache.org/confluence/display/KAFKA/A+Guide+To+The+Kafka+Protocol#AGuideToTheKafkaProtocol-ErrorCodes +type KError int16 + +// Numeric error codes returned by the Kafka server. +const ( + ErrNoError KError = 0 + ErrUnknown KError = -1 + ErrOffsetOutOfRange KError = 1 + ErrInvalidMessage KError = 2 + ErrUnknownTopicOrPartition KError = 3 + ErrInvalidMessageSize KError = 4 + ErrLeaderNotAvailable KError = 5 + ErrNotLeaderForPartition KError = 6 + ErrRequestTimedOut KError = 7 + ErrBrokerNotAvailable KError = 8 + ErrReplicaNotAvailable KError = 9 + ErrMessageSizeTooLarge KError = 10 + ErrStaleControllerEpochCode KError = 11 + ErrOffsetMetadataTooLarge KError = 12 + ErrOffsetsLoadInProgress KError = 14 + ErrConsumerCoordinatorNotAvailable KError = 15 + ErrNotCoordinatorForConsumer KError = 16 + ErrInvalidTopic KError = 17 + ErrMessageSetSizeTooLarge KError = 18 + ErrNotEnoughReplicas KError = 19 + ErrNotEnoughReplicasAfterAppend KError = 20 +) + +func (err KError) Error() string { + // Error messages stolen/adapted from + // https://cwiki.apache.org/confluence/display/KAFKA/A+Guide+To+The+Kafka+Protocol + switch err { + case ErrNoError: + return "kafka server: Not an error, why are you printing me?" + case ErrUnknown: + return "kafka server: Unexpected (unknown?) server error." + case ErrOffsetOutOfRange: + return "kafka server: The requested offset is outside the range of offsets maintained by the server for the given topic/partition." + case ErrInvalidMessage: + return "kafka server: Message contents does not match its CRC." + case ErrUnknownTopicOrPartition: + return "kafka server: Request was for a topic or partition that does not exist on this broker." + case ErrInvalidMessageSize: + return "kafka server: The message has a negative size." + case ErrLeaderNotAvailable: + return "kafka server: In the middle of a leadership election, there is currently no leader for this partition and hence it is unavailable for writes." + case ErrNotLeaderForPartition: + return "kafka server: Tried to send a message to a replica that is not the leader for some partition. Your metadata is out of date." + case ErrRequestTimedOut: + return "kafka server: Request exceeded the user-specified time limit in the request." + case ErrBrokerNotAvailable: + return "kafka server: Broker not available. Not a client facing error, we should never receive this!!!" + case ErrReplicaNotAvailable: + return "kafka server: Replica infomation not available, one or more brokers are down." + case ErrMessageSizeTooLarge: + return "kafka server: Message was too large, server rejected it to avoid allocation error." + case ErrStaleControllerEpochCode: + return "kafka server: StaleControllerEpochCode (internal error code for broker-to-broker communication)." + case ErrOffsetMetadataTooLarge: + return "kafka server: Specified a string larger than the configured maximum for offset metadata." + case ErrOffsetsLoadInProgress: + return "kafka server: The broker is still loading offsets after a leader change for that offset's topic partition." + case ErrConsumerCoordinatorNotAvailable: + return "kafka server: Offset's topic has not yet been created." + case ErrNotCoordinatorForConsumer: + return "kafka server: Request was for a consumer group that is not coordinated by this broker." + case ErrInvalidTopic: + return "kafka server: The request attempted to perform an operation on an invalid topic." + case ErrMessageSetSizeTooLarge: + return "kafka server: The request included message batch larger than the configured segment size on the server." + case ErrNotEnoughReplicas: + return "kafka server: Messages are rejected since there are fewer in-sync replicas than required." + case ErrNotEnoughReplicasAfterAppend: + return "kafka server: Messages are written to the log, but to fewer in-sync replicas than required." + } + + return fmt.Sprintf("Unknown error, how did this happen? Error code = %d", err) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/examples/README.md b/Godeps/_workspace/src/github.com/Shopify/sarama/examples/README.md new file mode 100644 index 0000000000000..b6588051eb3e4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/examples/README.md @@ -0,0 +1,9 @@ +# Sarama examples + +This folder contains example applications to demonstrate the use of Sarama. For code snippet examples on how to use the different types in Sarama, see [Sarams's API documentation on godoc.org](https://godoc.org/github.com/Shopify/sarama) + +In these examples, we use `github.com/Shopify/sarama` as import path. We do this to ensure all the examples are up to date with the latest changes in Sarama. For your own applications, you may want to use `gopkg.in/Shopify/sarama.v1` to lock into a stable API version. + +#### HTTP server + +[http_server](./http_server) is a simple HTTP server uses both the sync producer to produce data as part of the request handling cycle, as well as the async producer to maintain an access log. It also uses the [mocks subpackage](https://godoc.org/github.com/Shopify/sarama/mocks) to test both. diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/examples/http_server/.gitignore b/Godeps/_workspace/src/github.com/Shopify/sarama/examples/http_server/.gitignore new file mode 100644 index 0000000000000..9f6ed425f9c7a --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/examples/http_server/.gitignore @@ -0,0 +1,2 @@ +http_server +http_server.test diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/examples/http_server/README.md b/Godeps/_workspace/src/github.com/Shopify/sarama/examples/http_server/README.md new file mode 100644 index 0000000000000..5ff2bc2533b85 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/examples/http_server/README.md @@ -0,0 +1,7 @@ +# HTTP server example + +This HTTP server example shows you how to use the AsyncProducer and SyncProducer, and how to test them using mocks. The server simply sends the data of the HTTP request's query string to Kafka, and send a 200 result if that succeeds. For every request, it will send an access log entry to Kafka as well in the background. + +If you need to know whether a message was successfully sent to the Kafka cluster before you can send your HTTP response, using the `SyncProducer` is probably the simplest way to achieve this. If you don't care, e.g. for the access log, using the `AsyncProducer` will let you fire and forget. You can send the HTTP response, while the message is being produced in the background. + +One important thing to note is that both the `SyncProducer` and `AsyncProducer` are **thread-safe**. Go's `http.Server` handles requests concurrently in different goroutines, but you can use a single producer safely. This will actually achieve efficiency gains as the producer will be able to batch messages from concurrent requests together. diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/examples/http_server/http_server.go b/Godeps/_workspace/src/github.com/Shopify/sarama/examples/http_server/http_server.go new file mode 100644 index 0000000000000..03e47b6b23677 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/examples/http_server/http_server.go @@ -0,0 +1,246 @@ +package main + +import ( + "github.com/Shopify/sarama" + + "crypto/tls" + "crypto/x509" + "encoding/json" + "flag" + "fmt" + "io/ioutil" + "log" + "net/http" + "os" + "strings" + "time" +) + +var ( + addr = flag.String("addr", ":8080", "The address to bind to") + brokers = flag.String("brokers", os.Getenv("KAFKA_PEERS"), "The Kafka brokers to connect to, as a comma separated list") + verbose = flag.Bool("verbose", false, "Turn on Sarama logging") + certFile = flag.String("certificate", "", "The optional certificate file for client authentication") + keyFile = flag.String("key", "", "The optional key file for client authentication") + caFile = flag.String("ca", "", "The optional certificate authority file for TLS client authentication") + verifySsl = flag.Bool("verify", false, "Optional verify ssl certificates chain") +) + +func main() { + flag.Parse() + + if *verbose { + sarama.Logger = log.New(os.Stdout, "[sarama] ", log.LstdFlags) + } + + if *brokers == "" { + flag.PrintDefaults() + os.Exit(1) + } + + brokerList := strings.Split(*brokers, ",") + log.Printf("Kafka brokers: %s", strings.Join(brokerList, ", ")) + + server := &Server{ + DataCollector: newDataCollector(brokerList), + AccessLogProducer: newAccessLogProducer(brokerList), + } + defer func() { + if err := server.Close(); err != nil { + log.Println("Failed to close server", err) + } + }() + + log.Fatal(server.Run(*addr)) +} + +func createTlsConfiguration() (t *tls.Config) { + if *certFile != "" && *keyFile != "" && *caFile != "" { + cert, err := tls.LoadX509KeyPair(*certFile, *keyFile) + if err != nil { + log.Fatal(err) + } + + caCert, err := ioutil.ReadFile(*caFile) + if err != nil { + log.Fatal(err) + } + + caCertPool := x509.NewCertPool() + caCertPool.AppendCertsFromPEM(caCert) + + t = &tls.Config{ + Certificates: []tls.Certificate{cert}, + RootCAs: caCertPool, + InsecureSkipVerify: *verifySsl, + } + } + // will be nil by default if nothing is provided + return t +} + +type Server struct { + DataCollector sarama.SyncProducer + AccessLogProducer sarama.AsyncProducer +} + +func (s *Server) Close() error { + if err := s.DataCollector.Close(); err != nil { + log.Println("Failed to shut down data collector cleanly", err) + } + + if err := s.AccessLogProducer.Close(); err != nil { + log.Println("Failed to shut down access log producer cleanly", err) + } + + return nil +} + +func (s *Server) Handler() http.Handler { + return s.withAccessLog(s.collectQueryStringData()) +} + +func (s *Server) Run(addr string) error { + httpServer := &http.Server{ + Addr: addr, + Handler: s.Handler(), + } + + log.Printf("Listening for requests on %s...\n", addr) + return httpServer.ListenAndServe() +} + +func (s *Server) collectQueryStringData() http.Handler { + return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + if r.URL.Path != "/" { + http.NotFound(w, r) + return + } + + // We are not setting a message key, which means that all messages will + // be distributed randomly over the different partitions. + partition, offset, err := s.DataCollector.SendMessage(&sarama.ProducerMessage{ + Topic: "important", + Value: sarama.StringEncoder(r.URL.RawQuery), + }) + + if err != nil { + w.WriteHeader(http.StatusInternalServerError) + fmt.Fprintf(w, "Failed to store your data:, %s", err) + } else { + // The tuple (topic, partition, offset) can be used as a unique identifier + // for a message in a Kafka cluster. + fmt.Fprintf(w, "Your data is stored with unique identifier important/%d/%d", partition, offset) + } + }) +} + +type accessLogEntry struct { + Method string `json:"method"` + Host string `json:"host"` + Path string `json:"path"` + IP string `json:"ip"` + ResponseTime float64 `json:"response_time"` + + encoded []byte + err error +} + +func (ale *accessLogEntry) ensureEncoded() { + if ale.encoded == nil && ale.err == nil { + ale.encoded, ale.err = json.Marshal(ale) + } +} + +func (ale *accessLogEntry) Length() int { + ale.ensureEncoded() + return len(ale.encoded) +} + +func (ale *accessLogEntry) Encode() ([]byte, error) { + ale.ensureEncoded() + return ale.encoded, ale.err +} + +func (s *Server) withAccessLog(next http.Handler) http.Handler { + + return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + started := time.Now() + + next.ServeHTTP(w, r) + + entry := &accessLogEntry{ + Method: r.Method, + Host: r.Host, + Path: r.RequestURI, + IP: r.RemoteAddr, + ResponseTime: float64(time.Since(started)) / float64(time.Second), + } + + // We will use the client's IP address as key. This will cause + // all the access log entries of the same IP address to end up + // on the same partition. + s.AccessLogProducer.Input() <- &sarama.ProducerMessage{ + Topic: "access_log", + Key: sarama.StringEncoder(r.RemoteAddr), + Value: entry, + } + }) +} + +func newDataCollector(brokerList []string) sarama.SyncProducer { + + // For the data collector, we are looking for strong consistency semantics. + // Because we don't change the flush settings, sarama will try to produce messages + // as fast as possible to keep latency low. + config := sarama.NewConfig() + config.Producer.RequiredAcks = sarama.WaitForAll // Wait for all in-sync replicas to ack the message + config.Producer.Retry.Max = 10 // Retry up to 10 times to produce the message + tlsConfig := createTlsConfiguration() + if tlsConfig != nil { + config.Net.TLS.Config = tlsConfig + config.Net.TLS.Enable = true + } + + // On the broker side, you may want to change the following settings to get + // stronger consistency guarantees: + // - For your broker, set `unclean.leader.election.enable` to false + // - For the topic, you could increase `min.insync.replicas`. + + producer, err := sarama.NewSyncProducer(brokerList, config) + if err != nil { + log.Fatalln("Failed to start Sarama producer:", err) + } + + return producer +} + +func newAccessLogProducer(brokerList []string) sarama.AsyncProducer { + + // For the access log, we are looking for AP semantics, with high throughput. + // By creating batches of compressed messages, we reduce network I/O at a cost of more latency. + config := sarama.NewConfig() + tlsConfig := createTlsConfiguration() + if tlsConfig != nil { + config.Net.TLS.Enable = true + config.Net.TLS.Config = tlsConfig + } + config.Producer.RequiredAcks = sarama.WaitForLocal // Only wait for the leader to ack + config.Producer.Compression = sarama.CompressionSnappy // Compress messages + config.Producer.Flush.Frequency = 500 * time.Millisecond // Flush batches every 500ms + + producer, err := sarama.NewAsyncProducer(brokerList, config) + if err != nil { + log.Fatalln("Failed to start Sarama producer:", err) + } + + // We will just log to STDOUT if we're not able to produce messages. + // Note: messages will only be returned here after all retry attempts are exhausted. + go func() { + for err := range producer.Errors() { + log.Println("Failed to write access log entry:", err) + } + }() + + return producer +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/examples/http_server/http_server_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/examples/http_server/http_server_test.go new file mode 100644 index 0000000000000..7b2451e282cc3 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/examples/http_server/http_server_test.go @@ -0,0 +1,109 @@ +package main + +import ( + "io" + "net/http" + "net/http/httptest" + "testing" + + "github.com/Shopify/sarama" + "github.com/Shopify/sarama/mocks" +) + +// In normal operation, we expect one access log entry, +// and one data collector entry. Let's assume both will succeed. +// We should return a HTTP 200 status. +func TestCollectSuccessfully(t *testing.T) { + dataCollectorMock := mocks.NewSyncProducer(t, nil) + dataCollectorMock.ExpectSendMessageAndSucceed() + + accessLogProducerMock := mocks.NewAsyncProducer(t, nil) + accessLogProducerMock.ExpectInputAndSucceed() + + // Now, use dependency injection to use the mocks. + s := &Server{ + DataCollector: dataCollectorMock, + AccessLogProducer: accessLogProducerMock, + } + + // The Server's Close call is important; it will call Close on + // the two mock producers, which will then validate whether all + // expectations are resolved. + defer safeClose(t, s) + + req, err := http.NewRequest("GET", "http://example.com/?data", nil) + if err != nil { + t.Fatal(err) + } + res := httptest.NewRecorder() + s.Handler().ServeHTTP(res, req) + + if res.Code != 200 { + t.Errorf("Expected HTTP status 200, found %d", res.Code) + } + + if string(res.Body.Bytes()) != "Your data is stored with unique identifier important/0/1" { + t.Error("Unexpected response body", res.Body) + } +} + +// Now, let's see if we handle the case of not being able to produce +// to the data collector properly. In this case we should return a 500 status. +func TestCollectionFailure(t *testing.T) { + dataCollectorMock := mocks.NewSyncProducer(t, nil) + dataCollectorMock.ExpectSendMessageAndFail(sarama.ErrRequestTimedOut) + + accessLogProducerMock := mocks.NewAsyncProducer(t, nil) + accessLogProducerMock.ExpectInputAndSucceed() + + s := &Server{ + DataCollector: dataCollectorMock, + AccessLogProducer: accessLogProducerMock, + } + defer safeClose(t, s) + + req, err := http.NewRequest("GET", "http://example.com/?data", nil) + if err != nil { + t.Fatal(err) + } + res := httptest.NewRecorder() + s.Handler().ServeHTTP(res, req) + + if res.Code != 500 { + t.Errorf("Expected HTTP status 500, found %d", res.Code) + } +} + +// We don't expect any data collector calls because the path is wrong, +// so we are not setting any expectations on the dataCollectorMock. It +// will still generate an access log entry though. +func TestWrongPath(t *testing.T) { + dataCollectorMock := mocks.NewSyncProducer(t, nil) + + accessLogProducerMock := mocks.NewAsyncProducer(t, nil) + accessLogProducerMock.ExpectInputAndSucceed() + + s := &Server{ + DataCollector: dataCollectorMock, + AccessLogProducer: accessLogProducerMock, + } + defer safeClose(t, s) + + req, err := http.NewRequest("GET", "http://example.com/wrong?data", nil) + if err != nil { + t.Fatal(err) + } + res := httptest.NewRecorder() + + s.Handler().ServeHTTP(res, req) + + if res.Code != 404 { + t.Errorf("Expected HTTP status 404, found %d", res.Code) + } +} + +func safeClose(t *testing.T, o io.Closer) { + if err := o.Close(); err != nil { + t.Error(err) + } +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/fetch_request.go b/Godeps/_workspace/src/github.com/Shopify/sarama/fetch_request.go new file mode 100644 index 0000000000000..3c00fad653025 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/fetch_request.go @@ -0,0 +1,123 @@ +package sarama + +type fetchRequestBlock struct { + fetchOffset int64 + maxBytes int32 +} + +func (f *fetchRequestBlock) encode(pe packetEncoder) error { + pe.putInt64(f.fetchOffset) + pe.putInt32(f.maxBytes) + return nil +} + +func (f *fetchRequestBlock) decode(pd packetDecoder) (err error) { + if f.fetchOffset, err = pd.getInt64(); err != nil { + return err + } + if f.maxBytes, err = pd.getInt32(); err != nil { + return err + } + return nil +} + +type FetchRequest struct { + MaxWaitTime int32 + MinBytes int32 + blocks map[string]map[int32]*fetchRequestBlock +} + +func (f *FetchRequest) encode(pe packetEncoder) (err error) { + pe.putInt32(-1) // replica ID is always -1 for clients + pe.putInt32(f.MaxWaitTime) + pe.putInt32(f.MinBytes) + err = pe.putArrayLength(len(f.blocks)) + if err != nil { + return err + } + for topic, blocks := range f.blocks { + err = pe.putString(topic) + if err != nil { + return err + } + err = pe.putArrayLength(len(blocks)) + if err != nil { + return err + } + for partition, block := range blocks { + pe.putInt32(partition) + err = block.encode(pe) + if err != nil { + return err + } + } + } + return nil +} + +func (f *FetchRequest) decode(pd packetDecoder) (err error) { + if _, err = pd.getInt32(); err != nil { + return err + } + if f.MaxWaitTime, err = pd.getInt32(); err != nil { + return err + } + if f.MinBytes, err = pd.getInt32(); err != nil { + return err + } + topicCount, err := pd.getArrayLength() + if err != nil { + return err + } + if topicCount == 0 { + return nil + } + f.blocks = make(map[string]map[int32]*fetchRequestBlock) + for i := 0; i < topicCount; i++ { + topic, err := pd.getString() + if err != nil { + return err + } + partitionCount, err := pd.getArrayLength() + if err != nil { + return err + } + f.blocks[topic] = make(map[int32]*fetchRequestBlock) + for j := 0; j < partitionCount; j++ { + partition, err := pd.getInt32() + if err != nil { + return err + } + fetchBlock := &fetchRequestBlock{} + if err = fetchBlock.decode(pd); err != nil { + return nil + } + f.blocks[topic][partition] = fetchBlock + } + } + return nil +} + +func (f *FetchRequest) key() int16 { + return 1 +} + +func (f *FetchRequest) version() int16 { + return 0 +} + +func (f *FetchRequest) AddBlock(topic string, partitionID int32, fetchOffset int64, maxBytes int32) { + if f.blocks == nil { + f.blocks = make(map[string]map[int32]*fetchRequestBlock) + } + + if f.blocks[topic] == nil { + f.blocks[topic] = make(map[int32]*fetchRequestBlock) + } + + tmp := new(fetchRequestBlock) + tmp.maxBytes = maxBytes + tmp.fetchOffset = fetchOffset + + f.blocks[topic][partitionID] = tmp +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/fetch_request_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/fetch_request_test.go new file mode 100644 index 0000000000000..32c083c7d3ef7 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/fetch_request_test.go @@ -0,0 +1,34 @@ +package sarama + +import "testing" + +var ( + fetchRequestNoBlocks = []byte{ + 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00} + + fetchRequestWithProperties = []byte{ + 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x20, 0x00, 0x00, 0x00, 0xEF, + 0x00, 0x00, 0x00, 0x00} + + fetchRequestOneBlock = []byte{ + 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x05, 't', 'o', 'p', 'i', 'c', + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x00, 0x12, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x34, 0x00, 0x00, 0x00, 0x56} +) + +func TestFetchRequest(t *testing.T) { + request := new(FetchRequest) + testRequest(t, "no blocks", request, fetchRequestNoBlocks) + + request.MaxWaitTime = 0x20 + request.MinBytes = 0xEF + testRequest(t, "with properties", request, fetchRequestWithProperties) + + request.MaxWaitTime = 0 + request.MinBytes = 0 + request.AddBlock("topic", 0x12, 0x34, 0x56) + testRequest(t, "one block", request, fetchRequestOneBlock) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/fetch_response.go b/Godeps/_workspace/src/github.com/Shopify/sarama/fetch_response.go new file mode 100644 index 0000000000000..1ac5439214230 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/fetch_response.go @@ -0,0 +1,173 @@ +package sarama + +type FetchResponseBlock struct { + Err KError + HighWaterMarkOffset int64 + MsgSet MessageSet +} + +func (pr *FetchResponseBlock) decode(pd packetDecoder) (err error) { + tmp, err := pd.getInt16() + if err != nil { + return err + } + pr.Err = KError(tmp) + + pr.HighWaterMarkOffset, err = pd.getInt64() + if err != nil { + return err + } + + msgSetSize, err := pd.getInt32() + if err != nil { + return err + } + + msgSetDecoder, err := pd.getSubset(int(msgSetSize)) + if err != nil { + return err + } + err = (&pr.MsgSet).decode(msgSetDecoder) + + return err +} + +type FetchResponse struct { + Blocks map[string]map[int32]*FetchResponseBlock +} + +func (pr *FetchResponseBlock) encode(pe packetEncoder) (err error) { + pe.putInt16(int16(pr.Err)) + + pe.putInt64(pr.HighWaterMarkOffset) + + pe.push(&lengthField{}) + err = pr.MsgSet.encode(pe) + if err != nil { + return err + } + return pe.pop() +} + +func (fr *FetchResponse) decode(pd packetDecoder) (err error) { + numTopics, err := pd.getArrayLength() + if err != nil { + return err + } + + fr.Blocks = make(map[string]map[int32]*FetchResponseBlock, numTopics) + for i := 0; i < numTopics; i++ { + name, err := pd.getString() + if err != nil { + return err + } + + numBlocks, err := pd.getArrayLength() + if err != nil { + return err + } + + fr.Blocks[name] = make(map[int32]*FetchResponseBlock, numBlocks) + + for j := 0; j < numBlocks; j++ { + id, err := pd.getInt32() + if err != nil { + return err + } + + block := new(FetchResponseBlock) + err = block.decode(pd) + if err != nil { + return err + } + fr.Blocks[name][id] = block + } + } + + return nil +} + +func (fr *FetchResponse) encode(pe packetEncoder) (err error) { + err = pe.putArrayLength(len(fr.Blocks)) + if err != nil { + return err + } + + for topic, partitions := range fr.Blocks { + err = pe.putString(topic) + if err != nil { + return err + } + + err = pe.putArrayLength(len(partitions)) + if err != nil { + return err + } + + for id, block := range partitions { + pe.putInt32(id) + err = block.encode(pe) + if err != nil { + return err + } + } + + } + return nil +} + +func (fr *FetchResponse) GetBlock(topic string, partition int32) *FetchResponseBlock { + if fr.Blocks == nil { + return nil + } + + if fr.Blocks[topic] == nil { + return nil + } + + return fr.Blocks[topic][partition] +} + +func (fr *FetchResponse) AddError(topic string, partition int32, err KError) { + if fr.Blocks == nil { + fr.Blocks = make(map[string]map[int32]*FetchResponseBlock) + } + partitions, ok := fr.Blocks[topic] + if !ok { + partitions = make(map[int32]*FetchResponseBlock) + fr.Blocks[topic] = partitions + } + frb, ok := partitions[partition] + if !ok { + frb = new(FetchResponseBlock) + partitions[partition] = frb + } + frb.Err = err +} + +func (fr *FetchResponse) AddMessage(topic string, partition int32, key, value Encoder, offset int64) { + if fr.Blocks == nil { + fr.Blocks = make(map[string]map[int32]*FetchResponseBlock) + } + partitions, ok := fr.Blocks[topic] + if !ok { + partitions = make(map[int32]*FetchResponseBlock) + fr.Blocks[topic] = partitions + } + frb, ok := partitions[partition] + if !ok { + frb = new(FetchResponseBlock) + partitions[partition] = frb + } + var kb []byte + var vb []byte + if key != nil { + kb, _ = key.Encode() + } + if value != nil { + vb, _ = value.Encode() + } + msg := &Message{Key: kb, Value: vb} + msgBlock := &MessageBlock{Msg: msg, Offset: offset} + frb.MsgSet.Messages = append(frb.MsgSet.Messages, msgBlock) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/fetch_response_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/fetch_response_test.go new file mode 100644 index 0000000000000..a23a05340f812 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/fetch_response_test.go @@ -0,0 +1,84 @@ +package sarama + +import ( + "bytes" + "testing" +) + +var ( + emptyFetchResponse = []byte{ + 0x00, 0x00, 0x00, 0x00} + + oneMessageFetchResponse = []byte{ + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x05, 't', 'o', 'p', 'i', 'c', + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x00, 0x05, + 0x00, 0x01, + 0x00, 0x00, 0x00, 0x00, 0x10, 0x10, 0x10, 0x10, + 0x00, 0x00, 0x00, 0x1C, + // messageSet + 0x00, 0x00, 0x00, 0x00, 0x00, 0x55, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x10, + // message + 0x23, 0x96, 0x4a, 0xf7, // CRC + 0x00, + 0x00, + 0xFF, 0xFF, 0xFF, 0xFF, + 0x00, 0x00, 0x00, 0x02, 0x00, 0xEE} +) + +func TestEmptyFetchResponse(t *testing.T) { + response := FetchResponse{} + testDecodable(t, "empty", &response, emptyFetchResponse) + + if len(response.Blocks) != 0 { + t.Error("Decoding produced topic blocks where there were none.") + } + +} + +func TestOneMessageFetchResponse(t *testing.T) { + response := FetchResponse{} + testDecodable(t, "one message", &response, oneMessageFetchResponse) + + if len(response.Blocks) != 1 { + t.Fatal("Decoding produced incorrect number of topic blocks.") + } + + if len(response.Blocks["topic"]) != 1 { + t.Fatal("Decoding produced incorrect number of partition blocks for topic.") + } + + block := response.GetBlock("topic", 5) + if block == nil { + t.Fatal("GetBlock didn't return block.") + } + if block.Err != ErrOffsetOutOfRange { + t.Error("Decoding didn't produce correct error code.") + } + if block.HighWaterMarkOffset != 0x10101010 { + t.Error("Decoding didn't produce correct high water mark offset.") + } + if block.MsgSet.PartialTrailingMessage { + t.Error("Decoding detected a partial trailing message where there wasn't one.") + } + + if len(block.MsgSet.Messages) != 1 { + t.Fatal("Decoding produced incorrect number of messages.") + } + msgBlock := block.MsgSet.Messages[0] + if msgBlock.Offset != 0x550000 { + t.Error("Decoding produced incorrect message offset.") + } + msg := msgBlock.Msg + if msg.Codec != CompressionNone { + t.Error("Decoding produced incorrect message compression.") + } + if msg.Key != nil { + t.Error("Decoding produced message key where there was none.") + } + if !bytes.Equal(msg.Value, []byte{0x00, 0xEE}) { + t.Error("Decoding produced incorrect message value.") + } +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/functional_client_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/functional_client_test.go new file mode 100644 index 0000000000000..9e8e32968d2b3 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/functional_client_test.go @@ -0,0 +1,90 @@ +package sarama + +import ( + "fmt" + "testing" + "time" +) + +func TestFuncConnectionFailure(t *testing.T) { + setupFunctionalTest(t) + defer teardownFunctionalTest(t) + + Proxies["kafka1"].Enabled = false + SaveProxy(t, "kafka1") + + config := NewConfig() + config.Metadata.Retry.Max = 1 + + _, err := NewClient([]string{kafkaBrokers[0]}, config) + if err != ErrOutOfBrokers { + t.Fatal("Expected returned error to be ErrOutOfBrokers, but was: ", err) + } +} + +func TestFuncClientMetadata(t *testing.T) { + setupFunctionalTest(t) + defer teardownFunctionalTest(t) + + config := NewConfig() + config.Metadata.Retry.Max = 1 + config.Metadata.Retry.Backoff = 10 * time.Millisecond + client, err := NewClient(kafkaBrokers, config) + if err != nil { + t.Fatal(err) + } + + if err := client.RefreshMetadata("unknown_topic"); err != ErrUnknownTopicOrPartition { + t.Error("Expected ErrUnknownTopicOrPartition, got", err) + } + + if _, err := client.Leader("unknown_topic", 0); err != ErrUnknownTopicOrPartition { + t.Error("Expected ErrUnknownTopicOrPartition, got", err) + } + + if _, err := client.Replicas("invalid/topic", 0); err != ErrUnknownTopicOrPartition { + t.Error("Expected ErrUnknownTopicOrPartition, got", err) + } + + partitions, err := client.Partitions("test.4") + if err != nil { + t.Error(err) + } + if len(partitions) != 4 { + t.Errorf("Expected test.4 topic to have 4 partitions, found %v", partitions) + } + + partitions, err = client.Partitions("test.1") + if err != nil { + t.Error(err) + } + if len(partitions) != 1 { + t.Errorf("Expected test.1 topic to have 1 partitions, found %v", partitions) + } + + safeClose(t, client) +} + +func TestFuncClientCoordinator(t *testing.T) { + checkKafkaVersion(t, "0.8.2") + setupFunctionalTest(t) + defer teardownFunctionalTest(t) + + client, err := NewClient(kafkaBrokers, nil) + if err != nil { + t.Fatal(err) + } + + for i := 0; i < 10; i++ { + broker, err := client.Coordinator(fmt.Sprintf("another_new_consumer_group_%d", i)) + if err != nil { + t.Error(err) + } + + if connected, err := broker.Connected(); !connected || err != nil { + t.Errorf("Expected to coordinator %s broker to be properly connected.", broker.Addr()) + } + } + + safeClose(t, client) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/functional_consumer_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/functional_consumer_test.go new file mode 100644 index 0000000000000..ab84331091167 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/functional_consumer_test.go @@ -0,0 +1,61 @@ +package sarama + +import ( + "math" + "testing" +) + +func TestFuncConsumerOffsetOutOfRange(t *testing.T) { + setupFunctionalTest(t) + defer teardownFunctionalTest(t) + + consumer, err := NewConsumer(kafkaBrokers, nil) + if err != nil { + t.Fatal(err) + } + + if _, err := consumer.ConsumePartition("test.1", 0, -10); err != ErrOffsetOutOfRange { + t.Error("Expected ErrOffsetOutOfRange, got:", err) + } + + if _, err := consumer.ConsumePartition("test.1", 0, math.MaxInt64); err != ErrOffsetOutOfRange { + t.Error("Expected ErrOffsetOutOfRange, got:", err) + } + + safeClose(t, consumer) +} + +func TestConsumerHighWaterMarkOffset(t *testing.T) { + setupFunctionalTest(t) + defer teardownFunctionalTest(t) + + p, err := NewSyncProducer(kafkaBrokers, nil) + if err != nil { + t.Fatal(err) + } + defer safeClose(t, p) + + _, offset, err := p.SendMessage(&ProducerMessage{Topic: "test.1", Value: StringEncoder("Test")}) + if err != nil { + t.Fatal(err) + } + + c, err := NewConsumer(kafkaBrokers, nil) + if err != nil { + t.Fatal(err) + } + defer safeClose(t, c) + + pc, err := c.ConsumePartition("test.1", 0, OffsetOldest) + if err != nil { + t.Fatal(err) + } + + <-pc.Messages() + + if hwmo := pc.HighWaterMarkOffset(); hwmo != offset+1 { + t.Logf("Last produced offset %d; high water mark should be one higher but found %d.", offset, hwmo) + } + + safeClose(t, pc) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/functional_producer_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/functional_producer_test.go new file mode 100644 index 0000000000000..1504e76005bc5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/functional_producer_test.go @@ -0,0 +1,203 @@ +package sarama + +import ( + "fmt" + "sync" + "testing" + "time" +) + +const TestBatchSize = 1000 + +func TestFuncProducing(t *testing.T) { + config := NewConfig() + testProducingMessages(t, config) +} + +func TestFuncProducingGzip(t *testing.T) { + config := NewConfig() + config.Producer.Compression = CompressionGZIP + testProducingMessages(t, config) +} + +func TestFuncProducingSnappy(t *testing.T) { + config := NewConfig() + config.Producer.Compression = CompressionSnappy + testProducingMessages(t, config) +} + +func TestFuncProducingNoResponse(t *testing.T) { + config := NewConfig() + config.Producer.RequiredAcks = NoResponse + testProducingMessages(t, config) +} + +func TestFuncProducingFlushing(t *testing.T) { + config := NewConfig() + config.Producer.Flush.Messages = TestBatchSize / 8 + config.Producer.Flush.Frequency = 250 * time.Millisecond + testProducingMessages(t, config) +} + +func TestFuncMultiPartitionProduce(t *testing.T) { + setupFunctionalTest(t) + defer teardownFunctionalTest(t) + + config := NewConfig() + config.ChannelBufferSize = 20 + config.Producer.Flush.Frequency = 50 * time.Millisecond + config.Producer.Flush.Messages = 200 + config.Producer.Return.Successes = true + producer, err := NewSyncProducer(kafkaBrokers, config) + if err != nil { + t.Fatal(err) + } + + var wg sync.WaitGroup + wg.Add(TestBatchSize) + + for i := 1; i <= TestBatchSize; i++ { + go func(i int) { + defer wg.Done() + msg := &ProducerMessage{Topic: "test.64", Key: nil, Value: StringEncoder(fmt.Sprintf("hur %d", i))} + if _, _, err := producer.SendMessage(msg); err != nil { + t.Error(i, err) + } + }(i) + } + + wg.Wait() + if err := producer.Close(); err != nil { + t.Error(err) + } +} + +func TestFuncProducingToInvalidTopic(t *testing.T) { + setupFunctionalTest(t) + defer teardownFunctionalTest(t) + + producer, err := NewSyncProducer(kafkaBrokers, nil) + if err != nil { + t.Fatal(err) + } + + if _, _, err := producer.SendMessage(&ProducerMessage{Topic: "in/valid"}); err != ErrUnknownTopicOrPartition { + t.Error("Expected ErrUnknownTopicOrPartition, found", err) + } + + if _, _, err := producer.SendMessage(&ProducerMessage{Topic: "in/valid"}); err != ErrUnknownTopicOrPartition { + t.Error("Expected ErrUnknownTopicOrPartition, found", err) + } + + safeClose(t, producer) +} + +func testProducingMessages(t *testing.T, config *Config) { + setupFunctionalTest(t) + defer teardownFunctionalTest(t) + + config.Producer.Return.Successes = true + config.Consumer.Return.Errors = true + + client, err := NewClient(kafkaBrokers, config) + if err != nil { + t.Fatal(err) + } + + master, err := NewConsumerFromClient(client) + if err != nil { + t.Fatal(err) + } + consumer, err := master.ConsumePartition("test.1", 0, OffsetNewest) + if err != nil { + t.Fatal(err) + } + + producer, err := NewAsyncProducerFromClient(client) + if err != nil { + t.Fatal(err) + } + + expectedResponses := TestBatchSize + for i := 1; i <= TestBatchSize; { + msg := &ProducerMessage{Topic: "test.1", Key: nil, Value: StringEncoder(fmt.Sprintf("testing %d", i))} + select { + case producer.Input() <- msg: + i++ + case ret := <-producer.Errors(): + t.Fatal(ret.Err) + case <-producer.Successes(): + expectedResponses-- + } + } + for expectedResponses > 0 { + select { + case ret := <-producer.Errors(): + t.Fatal(ret.Err) + case <-producer.Successes(): + expectedResponses-- + } + } + safeClose(t, producer) + + for i := 1; i <= TestBatchSize; i++ { + select { + case <-time.After(10 * time.Second): + t.Fatal("Not received any more events in the last 10 seconds.") + + case err := <-consumer.Errors(): + t.Error(err) + + case message := <-consumer.Messages(): + if string(message.Value) != fmt.Sprintf("testing %d", i) { + t.Fatalf("Unexpected message with index %d: %s", i, message.Value) + } + } + + } + safeClose(t, consumer) + safeClose(t, client) +} + +// Benchmarks + +func BenchmarkProducerSmall(b *testing.B) { + benchmarkProducer(b, nil, "test.64", ByteEncoder(make([]byte, 128))) +} +func BenchmarkProducerMedium(b *testing.B) { + benchmarkProducer(b, nil, "test.64", ByteEncoder(make([]byte, 1024))) +} +func BenchmarkProducerLarge(b *testing.B) { + benchmarkProducer(b, nil, "test.64", ByteEncoder(make([]byte, 8192))) +} +func BenchmarkProducerSmallSinglePartition(b *testing.B) { + benchmarkProducer(b, nil, "test.1", ByteEncoder(make([]byte, 128))) +} +func BenchmarkProducerMediumSnappy(b *testing.B) { + conf := NewConfig() + conf.Producer.Compression = CompressionSnappy + benchmarkProducer(b, conf, "test.1", ByteEncoder(make([]byte, 1024))) +} + +func benchmarkProducer(b *testing.B, conf *Config, topic string, value Encoder) { + setupFunctionalTest(b) + defer teardownFunctionalTest(b) + + producer, err := NewAsyncProducer(kafkaBrokers, conf) + if err != nil { + b.Fatal(err) + } + + b.ResetTimer() + + for i := 1; i <= b.N; { + msg := &ProducerMessage{Topic: topic, Key: StringEncoder(fmt.Sprintf("%d", i)), Value: value} + select { + case producer.Input() <- msg: + i++ + case ret := <-producer.Errors(): + b.Fatal(ret.Err) + } + } + safeClose(b, producer) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/functional_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/functional_test.go new file mode 100644 index 0000000000000..171002ee99404 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/functional_test.go @@ -0,0 +1,146 @@ +package sarama + +import ( + "log" + "math/rand" + "net" + "os" + "strconv" + "strings" + "testing" + "time" + + toxiproxy "github.com/Shopify/toxiproxy/client" +) + +const ( + VagrantToxiproxy = "http://192.168.100.67:8474" + VagrantKafkaPeers = "192.168.100.67:9091,192.168.100.67:9092,192.168.100.67:9093,192.168.100.67:9094,192.168.100.67:9095" + VagrantZookeeperPeers = "192.168.100.67:2181,192.168.100.67:2182,192.168.100.67:2183,192.168.100.67:2184,192.168.100.67:2185" +) + +var ( + kafkaAvailable, kafkaRequired bool + kafkaBrokers []string + + proxyClient *toxiproxy.Client + Proxies map[string]*toxiproxy.Proxy + ZKProxies = []string{"zk1", "zk2", "zk3", "zk4", "zk5"} + KafkaProxies = []string{"kafka1", "kafka2", "kafka3", "kafka4", "kafka5"} +) + +func init() { + if os.Getenv("DEBUG") == "true" { + Logger = log.New(os.Stdout, "[sarama] ", log.LstdFlags) + } + + seed := time.Now().UTC().UnixNano() + if tmp := os.Getenv("TEST_SEED"); tmp != "" { + seed, _ = strconv.ParseInt(tmp, 0, 64) + } + Logger.Println("Using random seed:", seed) + rand.Seed(seed) + + proxyAddr := os.Getenv("TOXIPROXY_ADDR") + if proxyAddr == "" { + proxyAddr = VagrantToxiproxy + } + proxyClient = toxiproxy.NewClient(proxyAddr) + + kafkaPeers := os.Getenv("KAFKA_PEERS") + if kafkaPeers == "" { + kafkaPeers = VagrantKafkaPeers + } + kafkaBrokers = strings.Split(kafkaPeers, ",") + + if c, err := net.DialTimeout("tcp", kafkaBrokers[0], 5*time.Second); err == nil { + if err = c.Close(); err == nil { + kafkaAvailable = true + } + } + + kafkaRequired = os.Getenv("CI") != "" +} + +func checkKafkaAvailability(t testing.TB) { + if !kafkaAvailable { + if kafkaRequired { + t.Fatalf("Kafka broker is not available on %s. Set KAFKA_PEERS to connect to Kafka on a different location.", kafkaBrokers[0]) + } else { + t.Skipf("Kafka broker is not available on %s. Set KAFKA_PEERS to connect to Kafka on a different location.", kafkaBrokers[0]) + } + } +} + +func checkKafkaVersion(t testing.TB, requiredVersion string) { + kafkaVersion := os.Getenv("KAFKA_VERSION") + if kafkaVersion == "" { + t.Logf("No KAFKA_VERSION set. This tests requires Kafka version %s or higher. Continuing...", requiredVersion) + } else { + available := parseKafkaVersion(kafkaVersion) + required := parseKafkaVersion(requiredVersion) + if !available.satisfies(required) { + t.Skipf("Kafka version %s is required for this test; you have %s. Skipping...", requiredVersion, kafkaVersion) + } + } +} + +func resetProxies(t testing.TB) { + if err := proxyClient.ResetState(); err != nil { + t.Error(err) + } + Proxies = nil +} + +func fetchProxies(t testing.TB) { + var err error + Proxies, err = proxyClient.Proxies() + if err != nil { + t.Fatal(err) + } +} + +func SaveProxy(t *testing.T, px string) { + if err := Proxies[px].Save(); err != nil { + t.Fatal(err) + } +} + +func setupFunctionalTest(t testing.TB) { + checkKafkaAvailability(t) + resetProxies(t) + fetchProxies(t) +} + +func teardownFunctionalTest(t testing.TB) { + resetProxies(t) +} + +type kafkaVersion []int + +func (kv kafkaVersion) satisfies(other kafkaVersion) bool { + var ov int + for index, v := range kv { + if len(other) <= index { + ov = 0 + } else { + ov = other[index] + } + + if v < ov { + return false + } + } + return true +} + +func parseKafkaVersion(version string) kafkaVersion { + numbers := strings.Split(version, ".") + result := make(kafkaVersion, 0, len(numbers)) + for _, number := range numbers { + nr, _ := strconv.Atoi(number) + result = append(result, nr) + } + + return result +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/length_field.go b/Godeps/_workspace/src/github.com/Shopify/sarama/length_field.go new file mode 100644 index 0000000000000..70078be5d9f2d --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/length_field.go @@ -0,0 +1,29 @@ +package sarama + +import "encoding/binary" + +// LengthField implements the PushEncoder and PushDecoder interfaces for calculating 4-byte lengths. +type lengthField struct { + startOffset int +} + +func (l *lengthField) saveOffset(in int) { + l.startOffset = in +} + +func (l *lengthField) reserveLength() int { + return 4 +} + +func (l *lengthField) run(curOffset int, buf []byte) error { + binary.BigEndian.PutUint32(buf[l.startOffset:], uint32(curOffset-l.startOffset-4)) + return nil +} + +func (l *lengthField) check(curOffset int, buf []byte) error { + if uint32(curOffset-l.startOffset-4) != binary.BigEndian.Uint32(buf[l.startOffset:]) { + return PacketDecodingError{"length field invalid"} + } + + return nil +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/message.go b/Godeps/_workspace/src/github.com/Shopify/sarama/message.go new file mode 100644 index 0000000000000..49b19c5a69136 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/message.go @@ -0,0 +1,154 @@ +package sarama + +import ( + "bytes" + "compress/gzip" + "fmt" + "io/ioutil" +) + +// CompressionCodec represents the various compression codecs recognized by Kafka in messages. +type CompressionCodec int8 + +// only the last two bits are really used +const compressionCodecMask int8 = 0x03 + +const ( + CompressionNone CompressionCodec = 0 + CompressionGZIP CompressionCodec = 1 + CompressionSnappy CompressionCodec = 2 +) + +// The spec just says: "This is a version id used to allow backwards compatible evolution of the message +// binary format." but it doesn't say what the current value is, so presumably 0... +const messageFormat int8 = 0 + +type Message struct { + Codec CompressionCodec // codec used to compress the message contents + Key []byte // the message key, may be nil + Value []byte // the message contents + Set *MessageSet // the message set a message might wrap + + compressedCache []byte +} + +func (m *Message) encode(pe packetEncoder) error { + pe.push(&crc32Field{}) + + pe.putInt8(messageFormat) + + attributes := int8(m.Codec) & compressionCodecMask + pe.putInt8(attributes) + + err := pe.putBytes(m.Key) + if err != nil { + return err + } + + var payload []byte + + if m.compressedCache != nil { + payload = m.compressedCache + m.compressedCache = nil + } else { + switch m.Codec { + case CompressionNone: + payload = m.Value + case CompressionGZIP: + var buf bytes.Buffer + writer := gzip.NewWriter(&buf) + if _, err = writer.Write(m.Value); err != nil { + return err + } + if err = writer.Close(); err != nil { + return err + } + m.compressedCache = buf.Bytes() + payload = m.compressedCache + case CompressionSnappy: + tmp := snappyEncode(m.Value) + m.compressedCache = tmp + payload = m.compressedCache + default: + return PacketEncodingError{fmt.Sprintf("unsupported compression codec (%d)", m.Codec)} + } + } + + if err = pe.putBytes(payload); err != nil { + return err + } + + return pe.pop() +} + +func (m *Message) decode(pd packetDecoder) (err error) { + err = pd.push(&crc32Field{}) + if err != nil { + return err + } + + format, err := pd.getInt8() + if err != nil { + return err + } + if format != messageFormat { + return PacketDecodingError{"unexpected messageFormat"} + } + + attribute, err := pd.getInt8() + if err != nil { + return err + } + m.Codec = CompressionCodec(attribute & compressionCodecMask) + + m.Key, err = pd.getBytes() + if err != nil { + return err + } + + m.Value, err = pd.getBytes() + if err != nil { + return err + } + + switch m.Codec { + case CompressionNone: + // nothing to do + case CompressionGZIP: + if m.Value == nil { + return PacketDecodingError{"GZIP compression specified, but no data to uncompress"} + } + reader, err := gzip.NewReader(bytes.NewReader(m.Value)) + if err != nil { + return err + } + if m.Value, err = ioutil.ReadAll(reader); err != nil { + return err + } + return m.decodeSet() + case CompressionSnappy: + if m.Value == nil { + return PacketDecodingError{"Snappy compression specified, but no data to uncompress"} + } + if m.Value, err = snappyDecode(m.Value); err != nil { + return err + } + return m.decodeSet() + default: + return PacketDecodingError{fmt.Sprintf("invalid compression specified (%d)", m.Codec)} + } + + err = pd.pop() + if err != nil { + return err + } + + return nil +} + +// decodes a message set from a previousy encoded bulk-message +func (m *Message) decodeSet() (err error) { + pd := realDecoder{raw: m.Value} + m.Set = &MessageSet{} + return m.Set.decode(&pd) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/message_set.go b/Godeps/_workspace/src/github.com/Shopify/sarama/message_set.go new file mode 100644 index 0000000000000..f028784e51a0d --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/message_set.go @@ -0,0 +1,89 @@ +package sarama + +type MessageBlock struct { + Offset int64 + Msg *Message +} + +// Messages convenience helper which returns either all the +// messages that are wrapped in this block +func (msb *MessageBlock) Messages() []*MessageBlock { + if msb.Msg.Set != nil { + return msb.Msg.Set.Messages + } + return []*MessageBlock{msb} +} + +func (msb *MessageBlock) encode(pe packetEncoder) error { + pe.putInt64(msb.Offset) + pe.push(&lengthField{}) + err := msb.Msg.encode(pe) + if err != nil { + return err + } + return pe.pop() +} + +func (msb *MessageBlock) decode(pd packetDecoder) (err error) { + if msb.Offset, err = pd.getInt64(); err != nil { + return err + } + + if err = pd.push(&lengthField{}); err != nil { + return err + } + + msb.Msg = new(Message) + if err = msb.Msg.decode(pd); err != nil { + return err + } + + if err = pd.pop(); err != nil { + return err + } + + return nil +} + +type MessageSet struct { + PartialTrailingMessage bool // whether the set on the wire contained an incomplete trailing MessageBlock + Messages []*MessageBlock +} + +func (ms *MessageSet) encode(pe packetEncoder) error { + for i := range ms.Messages { + err := ms.Messages[i].encode(pe) + if err != nil { + return err + } + } + return nil +} + +func (ms *MessageSet) decode(pd packetDecoder) (err error) { + ms.Messages = nil + + for pd.remaining() > 0 { + msb := new(MessageBlock) + err = msb.decode(pd) + switch err { + case nil: + ms.Messages = append(ms.Messages, msb) + case ErrInsufficientData: + // As an optimization the server is allowed to return a partial message at the + // end of the message set. Clients should handle this case. So we just ignore such things. + ms.PartialTrailingMessage = true + return nil + default: + return err + } + } + + return nil +} + +func (ms *MessageSet) addMessage(msg *Message) { + block := new(MessageBlock) + block.Msg = msg + ms.Messages = append(ms.Messages, block) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/message_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/message_test.go new file mode 100644 index 0000000000000..1dae896fee247 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/message_test.go @@ -0,0 +1,113 @@ +package sarama + +import "testing" + +var ( + emptyMessage = []byte{ + 167, 236, 104, 3, // CRC + 0x00, // magic version byte + 0x00, // attribute flags + 0xFF, 0xFF, 0xFF, 0xFF, // key + 0xFF, 0xFF, 0xFF, 0xFF} // value + + emptyGzipMessage = []byte{ + 97, 79, 149, 90, //CRC + 0x00, // magic version byte + 0x01, // attribute flags + 0xFF, 0xFF, 0xFF, 0xFF, // key + // value + 0x00, 0x00, 0x00, 0x17, + 0x1f, 0x8b, + 0x08, + 0, 0, 9, 110, 136, 0, 255, 1, 0, 0, 255, 255, 0, 0, 0, 0, 0, 0, 0, 0} + + emptyBulkSnappyMessage = []byte{ + 180, 47, 53, 209, //CRC + 0x00, // magic version byte + 0x02, // attribute flags + 0xFF, 0xFF, 0xFF, 0xFF, // key + 0, 0, 0, 42, + 130, 83, 78, 65, 80, 80, 89, 0, // SNAPPY magic + 0, 0, 0, 1, // min version + 0, 0, 0, 1, // default version + 0, 0, 0, 22, 52, 0, 0, 25, 1, 16, 14, 227, 138, 104, 118, 25, 15, 13, 1, 8, 1, 0, 0, 62, 26, 0} + + emptyBulkGzipMessage = []byte{ + 139, 160, 63, 141, //CRC + 0x00, // magic version byte + 0x01, // attribute flags + 0xFF, 0xFF, 0xFF, 0xFF, // key + 0x00, 0x00, 0x00, 0x27, // len + 0x1f, 0x8b, // Gzip Magic + 0x08, // deflate compressed + 0, 0, 0, 0, 0, 0, 0, 99, 96, 128, 3, 190, 202, 112, 143, 7, 12, 12, 255, 129, 0, 33, 200, 192, 136, 41, 3, 0, 199, 226, 155, 70, 52, 0, 0, 0} +) + +func TestMessageEncoding(t *testing.T) { + message := Message{} + testEncodable(t, "empty", &message, emptyMessage) + + message.Value = []byte{} + message.Codec = CompressionGZIP + testEncodable(t, "empty gzip", &message, emptyGzipMessage) +} + +func TestMessageDecoding(t *testing.T) { + message := Message{} + testDecodable(t, "empty", &message, emptyMessage) + if message.Codec != CompressionNone { + t.Error("Decoding produced compression codec where there was none.") + } + if message.Key != nil { + t.Error("Decoding produced key where there was none.") + } + if message.Value != nil { + t.Error("Decoding produced value where there was none.") + } + if message.Set != nil { + t.Error("Decoding produced set where there was none.") + } + + testDecodable(t, "empty gzip", &message, emptyGzipMessage) + if message.Codec != CompressionGZIP { + t.Error("Decoding produced incorrect compression codec (was gzip).") + } + if message.Key != nil { + t.Error("Decoding produced key where there was none.") + } + if message.Value == nil || len(message.Value) != 0 { + t.Error("Decoding produced nil or content-ful value where there was an empty array.") + } +} + +func TestMessageDecodingBulkSnappy(t *testing.T) { + message := Message{} + testDecodable(t, "bulk snappy", &message, emptyBulkSnappyMessage) + if message.Codec != CompressionSnappy { + t.Errorf("Decoding produced codec %d, but expected %d.", message.Codec, CompressionSnappy) + } + if message.Key != nil { + t.Errorf("Decoding produced key %+v, but none was expected.", message.Key) + } + if message.Set == nil { + t.Error("Decoding produced no set, but one was expected.") + } else if len(message.Set.Messages) != 2 { + t.Errorf("Decoding produced a set with %d messages, but 2 were expected.", len(message.Set.Messages)) + } +} + +func TestMessageDecodingBulkGzip(t *testing.T) { + message := Message{} + testDecodable(t, "bulk gzip", &message, emptyBulkGzipMessage) + if message.Codec != CompressionGZIP { + t.Errorf("Decoding produced codec %d, but expected %d.", message.Codec, CompressionGZIP) + } + if message.Key != nil { + t.Errorf("Decoding produced key %+v, but none was expected.", message.Key) + } + if message.Set == nil { + t.Error("Decoding produced no set, but one was expected.") + } else if len(message.Set.Messages) != 2 { + t.Errorf("Decoding produced a set with %d messages, but 2 were expected.", len(message.Set.Messages)) + } +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/metadata_request.go b/Godeps/_workspace/src/github.com/Shopify/sarama/metadata_request.go new file mode 100644 index 0000000000000..130cfd4f11962 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/metadata_request.go @@ -0,0 +1,48 @@ +package sarama + +type MetadataRequest struct { + Topics []string +} + +func (mr *MetadataRequest) encode(pe packetEncoder) error { + err := pe.putArrayLength(len(mr.Topics)) + if err != nil { + return err + } + + for i := range mr.Topics { + err = pe.putString(mr.Topics[i]) + if err != nil { + return err + } + } + return nil +} + +func (mr *MetadataRequest) decode(pd packetDecoder) error { + topicCount, err := pd.getArrayLength() + if err != nil { + return err + } + if topicCount == 0 { + return nil + } + + mr.Topics = make([]string, topicCount) + for i := range mr.Topics { + topic, err := pd.getString() + if err != nil { + return err + } + mr.Topics[i] = topic + } + return nil +} + +func (mr *MetadataRequest) key() int16 { + return 3 +} + +func (mr *MetadataRequest) version() int16 { + return 0 +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/metadata_request_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/metadata_request_test.go new file mode 100644 index 0000000000000..44f3146e4ad39 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/metadata_request_test.go @@ -0,0 +1,29 @@ +package sarama + +import "testing" + +var ( + metadataRequestNoTopics = []byte{ + 0x00, 0x00, 0x00, 0x00} + + metadataRequestOneTopic = []byte{ + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x06, 't', 'o', 'p', 'i', 'c', '1'} + + metadataRequestThreeTopics = []byte{ + 0x00, 0x00, 0x00, 0x03, + 0x00, 0x03, 'f', 'o', 'o', + 0x00, 0x03, 'b', 'a', 'r', + 0x00, 0x03, 'b', 'a', 'z'} +) + +func TestMetadataRequest(t *testing.T) { + request := new(MetadataRequest) + testRequest(t, "no topics", request, metadataRequestNoTopics) + + request.Topics = []string{"topic1"} + testRequest(t, "one topic", request, metadataRequestOneTopic) + + request.Topics = []string{"foo", "bar", "baz"} + testRequest(t, "three topics", request, metadataRequestThreeTopics) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/metadata_response.go b/Godeps/_workspace/src/github.com/Shopify/sarama/metadata_response.go new file mode 100644 index 0000000000000..b82221f7ed3e9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/metadata_response.go @@ -0,0 +1,227 @@ +package sarama + +type PartitionMetadata struct { + Err KError + ID int32 + Leader int32 + Replicas []int32 + Isr []int32 +} + +func (pm *PartitionMetadata) decode(pd packetDecoder) (err error) { + tmp, err := pd.getInt16() + if err != nil { + return err + } + pm.Err = KError(tmp) + + pm.ID, err = pd.getInt32() + if err != nil { + return err + } + + pm.Leader, err = pd.getInt32() + if err != nil { + return err + } + + pm.Replicas, err = pd.getInt32Array() + if err != nil { + return err + } + + pm.Isr, err = pd.getInt32Array() + if err != nil { + return err + } + + return nil +} + +func (pm *PartitionMetadata) encode(pe packetEncoder) (err error) { + pe.putInt16(int16(pm.Err)) + pe.putInt32(pm.ID) + pe.putInt32(pm.Leader) + + err = pe.putInt32Array(pm.Replicas) + if err != nil { + return err + } + + err = pe.putInt32Array(pm.Isr) + if err != nil { + return err + } + + return nil +} + +type TopicMetadata struct { + Err KError + Name string + Partitions []*PartitionMetadata +} + +func (tm *TopicMetadata) decode(pd packetDecoder) (err error) { + tmp, err := pd.getInt16() + if err != nil { + return err + } + tm.Err = KError(tmp) + + tm.Name, err = pd.getString() + if err != nil { + return err + } + + n, err := pd.getArrayLength() + if err != nil { + return err + } + tm.Partitions = make([]*PartitionMetadata, n) + for i := 0; i < n; i++ { + tm.Partitions[i] = new(PartitionMetadata) + err = tm.Partitions[i].decode(pd) + if err != nil { + return err + } + } + + return nil +} + +func (tm *TopicMetadata) encode(pe packetEncoder) (err error) { + pe.putInt16(int16(tm.Err)) + + err = pe.putString(tm.Name) + if err != nil { + return err + } + + err = pe.putArrayLength(len(tm.Partitions)) + if err != nil { + return err + } + + for _, pm := range tm.Partitions { + err = pm.encode(pe) + if err != nil { + return err + } + } + + return nil +} + +type MetadataResponse struct { + Brokers []*Broker + Topics []*TopicMetadata +} + +func (m *MetadataResponse) decode(pd packetDecoder) (err error) { + n, err := pd.getArrayLength() + if err != nil { + return err + } + + m.Brokers = make([]*Broker, n) + for i := 0; i < n; i++ { + m.Brokers[i] = new(Broker) + err = m.Brokers[i].decode(pd) + if err != nil { + return err + } + } + + n, err = pd.getArrayLength() + if err != nil { + return err + } + + m.Topics = make([]*TopicMetadata, n) + for i := 0; i < n; i++ { + m.Topics[i] = new(TopicMetadata) + err = m.Topics[i].decode(pd) + if err != nil { + return err + } + } + + return nil +} + +func (m *MetadataResponse) encode(pe packetEncoder) error { + err := pe.putArrayLength(len(m.Brokers)) + if err != nil { + return err + } + for _, broker := range m.Brokers { + err = broker.encode(pe) + if err != nil { + return err + } + } + + err = pe.putArrayLength(len(m.Topics)) + if err != nil { + return err + } + for _, tm := range m.Topics { + err = tm.encode(pe) + if err != nil { + return err + } + } + + return nil +} + +// testing API + +func (m *MetadataResponse) AddBroker(addr string, id int32) { + m.Brokers = append(m.Brokers, &Broker{id: id, addr: addr}) +} + +func (m *MetadataResponse) AddTopic(topic string, err KError) *TopicMetadata { + var tmatch *TopicMetadata + + for _, tm := range m.Topics { + if tm.Name == topic { + tmatch = tm + goto foundTopic + } + } + + tmatch = new(TopicMetadata) + tmatch.Name = topic + m.Topics = append(m.Topics, tmatch) + +foundTopic: + + tmatch.Err = err + return tmatch +} + +func (m *MetadataResponse) AddTopicPartition(topic string, partition, brokerID int32, replicas, isr []int32, err KError) { + tmatch := m.AddTopic(topic, ErrNoError) + var pmatch *PartitionMetadata + + for _, pm := range tmatch.Partitions { + if pm.ID == partition { + pmatch = pm + goto foundPartition + } + } + + pmatch = new(PartitionMetadata) + pmatch.ID = partition + tmatch.Partitions = append(tmatch.Partitions, pmatch) + +foundPartition: + + pmatch.Leader = brokerID + pmatch.Replicas = replicas + pmatch.Isr = isr + pmatch.Err = err + +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/metadata_response_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/metadata_response_test.go new file mode 100644 index 0000000000000..1f1a51549e616 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/metadata_response_test.go @@ -0,0 +1,139 @@ +package sarama + +import "testing" + +var ( + emptyMetadataResponse = []byte{ + 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00} + + brokersNoTopicsMetadataResponse = []byte{ + 0x00, 0x00, 0x00, 0x02, + + 0x00, 0x00, 0xab, 0xff, + 0x00, 0x09, 'l', 'o', 'c', 'a', 'l', 'h', 'o', 's', 't', + 0x00, 0x00, 0x00, 0x33, + + 0x00, 0x01, 0x02, 0x03, + 0x00, 0x0a, 'g', 'o', 'o', 'g', 'l', 'e', '.', 'c', 'o', 'm', + 0x00, 0x00, 0x01, 0x11, + + 0x00, 0x00, 0x00, 0x00} + + topicsNoBrokersMetadataResponse = []byte{ + 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x02, + + 0x00, 0x00, + 0x00, 0x03, 'f', 'o', 'o', + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x04, + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x00, 0x07, + 0x00, 0x00, 0x00, 0x03, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, 0x00, 0x03, + 0x00, 0x00, 0x00, 0x00, + + 0x00, 0x00, + 0x00, 0x03, 'b', 'a', 'r', + 0x00, 0x00, 0x00, 0x00} +) + +func TestEmptyMetadataResponse(t *testing.T) { + response := MetadataResponse{} + + testDecodable(t, "empty", &response, emptyMetadataResponse) + if len(response.Brokers) != 0 { + t.Error("Decoding produced", len(response.Brokers), "brokers where there were none!") + } + if len(response.Topics) != 0 { + t.Error("Decoding produced", len(response.Topics), "topics where there were none!") + } +} + +func TestMetadataResponseWithBrokers(t *testing.T) { + response := MetadataResponse{} + + testDecodable(t, "brokers, no topics", &response, brokersNoTopicsMetadataResponse) + if len(response.Brokers) != 2 { + t.Fatal("Decoding produced", len(response.Brokers), "brokers where there were two!") + } + + if response.Brokers[0].id != 0xabff { + t.Error("Decoding produced invalid broker 0 id.") + } + if response.Brokers[0].addr != "localhost:51" { + t.Error("Decoding produced invalid broker 0 address.") + } + if response.Brokers[1].id != 0x010203 { + t.Error("Decoding produced invalid broker 1 id.") + } + if response.Brokers[1].addr != "google.com:273" { + t.Error("Decoding produced invalid broker 1 address.") + } + + if len(response.Topics) != 0 { + t.Error("Decoding produced", len(response.Topics), "topics where there were none!") + } +} + +func TestMetadataResponseWithTopics(t *testing.T) { + response := MetadataResponse{} + + testDecodable(t, "topics, no brokers", &response, topicsNoBrokersMetadataResponse) + if len(response.Brokers) != 0 { + t.Error("Decoding produced", len(response.Brokers), "brokers where there were none!") + } + + if len(response.Topics) != 2 { + t.Fatal("Decoding produced", len(response.Topics), "topics where there were two!") + } + + if response.Topics[0].Err != ErrNoError { + t.Error("Decoding produced invalid topic 0 error.") + } + + if response.Topics[0].Name != "foo" { + t.Error("Decoding produced invalid topic 0 name.") + } + + if len(response.Topics[0].Partitions) != 1 { + t.Fatal("Decoding produced invalid partition count for topic 0.") + } + + if response.Topics[0].Partitions[0].Err != ErrInvalidMessageSize { + t.Error("Decoding produced invalid topic 0 partition 0 error.") + } + + if response.Topics[0].Partitions[0].ID != 0x01 { + t.Error("Decoding produced invalid topic 0 partition 0 id.") + } + + if response.Topics[0].Partitions[0].Leader != 0x07 { + t.Error("Decoding produced invalid topic 0 partition 0 leader.") + } + + if len(response.Topics[0].Partitions[0].Replicas) != 3 { + t.Fatal("Decoding produced invalid topic 0 partition 0 replicas.") + } + for i := 0; i < 3; i++ { + if response.Topics[0].Partitions[0].Replicas[i] != int32(i+1) { + t.Error("Decoding produced invalid topic 0 partition 0 replica", i) + } + } + + if len(response.Topics[0].Partitions[0].Isr) != 0 { + t.Error("Decoding produced invalid topic 0 partition 0 isr length.") + } + + if response.Topics[1].Err != ErrNoError { + t.Error("Decoding produced invalid topic 1 error.") + } + + if response.Topics[1].Name != "bar" { + t.Error("Decoding produced invalid topic 0 name.") + } + + if len(response.Topics[1].Partitions) != 0 { + t.Error("Decoding produced invalid partition count for topic 1.") + } +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/mockbroker_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/mockbroker_test.go new file mode 100644 index 0000000000000..987697380f6a3 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/mockbroker_test.go @@ -0,0 +1,273 @@ +package sarama + +import ( + "bytes" + "encoding/binary" + "fmt" + "io" + "net" + "reflect" + "strconv" + "sync" + "testing" + "time" + + "github.com/davecgh/go-spew/spew" +) + +const ( + expectationTimeout = 500 * time.Millisecond +) + +type requestHandlerFunc func(req *request) (res encoder) + +// mockBroker is a mock Kafka broker. It consists of a TCP server on a +// kernel-selected localhost port that can accept many connections. It reads +// Kafka requests from that connection and passes them to the user specified +// handler function (see SetHandler) that generates respective responses. If +// the handler has not been explicitly specified then the broker returns +// responses set by the Returns function in the exact order they were provided. +// (if a response has a len of 0, nothing is sent, and the client request will +// timeout in this case). +// +// When running tests with one of these, it is strongly recommended to specify +// a timeout to `go test` so that if the broker hangs waiting for a response, +// the test panics. +// +// It is not necessary to prefix message length or correlation ID to your +// response bytes, the server does that automatically as a convenience. +type mockBroker struct { + brokerID int32 + port int32 + closing chan none + stopper chan none + expectations chan encoder + listener net.Listener + t *testing.T + latency time.Duration + handler requestHandlerFunc + history []RequestResponse + lock sync.Mutex +} + +type RequestResponse struct { + Request requestBody + Response encoder +} + +func (b *mockBroker) SetLatency(latency time.Duration) { + b.latency = latency +} + +// SetHandler sets the specified function as the request handler. Whenever +// a mock broker reads a request from the wire it passes the request to the +// function and sends back whatever the handler function returns. +func (b *mockBroker) SetHandler(handler requestHandlerFunc) { + b.lock.Lock() + b.handler = handler + b.lock.Unlock() +} + +func (b *mockBroker) SetHandlerByMap(handlerMap map[string]MockResponse) { + b.SetHandler(func(req *request) (res encoder) { + reqTypeName := reflect.TypeOf(req.body).Elem().Name() + mockResponse := handlerMap[reqTypeName] + if mockResponse == nil { + return nil + } + return mockResponse.For(req.body) + }) +} + +func (b *mockBroker) BrokerID() int32 { + return b.brokerID +} + +func (b *mockBroker) History() []RequestResponse { + b.lock.Lock() + history := make([]RequestResponse, len(b.history)) + copy(history, b.history) + b.lock.Unlock() + return history +} + +func (b *mockBroker) Port() int32 { + return b.port +} + +func (b *mockBroker) Addr() string { + return b.listener.Addr().String() +} + +func (b *mockBroker) Close() { + close(b.expectations) + if len(b.expectations) > 0 { + buf := bytes.NewBufferString(fmt.Sprintf("mockbroker/%d: not all expectations were satisfied! Still waiting on:\n", b.BrokerID())) + for e := range b.expectations { + _, _ = buf.WriteString(spew.Sdump(e)) + } + b.t.Error(buf.String()) + } + close(b.closing) + <-b.stopper +} + +func (b *mockBroker) serverLoop() { + defer close(b.stopper) + var err error + var conn net.Conn + + go func() { + <-b.closing + safeClose(b.t, b.listener) + }() + + wg := &sync.WaitGroup{} + i := 0 + for conn, err = b.listener.Accept(); err == nil; conn, err = b.listener.Accept() { + wg.Add(1) + go b.handleRequests(conn, i, wg) + i++ + } + wg.Wait() + Logger.Printf("*** mockbroker/%d: listener closed, err=%v", b.BrokerID(), err) +} + +func (b *mockBroker) handleRequests(conn net.Conn, idx int, wg *sync.WaitGroup) { + defer wg.Done() + defer func() { + _ = conn.Close() + }() + Logger.Printf("*** mockbroker/%d/%d: connection opened", b.BrokerID(), idx) + var err error + + abort := make(chan none) + defer close(abort) + go func() { + select { + case <-b.closing: + _ = conn.Close() + case <-abort: + } + }() + + resHeader := make([]byte, 8) + for { + req, err := decodeRequest(conn) + if err != nil { + Logger.Printf("*** mockbroker/%d/%d: invalid request: err=%+v, %+v", b.brokerID, idx, err, spew.Sdump(req)) + b.serverError(err) + break + } + + if b.latency > 0 { + time.Sleep(b.latency) + } + + b.lock.Lock() + res := b.handler(req) + b.history = append(b.history, RequestResponse{req.body, res}) + b.lock.Unlock() + + if res == nil { + Logger.Printf("*** mockbroker/%d/%d: ignored %v", b.brokerID, idx, spew.Sdump(req)) + continue + } + Logger.Printf("*** mockbroker/%d/%d: served %v -> %v", b.brokerID, idx, req, res) + + encodedRes, err := encode(res) + if err != nil { + b.serverError(err) + break + } + if len(encodedRes) == 0 { + continue + } + + binary.BigEndian.PutUint32(resHeader, uint32(len(encodedRes)+4)) + binary.BigEndian.PutUint32(resHeader[4:], uint32(req.correlationID)) + if _, err = conn.Write(resHeader); err != nil { + b.serverError(err) + break + } + if _, err = conn.Write(encodedRes); err != nil { + b.serverError(err) + break + } + } + Logger.Printf("*** mockbroker/%d/%d: connection closed, err=%v", b.BrokerID(), idx, err) +} + +func (b *mockBroker) defaultRequestHandler(req *request) (res encoder) { + select { + case res, ok := <-b.expectations: + if !ok { + return nil + } + return res + case <-time.After(expectationTimeout): + return nil + } +} + +func (b *mockBroker) serverError(err error) { + isConnectionClosedError := false + if _, ok := err.(*net.OpError); ok { + isConnectionClosedError = true + } else if err == io.EOF { + isConnectionClosedError = true + } else if err.Error() == "use of closed network connection" { + isConnectionClosedError = true + } + + if isConnectionClosedError { + return + } + + b.t.Errorf(err.Error()) +} + +// newMockBroker launches a fake Kafka broker. It takes a *testing.T as provided by the +// test framework and a channel of responses to use. If an error occurs it is +// simply logged to the *testing.T and the broker exits. +func newMockBroker(t *testing.T, brokerID int32) *mockBroker { + return newMockBrokerAddr(t, brokerID, "localhost:0") +} + +// newMockBrokerAddr behaves like newMockBroker but listens on the address you give +// it rather than just some ephemeral port. +func newMockBrokerAddr(t *testing.T, brokerID int32, addr string) *mockBroker { + var err error + + broker := &mockBroker{ + closing: make(chan none), + stopper: make(chan none), + t: t, + brokerID: brokerID, + expectations: make(chan encoder, 512), + } + broker.handler = broker.defaultRequestHandler + + broker.listener, err = net.Listen("tcp", addr) + if err != nil { + t.Fatal(err) + } + Logger.Printf("*** mockbroker/%d listening on %s\n", brokerID, broker.listener.Addr().String()) + _, portStr, err := net.SplitHostPort(broker.listener.Addr().String()) + if err != nil { + t.Fatal(err) + } + tmp, err := strconv.ParseInt(portStr, 10, 32) + if err != nil { + t.Fatal(err) + } + broker.port = int32(tmp) + + go broker.serverLoop() + + return broker +} + +func (b *mockBroker) Returns(e encoder) { + b.expectations <- e +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/mockresponses_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/mockresponses_test.go new file mode 100644 index 0000000000000..655d9fb3b5589 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/mockresponses_test.go @@ -0,0 +1,411 @@ +package sarama + +import ( + "testing" +) + +// MockResponse is a response builder interface it defines one method that +// allows generating a response based on a request body. +type MockResponse interface { + For(reqBody decoder) (res encoder) +} + +type mockWrapper struct { + res encoder +} + +func (mw *mockWrapper) For(reqBody decoder) (res encoder) { + return mw.res +} + +func newMockWrapper(res encoder) *mockWrapper { + return &mockWrapper{res: res} +} + +// mockMetadataResponse is a `MetadataResponse` builder. +type mockMetadataResponse struct { + leaders map[string]map[int32]int32 + brokers map[string]int32 + t *testing.T +} + +func newMockMetadataResponse(t *testing.T) *mockMetadataResponse { + return &mockMetadataResponse{ + leaders: make(map[string]map[int32]int32), + brokers: make(map[string]int32), + t: t, + } +} + +func (mmr *mockMetadataResponse) SetLeader(topic string, partition, brokerID int32) *mockMetadataResponse { + partitions := mmr.leaders[topic] + if partitions == nil { + partitions = make(map[int32]int32) + mmr.leaders[topic] = partitions + } + partitions[partition] = brokerID + return mmr +} + +func (mmr *mockMetadataResponse) SetBroker(addr string, brokerID int32) *mockMetadataResponse { + mmr.brokers[addr] = brokerID + return mmr +} + +func (mor *mockMetadataResponse) For(reqBody decoder) encoder { + metadataRequest := reqBody.(*MetadataRequest) + metadataResponse := &MetadataResponse{} + for addr, brokerID := range mor.brokers { + metadataResponse.AddBroker(addr, brokerID) + } + if len(metadataRequest.Topics) == 0 { + for topic, partitions := range mor.leaders { + for partition, brokerID := range partitions { + metadataResponse.AddTopicPartition(topic, partition, brokerID, nil, nil, ErrNoError) + } + } + return metadataResponse + } + for _, topic := range metadataRequest.Topics { + for partition, brokerID := range mor.leaders[topic] { + metadataResponse.AddTopicPartition(topic, partition, brokerID, nil, nil, ErrNoError) + } + } + return metadataResponse +} + +// mockOffsetResponse is an `OffsetResponse` builder. +type mockOffsetResponse struct { + offsets map[string]map[int32]map[int64]int64 + t *testing.T +} + +func newMockOffsetResponse(t *testing.T) *mockOffsetResponse { + return &mockOffsetResponse{ + offsets: make(map[string]map[int32]map[int64]int64), + t: t, + } +} + +func (mor *mockOffsetResponse) SetOffset(topic string, partition int32, time, offset int64) *mockOffsetResponse { + partitions := mor.offsets[topic] + if partitions == nil { + partitions = make(map[int32]map[int64]int64) + mor.offsets[topic] = partitions + } + times := partitions[partition] + if times == nil { + times = make(map[int64]int64) + partitions[partition] = times + } + times[time] = offset + return mor +} + +func (mor *mockOffsetResponse) For(reqBody decoder) encoder { + offsetRequest := reqBody.(*OffsetRequest) + offsetResponse := &OffsetResponse{} + for topic, partitions := range offsetRequest.blocks { + for partition, block := range partitions { + offset := mor.getOffset(topic, partition, block.time) + offsetResponse.AddTopicPartition(topic, partition, offset) + } + } + return offsetResponse +} + +func (mor *mockOffsetResponse) getOffset(topic string, partition int32, time int64) int64 { + partitions := mor.offsets[topic] + if partitions == nil { + mor.t.Errorf("missing topic: %s", topic) + } + times := partitions[partition] + if times == nil { + mor.t.Errorf("missing partition: %d", partition) + } + offset, ok := times[time] + if !ok { + mor.t.Errorf("missing time: %d", time) + } + return offset +} + +// mockFetchResponse is a `FetchResponse` builder. +type mockFetchResponse struct { + messages map[string]map[int32]map[int64]Encoder + highWaterMarks map[string]map[int32]int64 + t *testing.T + batchSize int +} + +func newMockFetchResponse(t *testing.T, batchSize int) *mockFetchResponse { + return &mockFetchResponse{ + messages: make(map[string]map[int32]map[int64]Encoder), + highWaterMarks: make(map[string]map[int32]int64), + t: t, + batchSize: batchSize, + } +} + +func (mfr *mockFetchResponse) SetMessage(topic string, partition int32, offset int64, msg Encoder) *mockFetchResponse { + partitions := mfr.messages[topic] + if partitions == nil { + partitions = make(map[int32]map[int64]Encoder) + mfr.messages[topic] = partitions + } + messages := partitions[partition] + if messages == nil { + messages = make(map[int64]Encoder) + partitions[partition] = messages + } + messages[offset] = msg + return mfr +} + +func (mfr *mockFetchResponse) SetHighWaterMark(topic string, partition int32, offset int64) *mockFetchResponse { + partitions := mfr.highWaterMarks[topic] + if partitions == nil { + partitions = make(map[int32]int64) + mfr.highWaterMarks[topic] = partitions + } + partitions[partition] = offset + return mfr +} + +func (mfr *mockFetchResponse) For(reqBody decoder) encoder { + fetchRequest := reqBody.(*FetchRequest) + res := &FetchResponse{} + for topic, partitions := range fetchRequest.blocks { + for partition, block := range partitions { + initialOffset := block.fetchOffset + offset := initialOffset + maxOffset := initialOffset + int64(mfr.getMessageCount(topic, partition)) + for i := 0; i < mfr.batchSize && offset < maxOffset; { + msg := mfr.getMessage(topic, partition, offset) + if msg != nil { + res.AddMessage(topic, partition, nil, msg, offset) + i++ + } + offset++ + } + fb := res.GetBlock(topic, partition) + if fb == nil { + res.AddError(topic, partition, ErrNoError) + fb = res.GetBlock(topic, partition) + } + fb.HighWaterMarkOffset = mfr.getHighWaterMark(topic, partition) + } + } + return res +} + +func (mfr *mockFetchResponse) getMessage(topic string, partition int32, offset int64) Encoder { + partitions := mfr.messages[topic] + if partitions == nil { + return nil + } + messages := partitions[partition] + if messages == nil { + return nil + } + return messages[offset] +} + +func (mfr *mockFetchResponse) getMessageCount(topic string, partition int32) int { + partitions := mfr.messages[topic] + if partitions == nil { + return 0 + } + messages := partitions[partition] + if messages == nil { + return 0 + } + return len(messages) +} + +func (mfr *mockFetchResponse) getHighWaterMark(topic string, partition int32) int64 { + partitions := mfr.highWaterMarks[topic] + if partitions == nil { + return 0 + } + return partitions[partition] +} + +// mockConsumerMetadataResponse is a `ConsumerMetadataResponse` builder. +type mockConsumerMetadataResponse struct { + coordinators map[string]interface{} + t *testing.T +} + +func newMockConsumerMetadataResponse(t *testing.T) *mockConsumerMetadataResponse { + return &mockConsumerMetadataResponse{ + coordinators: make(map[string]interface{}), + t: t, + } +} + +func (mr *mockConsumerMetadataResponse) SetCoordinator(group string, broker *mockBroker) *mockConsumerMetadataResponse { + mr.coordinators[group] = broker + return mr +} + +func (mr *mockConsumerMetadataResponse) SetError(group string, kerror KError) *mockConsumerMetadataResponse { + mr.coordinators[group] = kerror + return mr +} + +func (mr *mockConsumerMetadataResponse) For(reqBody decoder) encoder { + req := reqBody.(*ConsumerMetadataRequest) + group := req.ConsumerGroup + res := &ConsumerMetadataResponse{} + v := mr.coordinators[group] + switch v := v.(type) { + case *mockBroker: + res.Coordinator = &Broker{id: v.BrokerID(), addr: v.Addr()} + case KError: + res.Err = v + } + return res +} + +// mockOffsetCommitResponse is a `OffsetCommitResponse` builder. +type mockOffsetCommitResponse struct { + errors map[string]map[string]map[int32]KError + t *testing.T +} + +func newMockOffsetCommitResponse(t *testing.T) *mockOffsetCommitResponse { + return &mockOffsetCommitResponse{t: t} +} + +func (mr *mockOffsetCommitResponse) SetError(group, topic string, partition int32, kerror KError) *mockOffsetCommitResponse { + if mr.errors == nil { + mr.errors = make(map[string]map[string]map[int32]KError) + } + topics := mr.errors[group] + if topics == nil { + topics = make(map[string]map[int32]KError) + mr.errors[group] = topics + } + partitions := topics[topic] + if partitions == nil { + partitions = make(map[int32]KError) + topics[topic] = partitions + } + partitions[partition] = kerror + return mr +} + +func (mr *mockOffsetCommitResponse) For(reqBody decoder) encoder { + req := reqBody.(*OffsetCommitRequest) + group := req.ConsumerGroup + res := &OffsetCommitResponse{} + for topic, partitions := range req.blocks { + for partition := range partitions { + res.AddError(topic, partition, mr.getError(group, topic, partition)) + } + } + return res +} + +func (mr *mockOffsetCommitResponse) getError(group, topic string, partition int32) KError { + topics := mr.errors[group] + if topics == nil { + return ErrNoError + } + partitions := topics[topic] + if partitions == nil { + return ErrNoError + } + kerror, ok := partitions[partition] + if !ok { + return ErrNoError + } + return kerror +} + +// mockProduceResponse is a `ProduceResponse` builder. +type mockProduceResponse struct { + errors map[string]map[int32]KError + t *testing.T +} + +func newMockProduceResponse(t *testing.T) *mockProduceResponse { + return &mockProduceResponse{t: t} +} + +func (mr *mockProduceResponse) SetError(topic string, partition int32, kerror KError) *mockProduceResponse { + if mr.errors == nil { + mr.errors = make(map[string]map[int32]KError) + } + partitions := mr.errors[topic] + if partitions == nil { + partitions = make(map[int32]KError) + mr.errors[topic] = partitions + } + partitions[partition] = kerror + return mr +} + +func (mr *mockProduceResponse) For(reqBody decoder) encoder { + req := reqBody.(*ProduceRequest) + res := &ProduceResponse{} + for topic, partitions := range req.msgSets { + for partition := range partitions { + res.AddTopicPartition(topic, partition, mr.getError(topic, partition)) + } + } + return res +} + +func (mr *mockProduceResponse) getError(topic string, partition int32) KError { + partitions := mr.errors[topic] + if partitions == nil { + return ErrNoError + } + kerror, ok := partitions[partition] + if !ok { + return ErrNoError + } + return kerror +} + +// mockOffsetFetchResponse is a `OffsetFetchResponse` builder. +type mockOffsetFetchResponse struct { + offsets map[string]map[string]map[int32]*OffsetFetchResponseBlock + t *testing.T +} + +func newMockOffsetFetchResponse(t *testing.T) *mockOffsetFetchResponse { + return &mockOffsetFetchResponse{t: t} +} + +func (mr *mockOffsetFetchResponse) SetOffset(group, topic string, partition int32, offset int64, metadata string, kerror KError) *mockOffsetFetchResponse { + if mr.offsets == nil { + mr.offsets = make(map[string]map[string]map[int32]*OffsetFetchResponseBlock) + } + topics := mr.offsets[group] + if topics == nil { + topics = make(map[string]map[int32]*OffsetFetchResponseBlock) + mr.offsets[group] = topics + } + partitions := topics[topic] + if partitions == nil { + partitions = make(map[int32]*OffsetFetchResponseBlock) + topics[topic] = partitions + } + partitions[partition] = &OffsetFetchResponseBlock{offset, metadata, kerror} + return mr +} + +func (mr *mockOffsetFetchResponse) For(reqBody decoder) encoder { + req := reqBody.(*OffsetFetchRequest) + group := req.ConsumerGroup + res := &OffsetFetchResponse{} + for topic, partitions := range mr.offsets[group] { + for partition, block := range partitions { + res.AddBlock(topic, partition, block) + } + } + return res +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/README.md b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/README.md new file mode 100644 index 0000000000000..55a6c2e61cd38 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/README.md @@ -0,0 +1,13 @@ +# sarama/mocks + +The `mocks` subpackage includes mock implementations that implement the interfaces of the major sarama types. +You can use them to test your sarama applications using dependency injection. + +The following mock objects are available: + +- [Consumer](https://godoc.org/github.com/Shopify/sarama/mocks#Consumer), which will create [PartitionConsumer](https://godoc.org/github.com/Shopify/sarama/mocks#PartitionConsumer) mocks. +- [AsyncProducer](https://godoc.org/github.com/Shopify/sarama/mocks#AsyncProducer) +- [SyncProducer](https://godoc.org/github.com/Shopify/sarama/mocks#SyncProducer) + +The mocks allow you to set expectations on them. When you close the mocks, the expectations will be verified, +and the results will be reported to the `*testing.T` object you provided when creating the mock. diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/async_producer.go b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/async_producer.go new file mode 100644 index 0000000000000..6ccf1f1455477 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/async_producer.go @@ -0,0 +1,142 @@ +package mocks + +import ( + "sync" + + "github.com/Shopify/sarama" +) + +// AsyncProducer implements sarama's Producer interface for testing purposes. +// Before you can send messages to it's Input channel, you have to set expectations +// so it knows how to handle the input. This way you can easily test success and +// failure scenarios. +type AsyncProducer struct { + l sync.Mutex + t ErrorReporter + expectations []*producerExpectation + closed chan struct{} + input chan *sarama.ProducerMessage + successes chan *sarama.ProducerMessage + errors chan *sarama.ProducerError + lastOffset int64 +} + +// NewAsyncProducer instantiates a new Producer mock. The t argument should +// be the *testing.T instance of your test method. An error will be written to it if +// an expectation is violated. The config argument is used to determine whether it +// should ack successes on the Successes channel. +func NewAsyncProducer(t ErrorReporter, config *sarama.Config) *AsyncProducer { + if config == nil { + config = sarama.NewConfig() + } + mp := &AsyncProducer{ + t: t, + closed: make(chan struct{}, 0), + expectations: make([]*producerExpectation, 0), + input: make(chan *sarama.ProducerMessage, config.ChannelBufferSize), + successes: make(chan *sarama.ProducerMessage, config.ChannelBufferSize), + errors: make(chan *sarama.ProducerError, config.ChannelBufferSize), + } + + go func() { + defer func() { + close(mp.successes) + close(mp.errors) + }() + + for msg := range mp.input { + mp.l.Lock() + if mp.expectations == nil || len(mp.expectations) == 0 { + mp.expectations = nil + mp.t.Errorf("No more expectation set on this mock producer to handle the input message.") + } else { + expectation := mp.expectations[0] + mp.expectations = mp.expectations[1:] + if expectation.Result == errProduceSuccess { + mp.lastOffset++ + if config.Producer.Return.Successes { + msg.Offset = mp.lastOffset + mp.successes <- msg + } + } else { + if config.Producer.Return.Errors { + mp.errors <- &sarama.ProducerError{Err: expectation.Result, Msg: msg} + } + } + } + mp.l.Unlock() + } + + mp.l.Lock() + if len(mp.expectations) > 0 { + mp.t.Errorf("Expected to exhaust all expectations, but %d are left.", len(mp.expectations)) + } + mp.l.Unlock() + + close(mp.closed) + }() + + return mp +} + +//////////////////////////////////////////////// +// Implement Producer interface +//////////////////////////////////////////////// + +// AsyncClose corresponds with the AsyncClose method of sarama's Producer implementation. +// By closing a mock producer, you also tell it that no more input will be provided, so it will +// write an error to the test state if there's any remaining expectations. +func (mp *AsyncProducer) AsyncClose() { + close(mp.input) +} + +// Close corresponds with the Close method of sarama's Producer implementation. +// By closing a mock producer, you also tell it that no more input will be provided, so it will +// write an error to the test state if there's any remaining expectations. +func (mp *AsyncProducer) Close() error { + mp.AsyncClose() + <-mp.closed + return nil +} + +// Input corresponds with the Input method of sarama's Producer implementation. +// You have to set expectations on the mock producer before writing messages to the Input +// channel, so it knows how to handle them. If there is no more remaining expectations and +// a messages is written to the Input channel, the mock producer will write an error to the test +// state object. +func (mp *AsyncProducer) Input() chan<- *sarama.ProducerMessage { + return mp.input +} + +// Successes corresponds with the Successes method of sarama's Producer implementation. +func (mp *AsyncProducer) Successes() <-chan *sarama.ProducerMessage { + return mp.successes +} + +// Errors corresponds with the Errors method of sarama's Producer implementation. +func (mp *AsyncProducer) Errors() <-chan *sarama.ProducerError { + return mp.errors +} + +//////////////////////////////////////////////// +// Setting expectations +//////////////////////////////////////////////// + +// ExpectInputAndSucceed sets an expectation on the mock producer that a message will be provided +// on the input channel. The mock producer will handle the message as if it is produced successfully, +// i.e. it will make it available on the Successes channel if the Producer.Return.Successes setting +// is set to true. +func (mp *AsyncProducer) ExpectInputAndSucceed() { + mp.l.Lock() + defer mp.l.Unlock() + mp.expectations = append(mp.expectations, &producerExpectation{Result: errProduceSuccess}) +} + +// ExpectInputAndFail sets an expectation on the mock producer that a message will be provided +// on the input channel. The mock producer will handle the message as if it failed to produce +// successfully. This means it will make a ProducerError available on the Errors channel. +func (mp *AsyncProducer) ExpectInputAndFail(err error) { + mp.l.Lock() + defer mp.l.Unlock() + mp.expectations = append(mp.expectations, &producerExpectation{Result: err}) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/async_producer_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/async_producer_test.go new file mode 100644 index 0000000000000..520bf58b9267e --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/async_producer_test.go @@ -0,0 +1,94 @@ +package mocks + +import ( + "fmt" + "testing" + + "github.com/Shopify/sarama" +) + +type testReporterMock struct { + errors []string +} + +func newTestReporterMock() *testReporterMock { + return &testReporterMock{errors: make([]string, 0)} +} + +func (trm *testReporterMock) Errorf(format string, args ...interface{}) { + trm.errors = append(trm.errors, fmt.Sprintf(format, args...)) +} + +func TestMockAsyncProducerImplementsAsyncProducerInterface(t *testing.T) { + var mp interface{} = &AsyncProducer{} + if _, ok := mp.(sarama.AsyncProducer); !ok { + t.Error("The mock producer should implement the sarama.Producer interface.") + } +} + +func TestProducerReturnsExpectationsToChannels(t *testing.T) { + config := sarama.NewConfig() + config.Producer.Return.Successes = true + mp := NewAsyncProducer(t, config) + + mp.ExpectInputAndSucceed() + mp.ExpectInputAndSucceed() + mp.ExpectInputAndFail(sarama.ErrOutOfBrokers) + + mp.Input() <- &sarama.ProducerMessage{Topic: "test 1"} + mp.Input() <- &sarama.ProducerMessage{Topic: "test 2"} + mp.Input() <- &sarama.ProducerMessage{Topic: "test 3"} + + msg1 := <-mp.Successes() + msg2 := <-mp.Successes() + err1 := <-mp.Errors() + + if msg1.Topic != "test 1" { + t.Error("Expected message 1 to be returned first") + } + + if msg2.Topic != "test 2" { + t.Error("Expected message 2 to be returned second") + } + + if err1.Msg.Topic != "test 3" || err1.Err != sarama.ErrOutOfBrokers { + t.Error("Expected message 3 to be returned as error") + } + + if err := mp.Close(); err != nil { + t.Error(err) + } +} + +func TestProducerWithTooFewExpectations(t *testing.T) { + trm := newTestReporterMock() + mp := NewAsyncProducer(trm, nil) + mp.ExpectInputAndSucceed() + + mp.Input() <- &sarama.ProducerMessage{Topic: "test"} + mp.Input() <- &sarama.ProducerMessage{Topic: "test"} + + if err := mp.Close(); err != nil { + t.Error(err) + } + + if len(trm.errors) != 1 { + t.Error("Expected to report an error") + } +} + +func TestProducerWithTooManyExpectations(t *testing.T) { + trm := newTestReporterMock() + mp := NewAsyncProducer(trm, nil) + mp.ExpectInputAndSucceed() + mp.ExpectInputAndFail(sarama.ErrOutOfBrokers) + + mp.Input() <- &sarama.ProducerMessage{Topic: "test"} + if err := mp.Close(); err != nil { + t.Error(err) + } + + if len(trm.errors) != 1 { + t.Error("Expected to report an error") + } +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/consumer.go b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/consumer.go new file mode 100644 index 0000000000000..acf0894eea8af --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/consumer.go @@ -0,0 +1,316 @@ +package mocks + +import ( + "sync" + "sync/atomic" + + "github.com/Shopify/sarama" +) + +// Consumer implements sarama's Consumer interface for testing purposes. +// Before you can start consuming from this consumer, you have to register +// topic/partitions using ExpectConsumePartition, and set expectations on them. +type Consumer struct { + l sync.Mutex + t ErrorReporter + config *sarama.Config + partitionConsumers map[string]map[int32]*PartitionConsumer + metadata map[string][]int32 +} + +// NewConsumer returns a new mock Consumer instance. The t argument should +// be the *testing.T instance of your test method. An error will be written to it if +// an expectation is violated. The config argument is currently unused and can be set to nil. +func NewConsumer(t ErrorReporter, config *sarama.Config) *Consumer { + if config == nil { + config = sarama.NewConfig() + } + + c := &Consumer{ + t: t, + config: config, + partitionConsumers: make(map[string]map[int32]*PartitionConsumer), + } + return c +} + +/////////////////////////////////////////////////// +// Consumer interface implementation +/////////////////////////////////////////////////// + +// ConsumePartition implements the ConsumePartition method from the sarama.Consumer interface. +// Before you can start consuming a partition, you have to set expectations on it using +// ExpectConsumePartition. You can only consume a partition once per consumer. +func (c *Consumer) ConsumePartition(topic string, partition int32, offset int64) (sarama.PartitionConsumer, error) { + c.l.Lock() + defer c.l.Unlock() + + if c.partitionConsumers[topic] == nil || c.partitionConsumers[topic][partition] == nil { + c.t.Errorf("No expectations set for %s/%d", topic, partition) + return nil, errOutOfExpectations + } + + pc := c.partitionConsumers[topic][partition] + if pc.consumed { + return nil, sarama.ConfigurationError("The topic/partition is already being consumed") + } + + if pc.offset != AnyOffset && pc.offset != offset { + c.t.Errorf("Unexpected offset when calling ConsumePartition for %s/%d. Expected %d, got %d.", topic, partition, pc.offset, offset) + } + + pc.consumed = true + go pc.handleExpectations() + return pc, nil +} + +// Topics returns a list of topics, as registered with SetMetadata +func (c *Consumer) Topics() ([]string, error) { + c.l.Lock() + defer c.l.Unlock() + + if c.metadata == nil { + c.t.Errorf("Unexpected call to Topics. Initialize the mock's topic metadata with SetMetadata.") + return nil, sarama.ErrOutOfBrokers + } + + var result []string + for topic, _ := range c.metadata { + result = append(result, topic) + } + return result, nil +} + +// Partitions returns the list of parititons for the given topic, as registered with SetMetadata +func (c *Consumer) Partitions(topic string) ([]int32, error) { + c.l.Lock() + defer c.l.Unlock() + + if c.metadata == nil { + c.t.Errorf("Unexpected call to Partitions. Initialize the mock's topic metadata with SetMetadata.") + return nil, sarama.ErrOutOfBrokers + } + if c.metadata[topic] == nil { + return nil, sarama.ErrUnknownTopicOrPartition + } + + return c.metadata[topic], nil +} + +// Close implements the Close method from the sarama.Consumer interface. It will close +// all registered PartitionConsumer instances. +func (c *Consumer) Close() error { + c.l.Lock() + defer c.l.Unlock() + + for _, partitions := range c.partitionConsumers { + for _, partitionConsumer := range partitions { + _ = partitionConsumer.Close() + } + } + + return nil +} + +/////////////////////////////////////////////////// +// Expectation API +/////////////////////////////////////////////////// + +// SetMetadata sets the clusters topic/partition metadata, +// which will be returned by Topics() and Partitions(). +func (c *Consumer) SetTopicMetadata(metadata map[string][]int32) { + c.l.Lock() + defer c.l.Unlock() + + c.metadata = metadata +} + +// ExpectConsumePartition will register a topic/partition, so you can set expectations on it. +// The registered PartitionConsumer will be returned, so you can set expectations +// on it using method chanining. Once a topic/partition is registered, you are +// expected to start consuming it using ConsumePartition. If that doesn't happen, +// an error will be written to the error reporter once the mock consumer is closed. It will +// also expect that the +func (c *Consumer) ExpectConsumePartition(topic string, partition int32, offset int64) *PartitionConsumer { + c.l.Lock() + defer c.l.Unlock() + + if c.partitionConsumers[topic] == nil { + c.partitionConsumers[topic] = make(map[int32]*PartitionConsumer) + } + + if c.partitionConsumers[topic][partition] == nil { + c.partitionConsumers[topic][partition] = &PartitionConsumer{ + t: c.t, + topic: topic, + partition: partition, + offset: offset, + expectations: make(chan *consumerExpectation, 1000), + messages: make(chan *sarama.ConsumerMessage, c.config.ChannelBufferSize), + errors: make(chan *sarama.ConsumerError, c.config.ChannelBufferSize), + } + } + + return c.partitionConsumers[topic][partition] +} + +/////////////////////////////////////////////////// +// PartitionConsumer mock type +/////////////////////////////////////////////////// + +// PartitionConsumer implements sarama's PartitionConsumer interface for testing purposes. +// It is returned by the mock Consumers ConsumePartitionMethod, but only if it is +// registered first using the Consumer's ExpectConsumePartition method. Before consuming the +// Errors and Messages channel, you should specify what values will be provided on these +// channels using YieldMessage and YieldError. +type PartitionConsumer struct { + l sync.Mutex + t ErrorReporter + topic string + partition int32 + offset int64 + expectations chan *consumerExpectation + messages chan *sarama.ConsumerMessage + errors chan *sarama.ConsumerError + singleClose sync.Once + consumed bool + errorsShouldBeDrained bool + messagesShouldBeDrained bool + highWaterMarkOffset int64 +} + +func (pc *PartitionConsumer) handleExpectations() { + pc.l.Lock() + defer pc.l.Unlock() + + for ex := range pc.expectations { + if ex.Err != nil { + pc.errors <- &sarama.ConsumerError{ + Topic: pc.topic, + Partition: pc.partition, + Err: ex.Err, + } + } else { + atomic.AddInt64(&pc.highWaterMarkOffset, 1) + + ex.Msg.Topic = pc.topic + ex.Msg.Partition = pc.partition + ex.Msg.Offset = atomic.LoadInt64(&pc.highWaterMarkOffset) + + pc.messages <- ex.Msg + } + } + + close(pc.messages) + close(pc.errors) +} + +/////////////////////////////////////////////////// +// PartitionConsumer interface implementation +/////////////////////////////////////////////////// + +// AsyncClose implements the AsyncClose method from the sarama.PartitionConsumer interface. +func (pc *PartitionConsumer) AsyncClose() { + pc.singleClose.Do(func() { + close(pc.expectations) + }) +} + +// Close implements the Close method from the sarama.PartitionConsumer interface. It will +// verify whether the partition consumer was actually started. +func (pc *PartitionConsumer) Close() error { + if !pc.consumed { + pc.t.Errorf("Expectations set on %s/%d, but no partition consumer was started.", pc.topic, pc.partition) + return errPartitionConsumerNotStarted + } + + if pc.errorsShouldBeDrained && len(pc.errors) > 0 { + pc.t.Errorf("Expected the errors channel for %s/%d to be drained on close, but found %d errors.", pc.topic, pc.partition, len(pc.errors)) + } + + if pc.messagesShouldBeDrained && len(pc.messages) > 0 { + pc.t.Errorf("Expected the messages channel for %s/%d to be drained on close, but found %d messages.", pc.topic, pc.partition, len(pc.messages)) + } + + pc.AsyncClose() + + var ( + closeErr error + wg sync.WaitGroup + ) + + wg.Add(1) + go func() { + defer wg.Done() + + var errs = make(sarama.ConsumerErrors, 0) + for err := range pc.errors { + errs = append(errs, err) + } + + if len(errs) > 0 { + closeErr = errs + } + }() + + wg.Add(1) + go func() { + defer wg.Done() + for _ = range pc.messages { + // drain + } + }() + + wg.Wait() + return closeErr +} + +// Errors implements the Errors method from the sarama.PartitionConsumer interface. +func (pc *PartitionConsumer) Errors() <-chan *sarama.ConsumerError { + return pc.errors +} + +// Messages implements the Messages method from the sarama.PartitionConsumer interface. +func (pc *PartitionConsumer) Messages() <-chan *sarama.ConsumerMessage { + return pc.messages +} + +func (pc *PartitionConsumer) HighWaterMarkOffset() int64 { + return atomic.LoadInt64(&pc.highWaterMarkOffset) + 1 +} + +/////////////////////////////////////////////////// +// Expectation API +/////////////////////////////////////////////////// + +// YieldMessage will yield a messages Messages channel of this partition consumer +// when it is consumed. By default, the mock consumer will not verify whether this +// message was consumed from the Messages channel, because there are legitimate +// reasons forthis not to happen. ou can call ExpectMessagesDrainedOnClose so it will +// verify that the channel is empty on close. +func (pc *PartitionConsumer) YieldMessage(msg *sarama.ConsumerMessage) { + pc.expectations <- &consumerExpectation{Msg: msg} +} + +// YieldError will yield an error on the Errors channel of this partition consumer +// when it is consumed. By default, the mock consumer will not verify whether this error was +// consumed from the Errors channel, because there are legitimate reasons for this +// not to happen. You can call ExpectErrorsDrainedOnClose so it will verify that +// the channel is empty on close. +func (pc *PartitionConsumer) YieldError(err error) { + pc.expectations <- &consumerExpectation{Err: err} +} + +// ExpectMessagesDrainedOnClose sets an expectation on the partition consumer +// that the messages channel will be fully drained when Close is called. If this +// expectation is not met, an error is reported to the error reporter. +func (pc *PartitionConsumer) ExpectMessagesDrainedOnClose() { + pc.messagesShouldBeDrained = true +} + +// ExpectErrorsDrainedOnClose sets an expectation on the partition consumer +// that the errors channel will be fully drained when Close is called. If this +// expectation is not met, an error is reported to the error reporter. +func (pc *PartitionConsumer) ExpectErrorsDrainedOnClose() { + pc.errorsShouldBeDrained = true +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/consumer_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/consumer_test.go new file mode 100644 index 0000000000000..50dad3a694142 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/consumer_test.go @@ -0,0 +1,249 @@ +package mocks + +import ( + "sort" + "testing" + + "github.com/Shopify/sarama" +) + +func TestMockConsumerImplementsConsumerInterface(t *testing.T) { + var c interface{} = &Consumer{} + if _, ok := c.(sarama.Consumer); !ok { + t.Error("The mock consumer should implement the sarama.Consumer interface.") + } + + var pc interface{} = &PartitionConsumer{} + if _, ok := pc.(sarama.PartitionConsumer); !ok { + t.Error("The mock partitionconsumer should implement the sarama.PartitionConsumer interface.") + } +} + +func TestConsumerHandlesExpectations(t *testing.T) { + consumer := NewConsumer(t, nil) + defer func() { + if err := consumer.Close(); err != nil { + t.Error(err) + } + }() + + consumer.ExpectConsumePartition("test", 0, sarama.OffsetOldest).YieldMessage(&sarama.ConsumerMessage{Value: []byte("hello world")}) + consumer.ExpectConsumePartition("test", 0, sarama.OffsetOldest).YieldError(sarama.ErrOutOfBrokers) + consumer.ExpectConsumePartition("test", 1, sarama.OffsetOldest).YieldMessage(&sarama.ConsumerMessage{Value: []byte("hello world again")}) + consumer.ExpectConsumePartition("other", 0, AnyOffset).YieldMessage(&sarama.ConsumerMessage{Value: []byte("hello other")}) + + pc_test0, err := consumer.ConsumePartition("test", 0, sarama.OffsetOldest) + if err != nil { + t.Fatal(err) + } + test0_msg := <-pc_test0.Messages() + if test0_msg.Topic != "test" || test0_msg.Partition != 0 || string(test0_msg.Value) != "hello world" { + t.Error("Message was not as expected:", test0_msg) + } + test0_err := <-pc_test0.Errors() + if test0_err.Err != sarama.ErrOutOfBrokers { + t.Error("Expected sarama.ErrOutOfBrokers, found:", test0_err.Err) + } + + pc_test1, err := consumer.ConsumePartition("test", 1, sarama.OffsetOldest) + if err != nil { + t.Fatal(err) + } + test1_msg := <-pc_test1.Messages() + if test1_msg.Topic != "test" || test1_msg.Partition != 1 || string(test1_msg.Value) != "hello world again" { + t.Error("Message was not as expected:", test1_msg) + } + + pc_other0, err := consumer.ConsumePartition("other", 0, sarama.OffsetNewest) + if err != nil { + t.Fatal(err) + } + other0_msg := <-pc_other0.Messages() + if other0_msg.Topic != "other" || other0_msg.Partition != 0 || string(other0_msg.Value) != "hello other" { + t.Error("Message was not as expected:", other0_msg) + } +} + +func TestConsumerReturnsNonconsumedErrorsOnClose(t *testing.T) { + consumer := NewConsumer(t, nil) + consumer.ExpectConsumePartition("test", 0, sarama.OffsetOldest).YieldError(sarama.ErrOutOfBrokers) + consumer.ExpectConsumePartition("test", 0, sarama.OffsetOldest).YieldError(sarama.ErrOutOfBrokers) + + pc, err := consumer.ConsumePartition("test", 0, sarama.OffsetOldest) + if err != nil { + t.Fatal(err) + } + + select { + case <-pc.Messages(): + t.Error("Did not epxect a message on the messages channel.") + case err := <-pc.Errors(): + if err.Err != sarama.ErrOutOfBrokers { + t.Error("Expected sarama.ErrOutOfBrokers, found", err) + } + } + + errs := pc.Close().(sarama.ConsumerErrors) + if len(errs) != 1 && errs[0].Err != sarama.ErrOutOfBrokers { + t.Error("Expected Close to return the remaining sarama.ErrOutOfBrokers") + } +} + +func TestConsumerWithoutExpectationsOnPartition(t *testing.T) { + trm := newTestReporterMock() + consumer := NewConsumer(trm, nil) + + _, err := consumer.ConsumePartition("test", 1, sarama.OffsetOldest) + if err != errOutOfExpectations { + t.Error("Expected ConsumePartition to return errOutOfExpectations") + } + + if err := consumer.Close(); err != nil { + t.Error("No error expected on close, but found:", err) + } + + if len(trm.errors) != 1 { + t.Errorf("Expected an expectation failure to be set on the error reporter.") + } +} + +func TestConsumerWithExpectationsOnUnconsumedPartition(t *testing.T) { + trm := newTestReporterMock() + consumer := NewConsumer(trm, nil) + consumer.ExpectConsumePartition("test", 0, sarama.OffsetOldest).YieldMessage(&sarama.ConsumerMessage{Value: []byte("hello world")}) + + if err := consumer.Close(); err != nil { + t.Error("No error expected on close, but found:", err) + } + + if len(trm.errors) != 1 { + t.Errorf("Expected an expectation failure to be set on the error reporter.") + } +} + +func TestConsumerWithWrongOffsetExpectation(t *testing.T) { + trm := newTestReporterMock() + consumer := NewConsumer(trm, nil) + consumer.ExpectConsumePartition("test", 0, sarama.OffsetOldest) + + _, err := consumer.ConsumePartition("test", 0, sarama.OffsetNewest) + if err != nil { + t.Error("Did not expect error, found:", err) + } + + if len(trm.errors) != 1 { + t.Errorf("Expected an expectation failure to be set on the error reporter.") + } + + if err := consumer.Close(); err != nil { + t.Error(err) + } +} + +func TestConsumerViolatesMessagesDrainedExpectation(t *testing.T) { + trm := newTestReporterMock() + consumer := NewConsumer(trm, nil) + pcmock := consumer.ExpectConsumePartition("test", 0, sarama.OffsetOldest) + pcmock.YieldMessage(&sarama.ConsumerMessage{Value: []byte("hello")}) + pcmock.YieldMessage(&sarama.ConsumerMessage{Value: []byte("hello")}) + pcmock.ExpectMessagesDrainedOnClose() + + pc, err := consumer.ConsumePartition("test", 0, sarama.OffsetOldest) + if err != nil { + t.Error(err) + } + + // consume first message, not second one + <-pc.Messages() + + if err := consumer.Close(); err != nil { + t.Error(err) + } + + if len(trm.errors) != 1 { + t.Errorf("Expected an expectation failure to be set on the error reporter.") + } +} + +func TestConsumerMeetsErrorsDrainedExpectation(t *testing.T) { + trm := newTestReporterMock() + consumer := NewConsumer(trm, nil) + + pcmock := consumer.ExpectConsumePartition("test", 0, sarama.OffsetOldest) + pcmock.YieldError(sarama.ErrInvalidMessage) + pcmock.YieldError(sarama.ErrInvalidMessage) + pcmock.ExpectErrorsDrainedOnClose() + + pc, err := consumer.ConsumePartition("test", 0, sarama.OffsetOldest) + if err != nil { + t.Error(err) + } + + // consume first and second error, + <-pc.Errors() + <-pc.Errors() + + if err := consumer.Close(); err != nil { + t.Error(err) + } + + if len(trm.errors) != 0 { + t.Errorf("Expected no expectation failures to be set on the error reporter.") + } +} + +func TestConsumerTopicMetadata(t *testing.T) { + trm := newTestReporterMock() + consumer := NewConsumer(trm, nil) + + consumer.SetTopicMetadata(map[string][]int32{ + "test1": []int32{0, 1, 2, 3}, + "test2": []int32{0, 1, 2, 3, 4, 5, 6, 7}, + }) + + topics, err := consumer.Topics() + if err != nil { + t.Error(t) + } + + sortedTopics := sort.StringSlice(topics) + sortedTopics.Sort() + if len(sortedTopics) != 2 || sortedTopics[0] != "test1" || sortedTopics[1] != "test2" { + t.Error("Unexpected topics returned:", sortedTopics) + } + + partitions1, err := consumer.Partitions("test1") + if err != nil { + t.Error(t) + } + + if len(partitions1) != 4 { + t.Error("Unexpected partitions returned:", len(partitions1)) + } + + partitions2, err := consumer.Partitions("test2") + if err != nil { + t.Error(t) + } + + if len(partitions2) != 8 { + t.Error("Unexpected partitions returned:", len(partitions2)) + } + + if len(trm.errors) != 0 { + t.Errorf("Expected no expectation failures to be set on the error reporter.") + } +} + +func TestConsumerUnexpectedTopicMetadata(t *testing.T) { + trm := newTestReporterMock() + consumer := NewConsumer(trm, nil) + + if _, err := consumer.Topics(); err != sarama.ErrOutOfBrokers { + t.Error("Expected sarama.ErrOutOfBrokers, found", err) + } + + if len(trm.errors) != 1 { + t.Errorf("Expected an expectation failure to be set on the error reporter.") + } +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/mocks.go b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/mocks.go new file mode 100644 index 0000000000000..ab24beebd7648 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/mocks.go @@ -0,0 +1,43 @@ +/* +Package mocks provides mocks that can be used for testing applications +that use Sarama. The mock types provided by this package implement the +interfaces Sarama exports, so you can use them for dependency injection +in your tests. + +All mock instances require you to set expectations on them before you +can use them. It will determine how the mock will behave. If an +expectation is not met, it will make your test fail. + +NOTE: this package currently does not fall under the API stability +guarantee of Sarama as it is still considered experimental. +*/ +package mocks + +import ( + "errors" + + "github.com/Shopify/sarama" +) + +// A simple interface that includes the testing.T methods we use to report +// expectation violations when using the mock objects. +type ErrorReporter interface { + Errorf(string, ...interface{}) +} + +var ( + errProduceSuccess error = nil + errOutOfExpectations = errors.New("No more expectations set on mock") + errPartitionConsumerNotStarted = errors.New("The partition consumer was never started") +) + +const AnyOffset int64 = -1000 + +type producerExpectation struct { + Result error +} + +type consumerExpectation struct { + Err error + Msg *sarama.ConsumerMessage +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/sync_producer.go b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/sync_producer.go new file mode 100644 index 0000000000000..be59ecdb42d7a --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/sync_producer.go @@ -0,0 +1,93 @@ +package mocks + +import ( + "github.com/Shopify/sarama" + "sync" +) + +// SyncProducer implements sarama's SyncProducer interface for testing purposes. +// Before you can use it, you have to set expectations on the mock SyncProducer +// to tell it how to handle calls to SendMessage, so you can easily test success +// and failure scenarios. +type SyncProducer struct { + l sync.Mutex + t ErrorReporter + expectations []*producerExpectation + lastOffset int64 +} + +// NewSyncProducer instantiates a new SyncProducer mock. The t argument should +// be the *testing.T instance of your test method. An error will be written to it if +// an expectation is violated. The config argument is currently unused, but is +// maintained to be compatible with the async Producer. +func NewSyncProducer(t ErrorReporter, config *sarama.Config) *SyncProducer { + return &SyncProducer{ + t: t, + expectations: make([]*producerExpectation, 0), + } +} + +//////////////////////////////////////////////// +// Implement SyncProducer interface +//////////////////////////////////////////////// + +// SendMessage corresponds with the SendMessage method of sarama's SyncProducer implementation. +// You have to set expectations on the mock producer before calling SendMessage, so it knows +// how to handle them. If there is no more remaining expectations when SendMessage is called, +// the mock producer will write an error to the test state object. +func (sp *SyncProducer) SendMessage(msg *sarama.ProducerMessage) (partition int32, offset int64, err error) { + sp.l.Lock() + defer sp.l.Unlock() + + if len(sp.expectations) > 0 { + expectation := sp.expectations[0] + sp.expectations = sp.expectations[1:] + + if expectation.Result == errProduceSuccess { + sp.lastOffset++ + msg.Offset = sp.lastOffset + return 0, msg.Offset, nil + } else { + return -1, -1, expectation.Result + } + } else { + sp.t.Errorf("No more expectation set on this mock producer to handle the input message.") + return -1, -1, errOutOfExpectations + } +} + +// Close corresponds with the Close method of sarama's SyncProducer implementation. +// By closing a mock syncproducer, you also tell it that no more SendMessage calls will follow, +// so it will write an error to the test state if there's any remaining expectations. +func (sp *SyncProducer) Close() error { + sp.l.Lock() + defer sp.l.Unlock() + + if len(sp.expectations) > 0 { + sp.t.Errorf("Expected to exhaust all expectations, but %d are left.", len(sp.expectations)) + } + + return nil +} + +//////////////////////////////////////////////// +// Setting expectations +//////////////////////////////////////////////// + +// ExpectSendMessageAndSucceed sets an expectation on the mock producer that SendMessage will be +// called. The mock producer will handle the message as if it produced successfully, i.e. by +// returning a valid partition, and offset, and a nil error. +func (sp *SyncProducer) ExpectSendMessageAndSucceed() { + sp.l.Lock() + defer sp.l.Unlock() + sp.expectations = append(sp.expectations, &producerExpectation{Result: errProduceSuccess}) +} + +// ExpectSendMessageAndFail sets an expectation on the mock producer that SendMessage will be +// called. The mock producer will handle the message as if it failed to produce +// successfully, i.e. by returning the provided error. +func (sp *SyncProducer) ExpectSendMessageAndFail(err error) { + sp.l.Lock() + defer sp.l.Unlock() + sp.expectations = append(sp.expectations, &producerExpectation{Result: err}) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/sync_producer_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/sync_producer_test.go new file mode 100644 index 0000000000000..a674138e90f11 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/mocks/sync_producer_test.go @@ -0,0 +1,98 @@ +package mocks + +import ( + "testing" + + "github.com/Shopify/sarama" +) + +func TestMockSyncProducerImplementsSyncProducerInterface(t *testing.T) { + var mp interface{} = &SyncProducer{} + if _, ok := mp.(sarama.SyncProducer); !ok { + t.Error("The mock async producer should implement the sarama.SyncProducer interface.") + } +} + +func TestSyncProducerReturnsExpectationsToSendMessage(t *testing.T) { + sp := NewSyncProducer(t, nil) + defer func() { + if err := sp.Close(); err != nil { + t.Error(err) + } + }() + + sp.ExpectSendMessageAndSucceed() + sp.ExpectSendMessageAndSucceed() + sp.ExpectSendMessageAndFail(sarama.ErrOutOfBrokers) + + msg := &sarama.ProducerMessage{Topic: "test", Value: sarama.StringEncoder("test")} + + _, offset, err := sp.SendMessage(msg) + if err != nil { + t.Errorf("The first message should have been produced successfully, but got %s", err) + } + if offset != 1 || offset != msg.Offset { + t.Errorf("The first message should have been assigned offset 1, but got %d", msg.Offset) + } + + _, offset, err = sp.SendMessage(msg) + if err != nil { + t.Errorf("The second message should have been produced successfully, but got %s", err) + } + if offset != 2 || offset != msg.Offset { + t.Errorf("The second message should have been assigned offset 2, but got %d", offset) + } + + _, _, err = sp.SendMessage(msg) + if err != sarama.ErrOutOfBrokers { + t.Errorf("The third message should not have been produced successfully") + } + + if err := sp.Close(); err != nil { + t.Error(err) + } +} + +func TestSyncProducerWithTooManyExpectations(t *testing.T) { + trm := newTestReporterMock() + + sp := NewSyncProducer(trm, nil) + sp.ExpectSendMessageAndSucceed() + sp.ExpectSendMessageAndFail(sarama.ErrOutOfBrokers) + + msg := &sarama.ProducerMessage{Topic: "test", Value: sarama.StringEncoder("test")} + if _, _, err := sp.SendMessage(msg); err != nil { + t.Error("No error expected on first SendMessage call", err) + } + + if err := sp.Close(); err != nil { + t.Error(err) + } + + if len(trm.errors) != 1 { + t.Error("Expected to report an error") + } +} + +func TestSyncProducerWithTooFewExpectations(t *testing.T) { + trm := newTestReporterMock() + + sp := NewSyncProducer(trm, nil) + sp.ExpectSendMessageAndSucceed() + + msg := &sarama.ProducerMessage{Topic: "test", Value: sarama.StringEncoder("test")} + if _, _, err := sp.SendMessage(msg); err != nil { + t.Error("No error expected on first SendMessage call", err) + } + if _, _, err := sp.SendMessage(msg); err != errOutOfExpectations { + t.Error("errOutOfExpectations expected on second SendMessage call, found:", err) + } + + if err := sp.Close(); err != nil { + t.Error(err) + } + + if len(trm.errors) != 1 { + t.Error("Expected to report an error") + } +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/offset_commit_request.go b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_commit_request.go new file mode 100644 index 0000000000000..ba4ac76aa21c6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_commit_request.go @@ -0,0 +1,172 @@ +package sarama + +// ReceiveTime is a special value for the timestamp field of Offset Commit Requests which +// tells the broker to set the timestamp to the time at which the request was received. +// The timestamp is only used if message version 1 is used, which requires kafka 0.8.2. +const ReceiveTime int64 = -1 + +type offsetCommitRequestBlock struct { + offset int64 + timestamp int64 + metadata string +} + +func (r *offsetCommitRequestBlock) encode(pe packetEncoder, version int16) error { + pe.putInt64(r.offset) + if version == 1 { + pe.putInt64(r.timestamp) + } else if r.timestamp != 0 { + Logger.Println("Non-zero timestamp specified for OffsetCommitRequest not v1, it will be ignored") + } + + return pe.putString(r.metadata) +} + +func (r *offsetCommitRequestBlock) decode(pd packetDecoder, version int16) (err error) { + if r.offset, err = pd.getInt64(); err != nil { + return err + } + if version == 1 { + if r.timestamp, err = pd.getInt64(); err != nil { + return err + } + } + r.metadata, err = pd.getString() + return err +} + +type OffsetCommitRequest struct { + ConsumerGroup string + ConsumerGroupGeneration int32 // v1 or later + ConsumerID string // v1 or later + RetentionTime int64 // v2 or later + + // Version can be: + // - 0 (kafka 0.8.1 and later) + // - 1 (kafka 0.8.2 and later) + // - 2 (kafka 0.8.3 and later) + Version int16 + blocks map[string]map[int32]*offsetCommitRequestBlock +} + +func (r *OffsetCommitRequest) encode(pe packetEncoder) error { + if r.Version < 0 || r.Version > 2 { + return PacketEncodingError{"invalid or unsupported OffsetCommitRequest version field"} + } + + if err := pe.putString(r.ConsumerGroup); err != nil { + return err + } + + if r.Version >= 1 { + pe.putInt32(r.ConsumerGroupGeneration) + if err := pe.putString(r.ConsumerID); err != nil { + return err + } + } else { + if r.ConsumerGroupGeneration != 0 { + Logger.Println("Non-zero ConsumerGroupGeneration specified for OffsetCommitRequest v0, it will be ignored") + } + if r.ConsumerID != "" { + Logger.Println("Non-empty ConsumerID specified for OffsetCommitRequest v0, it will be ignored") + } + } + + if r.Version >= 2 { + pe.putInt64(r.RetentionTime) + } else if r.RetentionTime != 0 { + Logger.Println("Non-zero RetentionTime specified for OffsetCommitRequest version <2, it will be ignored") + } + + if err := pe.putArrayLength(len(r.blocks)); err != nil { + return err + } + for topic, partitions := range r.blocks { + if err := pe.putString(topic); err != nil { + return err + } + if err := pe.putArrayLength(len(partitions)); err != nil { + return err + } + for partition, block := range partitions { + pe.putInt32(partition) + if err := block.encode(pe, r.Version); err != nil { + return err + } + } + } + return nil +} + +func (r *OffsetCommitRequest) decode(pd packetDecoder) (err error) { + if r.ConsumerGroup, err = pd.getString(); err != nil { + return err + } + + if r.Version >= 1 { + if r.ConsumerGroupGeneration, err = pd.getInt32(); err != nil { + return err + } + if r.ConsumerID, err = pd.getString(); err != nil { + return err + } + } + + if r.Version >= 2 { + if r.RetentionTime, err = pd.getInt64(); err != nil { + return err + } + } + + topicCount, err := pd.getArrayLength() + if err != nil { + return err + } + if topicCount == 0 { + return nil + } + r.blocks = make(map[string]map[int32]*offsetCommitRequestBlock) + for i := 0; i < topicCount; i++ { + topic, err := pd.getString() + if err != nil { + return err + } + partitionCount, err := pd.getArrayLength() + if err != nil { + return err + } + r.blocks[topic] = make(map[int32]*offsetCommitRequestBlock) + for j := 0; j < partitionCount; j++ { + partition, err := pd.getInt32() + if err != nil { + return err + } + block := &offsetCommitRequestBlock{} + if err := block.decode(pd, r.Version); err != nil { + return err + } + r.blocks[topic][partition] = block + } + } + return nil +} + +func (r *OffsetCommitRequest) key() int16 { + return 8 +} + +func (r *OffsetCommitRequest) version() int16 { + return r.Version +} + +func (r *OffsetCommitRequest) AddBlock(topic string, partitionID int32, offset int64, timestamp int64, metadata string) { + if r.blocks == nil { + r.blocks = make(map[string]map[int32]*offsetCommitRequestBlock) + } + + if r.blocks[topic] == nil { + r.blocks[topic] = make(map[int32]*offsetCommitRequestBlock) + } + + r.blocks[topic][partitionID] = &offsetCommitRequestBlock{offset, timestamp, metadata} +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/offset_commit_request_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_commit_request_test.go new file mode 100644 index 0000000000000..afc25b7b380c5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_commit_request_test.go @@ -0,0 +1,90 @@ +package sarama + +import "testing" + +var ( + offsetCommitRequestNoBlocksV0 = []byte{ + 0x00, 0x06, 'f', 'o', 'o', 'b', 'a', 'r', + 0x00, 0x00, 0x00, 0x00} + + offsetCommitRequestNoBlocksV1 = []byte{ + 0x00, 0x06, 'f', 'o', 'o', 'b', 'a', 'r', + 0x00, 0x00, 0x11, 0x22, + 0x00, 0x04, 'c', 'o', 'n', 's', + 0x00, 0x00, 0x00, 0x00} + + offsetCommitRequestNoBlocksV2 = []byte{ + 0x00, 0x06, 'f', 'o', 'o', 'b', 'a', 'r', + 0x00, 0x00, 0x11, 0x22, + 0x00, 0x04, 'c', 'o', 'n', 's', + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x44, 0x33, + 0x00, 0x00, 0x00, 0x00} + + offsetCommitRequestOneBlockV0 = []byte{ + 0x00, 0x06, 'f', 'o', 'o', 'b', 'a', 'r', + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x05, 't', 'o', 'p', 'i', 'c', + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x52, 0x21, + 0x00, 0x00, 0x00, 0x00, 0xDE, 0xAD, 0xBE, 0xEF, + 0x00, 0x08, 'm', 'e', 't', 'a', 'd', 'a', 't', 'a'} + + offsetCommitRequestOneBlockV1 = []byte{ + 0x00, 0x06, 'f', 'o', 'o', 'b', 'a', 'r', + 0x00, 0x00, 0x11, 0x22, + 0x00, 0x04, 'c', 'o', 'n', 's', + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x05, 't', 'o', 'p', 'i', 'c', + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x52, 0x21, + 0x00, 0x00, 0x00, 0x00, 0xDE, 0xAD, 0xBE, 0xEF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0x00, 0x08, 'm', 'e', 't', 'a', 'd', 'a', 't', 'a'} + + offsetCommitRequestOneBlockV2 = []byte{ + 0x00, 0x06, 'f', 'o', 'o', 'b', 'a', 'r', + 0x00, 0x00, 0x11, 0x22, + 0x00, 0x04, 'c', 'o', 'n', 's', + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x44, 0x33, + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x05, 't', 'o', 'p', 'i', 'c', + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x52, 0x21, + 0x00, 0x00, 0x00, 0x00, 0xDE, 0xAD, 0xBE, 0xEF, + 0x00, 0x08, 'm', 'e', 't', 'a', 'd', 'a', 't', 'a'} +) + +func TestOffsetCommitRequestV0(t *testing.T) { + request := new(OffsetCommitRequest) + request.Version = 0 + request.ConsumerGroup = "foobar" + testRequest(t, "no blocks v0", request, offsetCommitRequestNoBlocksV0) + + request.AddBlock("topic", 0x5221, 0xDEADBEEF, 0, "metadata") + testRequest(t, "one block v0", request, offsetCommitRequestOneBlockV0) +} + +func TestOffsetCommitRequestV1(t *testing.T) { + request := new(OffsetCommitRequest) + request.ConsumerGroup = "foobar" + request.ConsumerID = "cons" + request.ConsumerGroupGeneration = 0x1122 + request.Version = 1 + testRequest(t, "no blocks v1", request, offsetCommitRequestNoBlocksV1) + + request.AddBlock("topic", 0x5221, 0xDEADBEEF, ReceiveTime, "metadata") + testRequest(t, "one block v1", request, offsetCommitRequestOneBlockV1) +} + +func TestOffsetCommitRequestV2(t *testing.T) { + request := new(OffsetCommitRequest) + request.ConsumerGroup = "foobar" + request.ConsumerID = "cons" + request.ConsumerGroupGeneration = 0x1122 + request.RetentionTime = 0x4433 + request.Version = 2 + testRequest(t, "no blocks v2", request, offsetCommitRequestNoBlocksV2) + + request.AddBlock("topic", 0x5221, 0xDEADBEEF, 0, "metadata") + testRequest(t, "one block v2", request, offsetCommitRequestOneBlockV2) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/offset_commit_response.go b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_commit_response.go new file mode 100644 index 0000000000000..573a3b6a100f4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_commit_response.go @@ -0,0 +1,73 @@ +package sarama + +type OffsetCommitResponse struct { + Errors map[string]map[int32]KError +} + +func (r *OffsetCommitResponse) AddError(topic string, partition int32, kerror KError) { + if r.Errors == nil { + r.Errors = make(map[string]map[int32]KError) + } + partitions := r.Errors[topic] + if partitions == nil { + partitions = make(map[int32]KError) + r.Errors[topic] = partitions + } + partitions[partition] = kerror +} + +func (r *OffsetCommitResponse) encode(pe packetEncoder) error { + if err := pe.putArrayLength(len(r.Errors)); err != nil { + return err + } + for topic, partitions := range r.Errors { + if err := pe.putString(topic); err != nil { + return err + } + if err := pe.putArrayLength(len(partitions)); err != nil { + return err + } + for partition, kerror := range partitions { + pe.putInt32(partition) + pe.putInt16(int16(kerror)) + } + } + return nil +} + +func (r *OffsetCommitResponse) decode(pd packetDecoder) (err error) { + numTopics, err := pd.getArrayLength() + if err != nil || numTopics == 0 { + return err + } + + r.Errors = make(map[string]map[int32]KError, numTopics) + for i := 0; i < numTopics; i++ { + name, err := pd.getString() + if err != nil { + return err + } + + numErrors, err := pd.getArrayLength() + if err != nil { + return err + } + + r.Errors[name] = make(map[int32]KError, numErrors) + + for j := 0; j < numErrors; j++ { + id, err := pd.getInt32() + if err != nil { + return err + } + + tmp, err := pd.getInt16() + if err != nil { + return err + } + r.Errors[name][id] = KError(tmp) + } + } + + return nil +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/offset_commit_response_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_commit_response_test.go new file mode 100644 index 0000000000000..074ec923220a5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_commit_response_test.go @@ -0,0 +1,24 @@ +package sarama + +import ( + "testing" +) + +var ( + emptyOffsetCommitResponse = []byte{ + 0x00, 0x00, 0x00, 0x00} +) + +func TestEmptyOffsetCommitResponse(t *testing.T) { + response := OffsetCommitResponse{} + testResponse(t, "empty", &response, emptyOffsetCommitResponse) +} + +func TestNormalOffsetCommitResponse(t *testing.T) { + response := OffsetCommitResponse{} + response.AddError("t", 0, ErrNotLeaderForPartition) + response.Errors["m"] = make(map[int32]KError) + // The response encoded form cannot be checked for it varies due to + // unpredictable map traversal order. + testResponse(t, "normal", &response, nil) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/offset_fetch_request.go b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_fetch_request.go new file mode 100644 index 0000000000000..30bbbbbd0d219 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_fetch_request.go @@ -0,0 +1,71 @@ +package sarama + +type OffsetFetchRequest struct { + ConsumerGroup string + Version int16 + partitions map[string][]int32 +} + +func (r *OffsetFetchRequest) encode(pe packetEncoder) (err error) { + if r.Version < 0 || r.Version > 1 { + return PacketEncodingError{"invalid or unsupported OffsetFetchRequest version field"} + } + + if err = pe.putString(r.ConsumerGroup); err != nil { + return err + } + if err = pe.putArrayLength(len(r.partitions)); err != nil { + return err + } + for topic, partitions := range r.partitions { + if err = pe.putString(topic); err != nil { + return err + } + if err = pe.putInt32Array(partitions); err != nil { + return err + } + } + return nil +} + +func (r *OffsetFetchRequest) decode(pd packetDecoder) (err error) { + if r.ConsumerGroup, err = pd.getString(); err != nil { + return err + } + partitionCount, err := pd.getArrayLength() + if err != nil { + return err + } + if partitionCount == 0 { + return nil + } + r.partitions = make(map[string][]int32) + for i := 0; i < partitionCount; i++ { + topic, err := pd.getString() + if err != nil { + return err + } + partitions, err := pd.getInt32Array() + if err != nil { + return err + } + r.partitions[topic] = partitions + } + return nil +} + +func (r *OffsetFetchRequest) key() int16 { + return 9 +} + +func (r *OffsetFetchRequest) version() int16 { + return r.Version +} + +func (r *OffsetFetchRequest) AddPartition(topic string, partitionID int32) { + if r.partitions == nil { + r.partitions = make(map[string][]int32) + } + + r.partitions[topic] = append(r.partitions[topic], partitionID) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/offset_fetch_request_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_fetch_request_test.go new file mode 100644 index 0000000000000..025d725c986b0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_fetch_request_test.go @@ -0,0 +1,31 @@ +package sarama + +import "testing" + +var ( + offsetFetchRequestNoGroupNoPartitions = []byte{ + 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00} + + offsetFetchRequestNoPartitions = []byte{ + 0x00, 0x04, 'b', 'l', 'a', 'h', + 0x00, 0x00, 0x00, 0x00} + + offsetFetchRequestOnePartition = []byte{ + 0x00, 0x04, 'b', 'l', 'a', 'h', + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x0D, 't', 'o', 'p', 'i', 'c', 'T', 'h', 'e', 'F', 'i', 'r', 's', 't', + 0x00, 0x00, 0x00, 0x01, + 0x4F, 0x4F, 0x4F, 0x4F} +) + +func TestOffsetFetchRequest(t *testing.T) { + request := new(OffsetFetchRequest) + testRequest(t, "no group, no partitions", request, offsetFetchRequestNoGroupNoPartitions) + + request.ConsumerGroup = "blah" + testRequest(t, "no partitions", request, offsetFetchRequestNoPartitions) + + request.AddPartition("topicTheFirst", 0x4F4F4F4F) + testRequest(t, "one partition", request, offsetFetchRequestOnePartition) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/offset_fetch_response.go b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_fetch_response.go new file mode 100644 index 0000000000000..93078c350f8da --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_fetch_response.go @@ -0,0 +1,131 @@ +package sarama + +type OffsetFetchResponseBlock struct { + Offset int64 + Metadata string + Err KError +} + +func (r *OffsetFetchResponseBlock) decode(pd packetDecoder) (err error) { + r.Offset, err = pd.getInt64() + if err != nil { + return err + } + + r.Metadata, err = pd.getString() + if err != nil { + return err + } + + tmp, err := pd.getInt16() + if err != nil { + return err + } + r.Err = KError(tmp) + + return nil +} + +func (r *OffsetFetchResponseBlock) encode(pe packetEncoder) (err error) { + pe.putInt64(r.Offset) + + err = pe.putString(r.Metadata) + if err != nil { + return err + } + + pe.putInt16(int16(r.Err)) + + return nil +} + +type OffsetFetchResponse struct { + Blocks map[string]map[int32]*OffsetFetchResponseBlock +} + +func (r *OffsetFetchResponse) encode(pe packetEncoder) error { + if err := pe.putArrayLength(len(r.Blocks)); err != nil { + return err + } + for topic, partitions := range r.Blocks { + if err := pe.putString(topic); err != nil { + return err + } + if err := pe.putArrayLength(len(partitions)); err != nil { + return err + } + for partition, block := range partitions { + pe.putInt32(partition) + if err := block.encode(pe); err != nil { + return err + } + } + } + return nil +} + +func (r *OffsetFetchResponse) decode(pd packetDecoder) (err error) { + numTopics, err := pd.getArrayLength() + if err != nil || numTopics == 0 { + return err + } + + r.Blocks = make(map[string]map[int32]*OffsetFetchResponseBlock, numTopics) + for i := 0; i < numTopics; i++ { + name, err := pd.getString() + if err != nil { + return err + } + + numBlocks, err := pd.getArrayLength() + if err != nil { + return err + } + + if numBlocks == 0 { + r.Blocks[name] = nil + continue + } + r.Blocks[name] = make(map[int32]*OffsetFetchResponseBlock, numBlocks) + + for j := 0; j < numBlocks; j++ { + id, err := pd.getInt32() + if err != nil { + return err + } + + block := new(OffsetFetchResponseBlock) + err = block.decode(pd) + if err != nil { + return err + } + r.Blocks[name][id] = block + } + } + + return nil +} + +func (r *OffsetFetchResponse) GetBlock(topic string, partition int32) *OffsetFetchResponseBlock { + if r.Blocks == nil { + return nil + } + + if r.Blocks[topic] == nil { + return nil + } + + return r.Blocks[topic][partition] +} + +func (r *OffsetFetchResponse) AddBlock(topic string, partition int32, block *OffsetFetchResponseBlock) { + if r.Blocks == nil { + r.Blocks = make(map[string]map[int32]*OffsetFetchResponseBlock) + } + partitions := r.Blocks[topic] + if partitions == nil { + partitions = make(map[int32]*OffsetFetchResponseBlock) + r.Blocks[topic] = partitions + } + partitions[partition] = block +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/offset_fetch_response_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_fetch_response_test.go new file mode 100644 index 0000000000000..7614ae4249309 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_fetch_response_test.go @@ -0,0 +1,22 @@ +package sarama + +import "testing" + +var ( + emptyOffsetFetchResponse = []byte{ + 0x00, 0x00, 0x00, 0x00} +) + +func TestEmptyOffsetFetchResponse(t *testing.T) { + response := OffsetFetchResponse{} + testResponse(t, "empty", &response, emptyOffsetFetchResponse) +} + +func TestNormalOffsetFetchResponse(t *testing.T) { + response := OffsetFetchResponse{} + response.AddBlock("t", 0, &OffsetFetchResponseBlock{0, "md", ErrRequestTimedOut}) + response.Blocks["m"] = nil + // The response encoded form cannot be checked for it varies due to + // unpredictable map traversal order. + testResponse(t, "normal", &response, nil) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/offset_request.go b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_request.go new file mode 100644 index 0000000000000..842d5c0f8360c --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_request.go @@ -0,0 +1,113 @@ +package sarama + +type offsetRequestBlock struct { + time int64 + maxOffsets int32 +} + +func (r *offsetRequestBlock) encode(pe packetEncoder) error { + pe.putInt64(int64(r.time)) + pe.putInt32(r.maxOffsets) + return nil +} + +func (r *offsetRequestBlock) decode(pd packetDecoder) (err error) { + if r.time, err = pd.getInt64(); err != nil { + return err + } + if r.maxOffsets, err = pd.getInt32(); err != nil { + return err + } + return nil +} + +type OffsetRequest struct { + blocks map[string]map[int32]*offsetRequestBlock +} + +func (r *OffsetRequest) encode(pe packetEncoder) error { + pe.putInt32(-1) // replica ID is always -1 for clients + err := pe.putArrayLength(len(r.blocks)) + if err != nil { + return err + } + for topic, partitions := range r.blocks { + err = pe.putString(topic) + if err != nil { + return err + } + err = pe.putArrayLength(len(partitions)) + if err != nil { + return err + } + for partition, block := range partitions { + pe.putInt32(partition) + if err = block.encode(pe); err != nil { + return err + } + } + } + return nil +} + +func (r *OffsetRequest) decode(pd packetDecoder) error { + // Ignore replica ID + if _, err := pd.getInt32(); err != nil { + return err + } + blockCount, err := pd.getArrayLength() + if err != nil { + return err + } + if blockCount == 0 { + return nil + } + r.blocks = make(map[string]map[int32]*offsetRequestBlock) + for i := 0; i < blockCount; i++ { + topic, err := pd.getString() + if err != nil { + return err + } + partitionCount, err := pd.getArrayLength() + if err != nil { + return err + } + r.blocks[topic] = make(map[int32]*offsetRequestBlock) + for j := 0; j < partitionCount; j++ { + partition, err := pd.getInt32() + if err != nil { + return err + } + block := &offsetRequestBlock{} + if err := block.decode(pd); err != nil { + return err + } + r.blocks[topic][partition] = block + } + } + return nil +} + +func (r *OffsetRequest) key() int16 { + return 2 +} + +func (r *OffsetRequest) version() int16 { + return 0 +} + +func (r *OffsetRequest) AddBlock(topic string, partitionID int32, time int64, maxOffsets int32) { + if r.blocks == nil { + r.blocks = make(map[string]map[int32]*offsetRequestBlock) + } + + if r.blocks[topic] == nil { + r.blocks[topic] = make(map[int32]*offsetRequestBlock) + } + + tmp := new(offsetRequestBlock) + tmp.time = time + tmp.maxOffsets = maxOffsets + + r.blocks[topic][partitionID] = tmp +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/offset_request_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_request_test.go new file mode 100644 index 0000000000000..f3b3046bbb511 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_request_test.go @@ -0,0 +1,26 @@ +package sarama + +import "testing" + +var ( + offsetRequestNoBlocks = []byte{ + 0xFF, 0xFF, 0xFF, 0xFF, + 0x00, 0x00, 0x00, 0x00} + + offsetRequestOneBlock = []byte{ + 0xFF, 0xFF, 0xFF, 0xFF, + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x03, 'f', 'o', 'o', + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x00, 0x04, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x00, 0x02} +) + +func TestOffsetRequest(t *testing.T) { + request := new(OffsetRequest) + testRequest(t, "no blocks", request, offsetRequestNoBlocks) + + request.AddBlock("foo", 4, 1, 2) + testRequest(t, "one block", request, offsetRequestOneBlock) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/offset_response.go b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_response.go new file mode 100644 index 0000000000000..07d71ca7218bc --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_response.go @@ -0,0 +1,130 @@ +package sarama + +type OffsetResponseBlock struct { + Err KError + Offsets []int64 +} + +func (r *OffsetResponseBlock) decode(pd packetDecoder) (err error) { + tmp, err := pd.getInt16() + if err != nil { + return err + } + r.Err = KError(tmp) + + r.Offsets, err = pd.getInt64Array() + + return err +} + +func (r *OffsetResponseBlock) encode(pe packetEncoder) (err error) { + pe.putInt16(int16(r.Err)) + + return pe.putInt64Array(r.Offsets) +} + +type OffsetResponse struct { + Blocks map[string]map[int32]*OffsetResponseBlock +} + +func (r *OffsetResponse) decode(pd packetDecoder) (err error) { + numTopics, err := pd.getArrayLength() + if err != nil { + return err + } + + r.Blocks = make(map[string]map[int32]*OffsetResponseBlock, numTopics) + for i := 0; i < numTopics; i++ { + name, err := pd.getString() + if err != nil { + return err + } + + numBlocks, err := pd.getArrayLength() + if err != nil { + return err + } + + r.Blocks[name] = make(map[int32]*OffsetResponseBlock, numBlocks) + + for j := 0; j < numBlocks; j++ { + id, err := pd.getInt32() + if err != nil { + return err + } + + block := new(OffsetResponseBlock) + err = block.decode(pd) + if err != nil { + return err + } + r.Blocks[name][id] = block + } + } + + return nil +} + +func (r *OffsetResponse) GetBlock(topic string, partition int32) *OffsetResponseBlock { + if r.Blocks == nil { + return nil + } + + if r.Blocks[topic] == nil { + return nil + } + + return r.Blocks[topic][partition] +} + +/* +// [0 0 0 1 ntopics +0 8 109 121 95 116 111 112 105 99 topic +0 0 0 1 npartitions +0 0 0 0 id +0 0 + +0 0 0 1 0 0 0 0 +0 1 1 1 0 0 0 1 +0 8 109 121 95 116 111 112 +105 99 0 0 0 1 0 0 +0 0 0 0 0 0 0 1 +0 0 0 0 0 1 1 1] + +*/ +func (r *OffsetResponse) encode(pe packetEncoder) (err error) { + if err = pe.putArrayLength(len(r.Blocks)); err != nil { + return err + } + + for topic, partitions := range r.Blocks { + if err = pe.putString(topic); err != nil { + return err + } + if err = pe.putArrayLength(len(partitions)); err != nil { + return err + } + for partition, block := range partitions { + pe.putInt32(partition) + if err = block.encode(pe); err != nil { + return err + } + } + } + + return nil +} + +// testing API + +func (r *OffsetResponse) AddTopicPartition(topic string, partition int32, offset int64) { + if r.Blocks == nil { + r.Blocks = make(map[string]map[int32]*OffsetResponseBlock) + } + byTopic, ok := r.Blocks[topic] + if !ok { + byTopic = make(map[int32]*OffsetResponseBlock) + r.Blocks[topic] = byTopic + } + byTopic[partition] = &OffsetResponseBlock{Offsets: []int64{offset}} +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/offset_response_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_response_test.go new file mode 100644 index 0000000000000..a427cbd20c62d --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/offset_response_test.go @@ -0,0 +1,62 @@ +package sarama + +import "testing" + +var ( + emptyOffsetResponse = []byte{ + 0x00, 0x00, 0x00, 0x00} + + normalOffsetResponse = []byte{ + 0x00, 0x00, 0x00, 0x02, + + 0x00, 0x01, 'a', + 0x00, 0x00, 0x00, 0x00, + + 0x00, 0x01, 'z', + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x00, 0x02, + 0x00, 0x00, + 0x00, 0x00, 0x00, 0x02, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x05, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x06} +) + +func TestEmptyOffsetResponse(t *testing.T) { + response := OffsetResponse{} + + testDecodable(t, "empty", &response, emptyOffsetResponse) + if len(response.Blocks) != 0 { + t.Error("Decoding produced", len(response.Blocks), "topics where there were none.") + } +} + +func TestNormalOffsetResponse(t *testing.T) { + response := OffsetResponse{} + + testDecodable(t, "normal", &response, normalOffsetResponse) + + if len(response.Blocks) != 2 { + t.Fatal("Decoding produced", len(response.Blocks), "topics where there were two.") + } + + if len(response.Blocks["a"]) != 0 { + t.Fatal("Decoding produced", len(response.Blocks["a"]), "partitions for topic 'a' where there were none.") + } + + if len(response.Blocks["z"]) != 1 { + t.Fatal("Decoding produced", len(response.Blocks["z"]), "partitions for topic 'z' where there was one.") + } + + if response.Blocks["z"][2].Err != ErrNoError { + t.Fatal("Decoding produced invalid error for topic z partition 2.") + } + + if len(response.Blocks["z"][2].Offsets) != 2 { + t.Fatal("Decoding produced invalid number of offsets for topic z partition 2.") + } + + if response.Blocks["z"][2].Offsets[0] != 5 || response.Blocks["z"][2].Offsets[1] != 6 { + t.Fatal("Decoding produced invalid offsets for topic z partition 2.") + } + +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/packet_decoder.go b/Godeps/_workspace/src/github.com/Shopify/sarama/packet_decoder.go new file mode 100644 index 0000000000000..0342223136935 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/packet_decoder.go @@ -0,0 +1,44 @@ +package sarama + +// PacketDecoder is the interface providing helpers for reading with Kafka's encoding rules. +// Types implementing Decoder only need to worry about calling methods like GetString, +// not about how a string is represented in Kafka. +type packetDecoder interface { + // Primitives + getInt8() (int8, error) + getInt16() (int16, error) + getInt32() (int32, error) + getInt64() (int64, error) + getArrayLength() (int, error) + + // Collections + getBytes() ([]byte, error) + getString() (string, error) + getInt32Array() ([]int32, error) + getInt64Array() ([]int64, error) + + // Subsets + remaining() int + getSubset(length int) (packetDecoder, error) + + // Stacks, see PushDecoder + push(in pushDecoder) error + pop() error +} + +// PushDecoder is the interface for decoding fields like CRCs and lengths where the validity +// of the field depends on what is after it in the packet. Start them with PacketDecoder.Push() where +// the actual value is located in the packet, then PacketDecoder.Pop() them when all the bytes they +// depend upon have been decoded. +type pushDecoder interface { + // Saves the offset into the input buffer as the location to actually read the calculated value when able. + saveOffset(in int) + + // Returns the length of data to reserve for the input of this encoder (eg 4 bytes for a CRC32). + reserveLength() int + + // Indicates that all required data is now available to calculate and check the field. + // SaveOffset is guaranteed to have been called first. The implementation should read ReserveLength() bytes + // of data from the saved offset, and verify it based on the data between the saved offset and curOffset. + check(curOffset int, buf []byte) error +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/packet_encoder.go b/Godeps/_workspace/src/github.com/Shopify/sarama/packet_encoder.go new file mode 100644 index 0000000000000..2c5710938ca7b --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/packet_encoder.go @@ -0,0 +1,41 @@ +package sarama + +// PacketEncoder is the interface providing helpers for writing with Kafka's encoding rules. +// Types implementing Encoder only need to worry about calling methods like PutString, +// not about how a string is represented in Kafka. +type packetEncoder interface { + // Primitives + putInt8(in int8) + putInt16(in int16) + putInt32(in int32) + putInt64(in int64) + putArrayLength(in int) error + + // Collections + putBytes(in []byte) error + putRawBytes(in []byte) error + putString(in string) error + putInt32Array(in []int32) error + putInt64Array(in []int64) error + + // Stacks, see PushEncoder + push(in pushEncoder) + pop() error +} + +// PushEncoder is the interface for encoding fields like CRCs and lengths where the value +// of the field depends on what is encoded after it in the packet. Start them with PacketEncoder.Push() where +// the actual value is located in the packet, then PacketEncoder.Pop() them when all the bytes they +// depend upon have been written. +type pushEncoder interface { + // Saves the offset into the input buffer as the location to actually write the calculated value when able. + saveOffset(in int) + + // Returns the length of data to reserve for the output of this encoder (eg 4 bytes for a CRC32). + reserveLength() int + + // Indicates that all required data is now available to calculate and write the field. + // SaveOffset is guaranteed to have been called first. The implementation should write ReserveLength() bytes + // of data to the saved offset, based on the data between the saved offset and curOffset. + run(curOffset int, buf []byte) error +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/partitioner.go b/Godeps/_workspace/src/github.com/Shopify/sarama/partitioner.go new file mode 100644 index 0000000000000..493ee0ec3d43f --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/partitioner.go @@ -0,0 +1,120 @@ +package sarama + +import ( + "hash" + "hash/fnv" + "math/rand" + "time" +) + +// Partitioner is anything that, given a Kafka message and a number of partitions indexed [0...numPartitions-1], +// decides to which partition to send the message. RandomPartitioner, RoundRobinPartitioner and HashPartitioner are provided +// as simple default implementations. +type Partitioner interface { + Partition(message *ProducerMessage, numPartitions int32) (int32, error) // Partition takes a message and partition count and chooses a partition + + // RequiresConsistency indicates to the user of the partitioner whether the mapping of key->partition is consistent or not. + // Specifically, if a partitioner requires consistency then it must be allowed to choose from all partitions (even ones known to + // be unavailable), and its choice must be respected by the caller. The obvious example is the HashPartitioner. + RequiresConsistency() bool +} + +// PartitionerConstructor is the type for a function capable of constructing new Partitioners. +type PartitionerConstructor func(topic string) Partitioner + +type manualPartitioner struct{} + +// NewManualPartitioner returns a Partitioner which uses the partition manually set in the provided +// ProducerMessage's Partition field as the partition to produce to. +func NewManualPartitioner(topic string) Partitioner { + return new(manualPartitioner) +} + +func (p *manualPartitioner) Partition(message *ProducerMessage, numPartitions int32) (int32, error) { + return message.Partition, nil +} + +func (p *manualPartitioner) RequiresConsistency() bool { + return true +} + +type randomPartitioner struct { + generator *rand.Rand +} + +// NewRandomPartitioner returns a Partitioner which chooses a random partition each time. +func NewRandomPartitioner(topic string) Partitioner { + p := new(randomPartitioner) + p.generator = rand.New(rand.NewSource(time.Now().UTC().UnixNano())) + return p +} + +func (p *randomPartitioner) Partition(message *ProducerMessage, numPartitions int32) (int32, error) { + return int32(p.generator.Intn(int(numPartitions))), nil +} + +func (p *randomPartitioner) RequiresConsistency() bool { + return false +} + +type roundRobinPartitioner struct { + partition int32 +} + +// NewRoundRobinPartitioner returns a Partitioner which walks through the available partitions one at a time. +func NewRoundRobinPartitioner(topic string) Partitioner { + return &roundRobinPartitioner{} +} + +func (p *roundRobinPartitioner) Partition(message *ProducerMessage, numPartitions int32) (int32, error) { + if p.partition >= numPartitions { + p.partition = 0 + } + ret := p.partition + p.partition++ + return ret, nil +} + +func (p *roundRobinPartitioner) RequiresConsistency() bool { + return false +} + +type hashPartitioner struct { + random Partitioner + hasher hash.Hash32 +} + +// NewHashPartitioner returns a Partitioner which behaves as follows. If the message's key is nil, or fails to +// encode, then a random partition is chosen. Otherwise the FNV-1a hash of the encoded bytes of the message key +// is used, modulus the number of partitions. This ensures that messages with the same key always end up on the +// same partition. +func NewHashPartitioner(topic string) Partitioner { + p := new(hashPartitioner) + p.random = NewRandomPartitioner(topic) + p.hasher = fnv.New32a() + return p +} + +func (p *hashPartitioner) Partition(message *ProducerMessage, numPartitions int32) (int32, error) { + if message.Key == nil { + return p.random.Partition(message, numPartitions) + } + bytes, err := message.Key.Encode() + if err != nil { + return -1, err + } + p.hasher.Reset() + _, err = p.hasher.Write(bytes) + if err != nil { + return -1, err + } + hash := int32(p.hasher.Sum32()) + if hash < 0 { + hash = -hash + } + return hash % numPartitions, nil +} + +func (p *hashPartitioner) RequiresConsistency() bool { + return true +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/partitioner_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/partitioner_test.go new file mode 100644 index 0000000000000..f44c509d65229 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/partitioner_test.go @@ -0,0 +1,198 @@ +package sarama + +import ( + "crypto/rand" + "log" + "testing" +) + +func assertPartitioningConsistent(t *testing.T, partitioner Partitioner, message *ProducerMessage, numPartitions int32) { + choice, err := partitioner.Partition(message, numPartitions) + if err != nil { + t.Error(partitioner, err) + } + if choice < 0 || choice >= numPartitions { + t.Error(partitioner, "returned partition", choice, "outside of range for", message) + } + for i := 1; i < 50; i++ { + newChoice, err := partitioner.Partition(message, numPartitions) + if err != nil { + t.Error(partitioner, err) + } + if newChoice != choice { + t.Error(partitioner, "returned partition", newChoice, "inconsistent with", choice, ".") + } + } +} + +func TestRandomPartitioner(t *testing.T) { + partitioner := NewRandomPartitioner("mytopic") + + choice, err := partitioner.Partition(nil, 1) + if err != nil { + t.Error(partitioner, err) + } + if choice != 0 { + t.Error("Returned non-zero partition when only one available.") + } + + for i := 1; i < 50; i++ { + choice, err := partitioner.Partition(nil, 50) + if err != nil { + t.Error(partitioner, err) + } + if choice < 0 || choice >= 50 { + t.Error("Returned partition", choice, "outside of range.") + } + } +} + +func TestRoundRobinPartitioner(t *testing.T) { + partitioner := NewRoundRobinPartitioner("mytopic") + + choice, err := partitioner.Partition(nil, 1) + if err != nil { + t.Error(partitioner, err) + } + if choice != 0 { + t.Error("Returned non-zero partition when only one available.") + } + + var i int32 + for i = 1; i < 50; i++ { + choice, err := partitioner.Partition(nil, 7) + if err != nil { + t.Error(partitioner, err) + } + if choice != i%7 { + t.Error("Returned partition", choice, "expecting", i%7) + } + } +} + +func TestHashPartitioner(t *testing.T) { + partitioner := NewHashPartitioner("mytopic") + + choice, err := partitioner.Partition(&ProducerMessage{}, 1) + if err != nil { + t.Error(partitioner, err) + } + if choice != 0 { + t.Error("Returned non-zero partition when only one available.") + } + + for i := 1; i < 50; i++ { + choice, err := partitioner.Partition(&ProducerMessage{}, 50) + if err != nil { + t.Error(partitioner, err) + } + if choice < 0 || choice >= 50 { + t.Error("Returned partition", choice, "outside of range for nil key.") + } + } + + buf := make([]byte, 256) + for i := 1; i < 50; i++ { + if _, err := rand.Read(buf); err != nil { + t.Error(err) + } + assertPartitioningConsistent(t, partitioner, &ProducerMessage{Key: ByteEncoder(buf)}, 50) + } +} + +func TestManualPartitioner(t *testing.T) { + partitioner := NewManualPartitioner("mytopic") + + choice, err := partitioner.Partition(&ProducerMessage{}, 1) + if err != nil { + t.Error(partitioner, err) + } + if choice != 0 { + t.Error("Returned non-zero partition when only one available.") + } + + for i := int32(1); i < 50; i++ { + choice, err := partitioner.Partition(&ProducerMessage{Partition: i}, 50) + if err != nil { + t.Error(partitioner, err) + } + if choice != i { + t.Error("Returned partition not the same as the input partition") + } + } +} + +// By default, Sarama uses the message's key to consistently assign a partition to +// a message using hashing. If no key is set, a random partition will be chosen. +// This example shows how you can partition messages randomly, even when a key is set, +// by overriding Config.Producer.Partitioner. +func ExamplePartitioner_random() { + config := NewConfig() + config.Producer.Partitioner = NewRandomPartitioner + + producer, err := NewSyncProducer([]string{"localhost:9092"}, config) + if err != nil { + log.Fatal(err) + } + defer func() { + if err := producer.Close(); err != nil { + log.Println("Failed to close producer:", err) + } + }() + + msg := &ProducerMessage{Topic: "test", Key: StringEncoder("key is set"), Value: StringEncoder("test")} + partition, offset, err := producer.SendMessage(msg) + if err != nil { + log.Fatalln("Failed to produce message to kafka cluster.") + } + + log.Printf("Produced message to partition %d with offset %d", partition, offset) +} + +// This example shows how to assign partitions to your messages manually. +func ExamplePartitioner_manual() { + config := NewConfig() + + // First, we tell the producer that we are going to partition ourselves. + config.Producer.Partitioner = NewManualPartitioner + + producer, err := NewSyncProducer([]string{"localhost:9092"}, config) + if err != nil { + log.Fatal(err) + } + defer func() { + if err := producer.Close(); err != nil { + log.Println("Failed to close producer:", err) + } + }() + + // Now, we set the Partition field of the ProducerMessage struct. + msg := &ProducerMessage{Topic: "test", Partition: 6, Value: StringEncoder("test")} + + partition, offset, err := producer.SendMessage(msg) + if err != nil { + log.Fatalln("Failed to produce message to kafka cluster.") + } + + if partition != 6 { + log.Fatal("Message should have been produced to partition 6!") + } + + log.Printf("Produced message to partition %d with offset %d", partition, offset) +} + +// This example shows how to set a different partitioner depending on the topic. +func ExamplePartitioner_per_topic() { + config := NewConfig() + config.Producer.Partitioner = func(topic string) Partitioner { + switch topic { + case "access_log", "error_log": + return NewRandomPartitioner(topic) + + default: + return NewHashPartitioner(topic) + } + } + + // ... +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/prep_encoder.go b/Godeps/_workspace/src/github.com/Shopify/sarama/prep_encoder.go new file mode 100644 index 0000000000000..ddeef780e96dc --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/prep_encoder.go @@ -0,0 +1,95 @@ +package sarama + +import ( + "fmt" + "math" +) + +type prepEncoder struct { + length int +} + +// primitives + +func (pe *prepEncoder) putInt8(in int8) { + pe.length += 1 +} + +func (pe *prepEncoder) putInt16(in int16) { + pe.length += 2 +} + +func (pe *prepEncoder) putInt32(in int32) { + pe.length += 4 +} + +func (pe *prepEncoder) putInt64(in int64) { + pe.length += 8 +} + +func (pe *prepEncoder) putArrayLength(in int) error { + if in > math.MaxInt32 { + return PacketEncodingError{fmt.Sprintf("array too long (%d)", in)} + } + pe.length += 4 + return nil +} + +// arrays + +func (pe *prepEncoder) putBytes(in []byte) error { + pe.length += 4 + if in == nil { + return nil + } + if len(in) > math.MaxInt32 { + return PacketEncodingError{fmt.Sprintf("byteslice too long (%d)", len(in))} + } + pe.length += len(in) + return nil +} + +func (pe *prepEncoder) putRawBytes(in []byte) error { + if len(in) > math.MaxInt32 { + return PacketEncodingError{fmt.Sprintf("byteslice too long (%d)", len(in))} + } + pe.length += len(in) + return nil +} + +func (pe *prepEncoder) putString(in string) error { + pe.length += 2 + if len(in) > math.MaxInt16 { + return PacketEncodingError{fmt.Sprintf("string too long (%d)", len(in))} + } + pe.length += len(in) + return nil +} + +func (pe *prepEncoder) putInt32Array(in []int32) error { + err := pe.putArrayLength(len(in)) + if err != nil { + return err + } + pe.length += 4 * len(in) + return nil +} + +func (pe *prepEncoder) putInt64Array(in []int64) error { + err := pe.putArrayLength(len(in)) + if err != nil { + return err + } + pe.length += 8 * len(in) + return nil +} + +// stackable + +func (pe *prepEncoder) push(in pushEncoder) { + pe.length += in.reserveLength() +} + +func (pe *prepEncoder) pop() error { + return nil +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/produce_request.go b/Godeps/_workspace/src/github.com/Shopify/sarama/produce_request.go new file mode 100644 index 0000000000000..f21956137ebc8 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/produce_request.go @@ -0,0 +1,148 @@ +package sarama + +// RequiredAcks is used in Produce Requests to tell the broker how many replica acknowledgements +// it must see before responding. Any of the constants defined here are valid. On broker versions +// prior to 0.8.2.0 any other positive int16 is also valid (the broker will wait for that many +// acknowledgements) but in 0.8.2.0 and later this will raise an exception (it has been replaced +// by setting the `min.isr` value in the brokers configuration). +type RequiredAcks int16 + +const ( + // NoResponse doesn't send any response, the TCP ACK is all you get. + NoResponse RequiredAcks = 0 + // WaitForLocal waits for only the local commit to succeed before responding. + WaitForLocal RequiredAcks = 1 + // WaitForAll waits for all replicas to commit before responding. + WaitForAll RequiredAcks = -1 +) + +type ProduceRequest struct { + RequiredAcks RequiredAcks + Timeout int32 + msgSets map[string]map[int32]*MessageSet +} + +func (p *ProduceRequest) encode(pe packetEncoder) error { + pe.putInt16(int16(p.RequiredAcks)) + pe.putInt32(p.Timeout) + err := pe.putArrayLength(len(p.msgSets)) + if err != nil { + return err + } + for topic, partitions := range p.msgSets { + err = pe.putString(topic) + if err != nil { + return err + } + err = pe.putArrayLength(len(partitions)) + if err != nil { + return err + } + for id, msgSet := range partitions { + pe.putInt32(id) + pe.push(&lengthField{}) + err = msgSet.encode(pe) + if err != nil { + return err + } + err = pe.pop() + if err != nil { + return err + } + } + } + return nil +} + +func (p *ProduceRequest) decode(pd packetDecoder) error { + requiredAcks, err := pd.getInt16() + if err != nil { + return err + } + p.RequiredAcks = RequiredAcks(requiredAcks) + if p.Timeout, err = pd.getInt32(); err != nil { + return err + } + topicCount, err := pd.getArrayLength() + if err != nil { + return err + } + if topicCount == 0 { + return nil + } + p.msgSets = make(map[string]map[int32]*MessageSet) + for i := 0; i < topicCount; i++ { + topic, err := pd.getString() + if err != nil { + return err + } + partitionCount, err := pd.getArrayLength() + if err != nil { + return err + } + p.msgSets[topic] = make(map[int32]*MessageSet) + for j := 0; j < partitionCount; j++ { + partition, err := pd.getInt32() + if err != nil { + return err + } + messageSetSize, err := pd.getInt32() + if err != nil { + return err + } + if messageSetSize == 0 { + continue + } + msgSetDecoder, err := pd.getSubset(int(messageSetSize)) + if err != nil { + return err + } + msgSet := &MessageSet{} + err = msgSet.decode(msgSetDecoder) + if err != nil { + return err + } + p.msgSets[topic][partition] = msgSet + } + } + return nil +} + +func (p *ProduceRequest) key() int16 { + return 0 +} + +func (p *ProduceRequest) version() int16 { + return 0 +} + +func (p *ProduceRequest) AddMessage(topic string, partition int32, msg *Message) { + if p.msgSets == nil { + p.msgSets = make(map[string]map[int32]*MessageSet) + } + + if p.msgSets[topic] == nil { + p.msgSets[topic] = make(map[int32]*MessageSet) + } + + set := p.msgSets[topic][partition] + + if set == nil { + set = new(MessageSet) + p.msgSets[topic][partition] = set + } + + set.addMessage(msg) +} + +func (p *ProduceRequest) AddSet(topic string, partition int32, set *MessageSet) { + if p.msgSets == nil { + p.msgSets = make(map[string]map[int32]*MessageSet) + } + + if p.msgSets[topic] == nil { + p.msgSets[topic] = make(map[int32]*MessageSet) + } + + p.msgSets[topic][partition] = set +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/produce_request_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/produce_request_test.go new file mode 100644 index 0000000000000..21f4ba5b14b31 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/produce_request_test.go @@ -0,0 +1,47 @@ +package sarama + +import ( + "testing" +) + +var ( + produceRequestEmpty = []byte{ + 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00} + + produceRequestHeader = []byte{ + 0x01, 0x23, + 0x00, 0x00, 0x04, 0x44, + 0x00, 0x00, 0x00, 0x00} + + produceRequestOneMessage = []byte{ + 0x01, 0x23, + 0x00, 0x00, 0x04, 0x44, + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x05, 't', 'o', 'p', 'i', 'c', + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, 0x00, 0xAD, + 0x00, 0x00, 0x00, 0x1C, + // messageSet + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x10, + // message + 0x23, 0x96, 0x4a, 0xf7, // CRC + 0x00, + 0x00, + 0xFF, 0xFF, 0xFF, 0xFF, + 0x00, 0x00, 0x00, 0x02, 0x00, 0xEE} +) + +func TestProduceRequest(t *testing.T) { + request := new(ProduceRequest) + testRequest(t, "empty", request, produceRequestEmpty) + + request.RequiredAcks = 0x123 + request.Timeout = 0x444 + testRequest(t, "header", request, produceRequestHeader) + + request.AddMessage("topic", 0xAD, &Message{Codec: CompressionNone, Key: nil, Value: []byte{0x00, 0xEE}}) + testRequest(t, "one message", request, produceRequestOneMessage) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/produce_response.go b/Godeps/_workspace/src/github.com/Shopify/sarama/produce_response.go new file mode 100644 index 0000000000000..1f49a85600fea --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/produce_response.go @@ -0,0 +1,112 @@ +package sarama + +type ProduceResponseBlock struct { + Err KError + Offset int64 +} + +func (pr *ProduceResponseBlock) decode(pd packetDecoder) (err error) { + tmp, err := pd.getInt16() + if err != nil { + return err + } + pr.Err = KError(tmp) + + pr.Offset, err = pd.getInt64() + if err != nil { + return err + } + + return nil +} + +type ProduceResponse struct { + Blocks map[string]map[int32]*ProduceResponseBlock +} + +func (pr *ProduceResponse) decode(pd packetDecoder) (err error) { + numTopics, err := pd.getArrayLength() + if err != nil { + return err + } + + pr.Blocks = make(map[string]map[int32]*ProduceResponseBlock, numTopics) + for i := 0; i < numTopics; i++ { + name, err := pd.getString() + if err != nil { + return err + } + + numBlocks, err := pd.getArrayLength() + if err != nil { + return err + } + + pr.Blocks[name] = make(map[int32]*ProduceResponseBlock, numBlocks) + + for j := 0; j < numBlocks; j++ { + id, err := pd.getInt32() + if err != nil { + return err + } + + block := new(ProduceResponseBlock) + err = block.decode(pd) + if err != nil { + return err + } + pr.Blocks[name][id] = block + } + } + + return nil +} + +func (pr *ProduceResponse) encode(pe packetEncoder) error { + err := pe.putArrayLength(len(pr.Blocks)) + if err != nil { + return err + } + for topic, partitions := range pr.Blocks { + err = pe.putString(topic) + if err != nil { + return err + } + err = pe.putArrayLength(len(partitions)) + if err != nil { + return err + } + for id, prb := range partitions { + pe.putInt32(id) + pe.putInt16(int16(prb.Err)) + pe.putInt64(prb.Offset) + } + } + return nil +} + +func (pr *ProduceResponse) GetBlock(topic string, partition int32) *ProduceResponseBlock { + if pr.Blocks == nil { + return nil + } + + if pr.Blocks[topic] == nil { + return nil + } + + return pr.Blocks[topic][partition] +} + +// Testing API + +func (pr *ProduceResponse) AddTopicPartition(topic string, partition int32, err KError) { + if pr.Blocks == nil { + pr.Blocks = make(map[string]map[int32]*ProduceResponseBlock) + } + byTopic, ok := pr.Blocks[topic] + if !ok { + byTopic = make(map[int32]*ProduceResponseBlock) + pr.Blocks[topic] = byTopic + } + byTopic[partition] = &ProduceResponseBlock{Err: err} +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/produce_response_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/produce_response_test.go new file mode 100644 index 0000000000000..5c3131af4a3a3 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/produce_response_test.go @@ -0,0 +1,67 @@ +package sarama + +import "testing" + +var ( + produceResponseNoBlocks = []byte{ + 0x00, 0x00, 0x00, 0x00} + + produceResponseManyBlocks = []byte{ + 0x00, 0x00, 0x00, 0x02, + + 0x00, 0x03, 'f', 'o', 'o', + 0x00, 0x00, 0x00, 0x00, + + 0x00, 0x03, 'b', 'a', 'r', + 0x00, 0x00, 0x00, 0x02, + + 0x00, 0x00, 0x00, 0x01, + 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, + + 0x00, 0x00, 0x00, 0x02, + 0x00, 0x02, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00} +) + +func TestProduceResponse(t *testing.T) { + response := ProduceResponse{} + + testDecodable(t, "no blocks", &response, produceResponseNoBlocks) + if len(response.Blocks) != 0 { + t.Error("Decoding produced", len(response.Blocks), "topics where there were none") + } + + testDecodable(t, "many blocks", &response, produceResponseManyBlocks) + if len(response.Blocks) != 2 { + t.Error("Decoding produced", len(response.Blocks), "topics where there were 2") + } + if len(response.Blocks["foo"]) != 0 { + t.Error("Decoding produced", len(response.Blocks["foo"]), "partitions for 'foo' where there were none") + } + if len(response.Blocks["bar"]) != 2 { + t.Error("Decoding produced", len(response.Blocks["bar"]), "partitions for 'bar' where there were two") + } + block := response.GetBlock("bar", 1) + if block == nil { + t.Error("Decoding did not produce a block for bar/1") + } else { + if block.Err != ErrNoError { + t.Error("Decoding failed for bar/1/Err, got:", int16(block.Err)) + } + if block.Offset != 0xFF { + t.Error("Decoding failed for bar/1/Offset, got:", block.Offset) + } + } + block = response.GetBlock("bar", 2) + if block == nil { + t.Error("Decoding did not produce a block for bar/2") + } else { + if block.Err != ErrInvalidMessage { + t.Error("Decoding failed for bar/2/Err, got:", int16(block.Err)) + } + if block.Offset != 0 { + t.Error("Decoding failed for bar/2/Offset, got:", block.Offset) + } + } +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/real_decoder.go b/Godeps/_workspace/src/github.com/Shopify/sarama/real_decoder.go new file mode 100644 index 0000000000000..b194b9bcca7e9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/real_decoder.go @@ -0,0 +1,225 @@ +package sarama + +import ( + "encoding/binary" + "math" +) + +type realDecoder struct { + raw []byte + off int + stack []pushDecoder +} + +// primitives + +func (rd *realDecoder) getInt8() (int8, error) { + if rd.remaining() < 1 { + rd.off = len(rd.raw) + return -1, ErrInsufficientData + } + tmp := int8(rd.raw[rd.off]) + rd.off += 1 + return tmp, nil +} + +func (rd *realDecoder) getInt16() (int16, error) { + if rd.remaining() < 2 { + rd.off = len(rd.raw) + return -1, ErrInsufficientData + } + tmp := int16(binary.BigEndian.Uint16(rd.raw[rd.off:])) + rd.off += 2 + return tmp, nil +} + +func (rd *realDecoder) getInt32() (int32, error) { + if rd.remaining() < 4 { + rd.off = len(rd.raw) + return -1, ErrInsufficientData + } + tmp := int32(binary.BigEndian.Uint32(rd.raw[rd.off:])) + rd.off += 4 + return tmp, nil +} + +func (rd *realDecoder) getInt64() (int64, error) { + if rd.remaining() < 8 { + rd.off = len(rd.raw) + return -1, ErrInsufficientData + } + tmp := int64(binary.BigEndian.Uint64(rd.raw[rd.off:])) + rd.off += 8 + return tmp, nil +} + +func (rd *realDecoder) getArrayLength() (int, error) { + if rd.remaining() < 4 { + rd.off = len(rd.raw) + return -1, ErrInsufficientData + } + tmp := int(binary.BigEndian.Uint32(rd.raw[rd.off:])) + rd.off += 4 + if tmp > rd.remaining() { + rd.off = len(rd.raw) + return -1, ErrInsufficientData + } else if tmp > 2*math.MaxUint16 { + return -1, PacketDecodingError{"invalid array length"} + } + return tmp, nil +} + +// collections + +func (rd *realDecoder) getBytes() ([]byte, error) { + tmp, err := rd.getInt32() + + if err != nil { + return nil, err + } + + n := int(tmp) + + switch { + case n < -1: + return nil, PacketDecodingError{"invalid byteslice length"} + case n == -1: + return nil, nil + case n == 0: + return make([]byte, 0), nil + case n > rd.remaining(): + rd.off = len(rd.raw) + return nil, ErrInsufficientData + } + + tmpStr := rd.raw[rd.off : rd.off+n] + rd.off += n + return tmpStr, nil +} + +func (rd *realDecoder) getString() (string, error) { + tmp, err := rd.getInt16() + + if err != nil { + return "", err + } + + n := int(tmp) + + switch { + case n < -1: + return "", PacketDecodingError{"invalid string length"} + case n == -1: + return "", nil + case n == 0: + return "", nil + case n > rd.remaining(): + rd.off = len(rd.raw) + return "", ErrInsufficientData + } + + tmpStr := string(rd.raw[rd.off : rd.off+n]) + rd.off += n + return tmpStr, nil +} + +func (rd *realDecoder) getInt32Array() ([]int32, error) { + if rd.remaining() < 4 { + rd.off = len(rd.raw) + return nil, ErrInsufficientData + } + n := int(binary.BigEndian.Uint32(rd.raw[rd.off:])) + rd.off += 4 + + if rd.remaining() < 4*n { + rd.off = len(rd.raw) + return nil, ErrInsufficientData + } + + if n == 0 { + return nil, nil + } + + if n < 0 { + return nil, PacketDecodingError{"invalid array length"} + } + + ret := make([]int32, n) + for i := range ret { + ret[i] = int32(binary.BigEndian.Uint32(rd.raw[rd.off:])) + rd.off += 4 + } + return ret, nil +} + +func (rd *realDecoder) getInt64Array() ([]int64, error) { + if rd.remaining() < 4 { + rd.off = len(rd.raw) + return nil, ErrInsufficientData + } + n := int(binary.BigEndian.Uint32(rd.raw[rd.off:])) + rd.off += 4 + + if rd.remaining() < 8*n { + rd.off = len(rd.raw) + return nil, ErrInsufficientData + } + + if n == 0 { + return nil, nil + } + + if n < 0 { + return nil, PacketDecodingError{"invalid array length"} + } + + ret := make([]int64, n) + for i := range ret { + ret[i] = int64(binary.BigEndian.Uint64(rd.raw[rd.off:])) + rd.off += 8 + } + return ret, nil +} + +// subsets + +func (rd *realDecoder) remaining() int { + return len(rd.raw) - rd.off +} + +func (rd *realDecoder) getSubset(length int) (packetDecoder, error) { + if length > rd.remaining() { + rd.off = len(rd.raw) + return nil, ErrInsufficientData + } + + start := rd.off + rd.off += length + return &realDecoder{raw: rd.raw[start:rd.off]}, nil +} + +// stacks + +func (rd *realDecoder) push(in pushDecoder) error { + in.saveOffset(rd.off) + + reserve := in.reserveLength() + if rd.remaining() < reserve { + rd.off = len(rd.raw) + return ErrInsufficientData + } + + rd.stack = append(rd.stack, in) + + rd.off += reserve + + return nil +} + +func (rd *realDecoder) pop() error { + // this is go's ugly pop pattern (the inverse of append) + in := rd.stack[len(rd.stack)-1] + rd.stack = rd.stack[:len(rd.stack)-1] + + return in.check(rd.off, rd.raw) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/real_encoder.go b/Godeps/_workspace/src/github.com/Shopify/sarama/real_encoder.go new file mode 100644 index 0000000000000..947ce98d90f81 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/real_encoder.go @@ -0,0 +1,100 @@ +package sarama + +import "encoding/binary" + +type realEncoder struct { + raw []byte + off int + stack []pushEncoder +} + +// primitives + +func (re *realEncoder) putInt8(in int8) { + re.raw[re.off] = byte(in) + re.off += 1 +} + +func (re *realEncoder) putInt16(in int16) { + binary.BigEndian.PutUint16(re.raw[re.off:], uint16(in)) + re.off += 2 +} + +func (re *realEncoder) putInt32(in int32) { + binary.BigEndian.PutUint32(re.raw[re.off:], uint32(in)) + re.off += 4 +} + +func (re *realEncoder) putInt64(in int64) { + binary.BigEndian.PutUint64(re.raw[re.off:], uint64(in)) + re.off += 8 +} + +func (re *realEncoder) putArrayLength(in int) error { + re.putInt32(int32(in)) + return nil +} + +// collection + +func (re *realEncoder) putRawBytes(in []byte) error { + copy(re.raw[re.off:], in) + re.off += len(in) + return nil +} + +func (re *realEncoder) putBytes(in []byte) error { + if in == nil { + re.putInt32(-1) + return nil + } + re.putInt32(int32(len(in))) + copy(re.raw[re.off:], in) + re.off += len(in) + return nil +} + +func (re *realEncoder) putString(in string) error { + re.putInt16(int16(len(in))) + copy(re.raw[re.off:], in) + re.off += len(in) + return nil +} + +func (re *realEncoder) putInt32Array(in []int32) error { + err := re.putArrayLength(len(in)) + if err != nil { + return err + } + for _, val := range in { + re.putInt32(val) + } + return nil +} + +func (re *realEncoder) putInt64Array(in []int64) error { + err := re.putArrayLength(len(in)) + if err != nil { + return err + } + for _, val := range in { + re.putInt64(val) + } + return nil +} + +// stacks + +func (re *realEncoder) push(in pushEncoder) { + in.saveOffset(re.off) + re.off += in.reserveLength() + re.stack = append(re.stack, in) +} + +func (re *realEncoder) pop() error { + // this is go's ugly pop pattern (the inverse of append) + in := re.stack[len(re.stack)-1] + re.stack = re.stack[:len(re.stack)-1] + + return in.run(re.off, re.raw) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/request.go b/Godeps/_workspace/src/github.com/Shopify/sarama/request.go new file mode 100644 index 0000000000000..d6d5cdfcd96df --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/request.go @@ -0,0 +1,100 @@ +package sarama + +import ( + "encoding/binary" + "fmt" + "io" +) + +type requestBody interface { + encoder + decoder + key() int16 + version() int16 +} + +type request struct { + correlationID int32 + clientID string + body requestBody +} + +func (r *request) encode(pe packetEncoder) (err error) { + pe.push(&lengthField{}) + pe.putInt16(r.body.key()) + pe.putInt16(r.body.version()) + pe.putInt32(r.correlationID) + err = pe.putString(r.clientID) + if err != nil { + return err + } + err = r.body.encode(pe) + if err != nil { + return err + } + return pe.pop() +} + +func (r *request) decode(pd packetDecoder) (err error) { + var key int16 + if key, err = pd.getInt16(); err != nil { + return err + } + var version int16 + if version, err = pd.getInt16(); err != nil { + return err + } + if r.correlationID, err = pd.getInt32(); err != nil { + return err + } + r.clientID, err = pd.getString() + + r.body = allocateBody(key, version) + if r.body == nil { + return PacketDecodingError{fmt.Sprintf("unknown request key (%d)", key)} + } + return r.body.decode(pd) +} + +func decodeRequest(r io.Reader) (req *request, err error) { + lengthBytes := make([]byte, 4) + if _, err := io.ReadFull(r, lengthBytes); err != nil { + return nil, err + } + + length := int32(binary.BigEndian.Uint32(lengthBytes)) + if length <= 4 || length > MaxRequestSize { + return nil, PacketDecodingError{fmt.Sprintf("message of length %d too large or too small", length)} + } + + encodedReq := make([]byte, length) + if _, err := io.ReadFull(r, encodedReq); err != nil { + return nil, err + } + + req = &request{} + if err := decode(encodedReq, req); err != nil { + return nil, err + } + return req, nil +} + +func allocateBody(key, version int16) requestBody { + switch key { + case 0: + return &ProduceRequest{} + case 1: + return &FetchRequest{} + case 2: + return &OffsetRequest{} + case 3: + return &MetadataRequest{} + case 8: + return &OffsetCommitRequest{Version: version} + case 9: + return &OffsetFetchRequest{} + case 10: + return &ConsumerMetadataRequest{} + } + return nil +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/request_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/request_test.go new file mode 100644 index 0000000000000..69e8b4cbebb80 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/request_test.go @@ -0,0 +1,80 @@ +package sarama + +import ( + "bytes" + "reflect" + "testing" +) + +type testRequestBody struct { +} + +func (s *testRequestBody) key() int16 { + return 0x666 +} + +func (s *testRequestBody) version() int16 { + return 0xD2 +} + +func (s *testRequestBody) encode(pe packetEncoder) error { + return pe.putString("abc") +} + +// not specific to request tests, just helper functions for testing structures that +// implement the encoder or decoder interfaces that needed somewhere to live + +func testEncodable(t *testing.T, name string, in encoder, expect []byte) { + packet, err := encode(in) + if err != nil { + t.Error(err) + } else if !bytes.Equal(packet, expect) { + t.Error("Encoding", name, "failed\ngot ", packet, "\nwant", expect) + } +} + +func testDecodable(t *testing.T, name string, out decoder, in []byte) { + err := decode(in, out) + if err != nil { + t.Error("Decoding", name, "failed:", err) + } +} + +func testRequest(t *testing.T, name string, rb requestBody, expected []byte) { + // Encoder request + req := &request{correlationID: 123, clientID: "foo", body: rb} + packet, err := encode(req) + headerSize := 14 + len("foo") + if err != nil { + t.Error(err) + } else if !bytes.Equal(packet[headerSize:], expected) { + t.Error("Encoding", name, "failed\ngot ", packet, "\nwant", expected) + } + // Decoder request + decoded, err := decodeRequest(bytes.NewReader(packet)) + if err != nil { + t.Error("Failed to decode request", err) + } else if decoded.correlationID != 123 || decoded.clientID != "foo" { + t.Errorf("Decoded header is not valid: %v", decoded) + } else if !reflect.DeepEqual(rb, decoded.body) { + t.Errorf("Decoded request does not match the encoded one\nencoded: %v\ndecoded: %v", rb, decoded) + } +} + +func testResponse(t *testing.T, name string, res encoder, expected []byte) { + encoded, err := encode(res) + if err != nil { + t.Error(err) + } else if expected != nil && !bytes.Equal(encoded, expected) { + t.Error("Encoding", name, "failed\ngot ", encoded, "\nwant", expected) + } + + decoded := reflect.New(reflect.TypeOf(res).Elem()).Interface().(decoder) + if err := decode(encoded, decoded); err != nil { + t.Error("Decoding", name, "failed:", err) + } + + if !reflect.DeepEqual(decoded, res) { + t.Errorf("Decoded response does not match the encoded one\nencoded: %#v\ndecoded: %#v", res, decoded) + } +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/response_header.go b/Godeps/_workspace/src/github.com/Shopify/sarama/response_header.go new file mode 100644 index 0000000000000..f3f4d27d6c428 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/response_header.go @@ -0,0 +1,21 @@ +package sarama + +import "fmt" + +type responseHeader struct { + length int32 + correlationID int32 +} + +func (r *responseHeader) decode(pd packetDecoder) (err error) { + r.length, err = pd.getInt32() + if err != nil { + return err + } + if r.length <= 4 || r.length > MaxResponseSize { + return PacketDecodingError{fmt.Sprintf("message of length %d too large or too small", r.length)} + } + + r.correlationID, err = pd.getInt32() + return err +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/response_header_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/response_header_test.go new file mode 100644 index 0000000000000..8f9fdb80c5b9a --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/response_header_test.go @@ -0,0 +1,21 @@ +package sarama + +import "testing" + +var ( + responseHeaderBytes = []byte{ + 0x00, 0x00, 0x0f, 0x00, + 0x0a, 0xbb, 0xcc, 0xff} +) + +func TestResponseHeader(t *testing.T) { + header := responseHeader{} + + testDecodable(t, "response header", &header, responseHeaderBytes) + if header.length != 0xf00 { + t.Error("Decoding header length failed, got", header.length) + } + if header.correlationID != 0x0abbccff { + t.Error("Decoding header correlation id failed, got", header.correlationID) + } +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/sarama.go b/Godeps/_workspace/src/github.com/Shopify/sarama/sarama.go new file mode 100644 index 0000000000000..d5982175030fa --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/sarama.go @@ -0,0 +1,47 @@ +/* +Package sarama provides client libraries for the Kafka 0.8 protocol. The AsyncProducer object is the high-level +API for producing messages asynchronously; the SyncProducer provides a blocking API for the same purpose. +The Consumer object is the high-level API for consuming messages. The Client object provides metadata +management functionality that is shared between the higher-level objects. + +For lower-level needs, the Broker and Request/Response objects permit precise control over each connection +and message sent on the wire. + +The Request/Response objects and properties are mostly undocumented, as they line up exactly with the +protocol fields documented by Kafka at https://cwiki.apache.org/confluence/display/KAFKA/A+Guide+To+The+Kafka+Protocol +*/ +package sarama + +import ( + "io/ioutil" + "log" +) + +// Logger is the instance of a StdLogger interface that Sarama writes connection +// management events to. By default it is set to discard all log messages via ioutil.Discard, +// but you can set it to redirect wherever you want. +var Logger StdLogger = log.New(ioutil.Discard, "[Sarama] ", log.LstdFlags) + +// StdLogger is used to log error messages. +type StdLogger interface { + Print(v ...interface{}) + Printf(format string, v ...interface{}) + Println(v ...interface{}) +} + +// PanicHandler is called for recovering from panics spawned internally to the library (and thus +// not recoverable by the caller's goroutine). Defaults to nil, which means panics are not recovered. +var PanicHandler func(interface{}) + +// MaxRequestSize is the maximum size (in bytes) of any request that Sarama will attempt to send. Trying +// to send a request larger than this will result in an PacketEncodingError. The default of 100 MiB is aligned +// with Kafka's default `socket.request.max.bytes`, which is the largest request the broker will attempt +// to process. +var MaxRequestSize int32 = 100 * 1024 * 1024 + +// MaxResponseSize is the maximum size (in bytes) of any response that Sarama will attempt to parse. If +// a broker returns a response message larger than this value, Sarama will return a PacketDecodingError to +// protect the client from running out of memory. Please note that brokers do not have any natural limit on +// the size of responses they send. In particular, they can send arbitrarily large fetch responses to consumers +// (see https://issues.apache.org/jira/browse/KAFKA-2063). +var MaxResponseSize int32 = 100 * 1024 * 1024 diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/snappy.go b/Godeps/_workspace/src/github.com/Shopify/sarama/snappy.go new file mode 100644 index 0000000000000..e86cb70397152 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/snappy.go @@ -0,0 +1,41 @@ +package sarama + +import ( + "bytes" + "encoding/binary" + + "github.com/golang/snappy" +) + +var snappyMagic = []byte{130, 83, 78, 65, 80, 80, 89, 0} + +// SnappyEncode encodes binary data +func snappyEncode(src []byte) []byte { + return snappy.Encode(nil, src) +} + +// SnappyDecode decodes snappy data +func snappyDecode(src []byte) ([]byte, error) { + if bytes.Equal(src[:8], snappyMagic) { + var ( + pos = uint32(16) + max = uint32(len(src)) + dst = make([]byte, 0, len(src)) + chunk []byte + err error + ) + for pos < max { + size := binary.BigEndian.Uint32(src[pos : pos+4]) + pos += 4 + + chunk, err = snappy.Decode(chunk, src[pos:pos+size]) + if err != nil { + return nil, err + } + pos += size + dst = append(dst, chunk...) + } + return dst, nil + } + return snappy.Decode(nil, src) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/snappy_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/snappy_test.go new file mode 100644 index 0000000000000..f3cf7ff5ca744 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/snappy_test.go @@ -0,0 +1,49 @@ +package sarama + +import ( + "bytes" + "testing" +) + +var snappyTestCases = map[string][]byte{ + "REPEATREPEATREPEATREPEATREPEATREPEAT": []byte{36, 20, 82, 69, 80, 69, 65, 84, 118, 6, 0}, + "REALLY SHORT": []byte{12, 44, 82, 69, 65, 76, 76, 89, 32, 83, 72, 79, 82, 84}, + "AXBXCXDXEXFX": []byte{12, 44, 65, 88, 66, 88, 67, 88, 68, 88, 69, 88, 70, 88}, +} + +var snappyStreamTestCases = map[string][]byte{ + "PLAINDATA": []byte{130, 83, 78, 65, 80, 80, 89, 0, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 11, 9, 32, 80, 76, 65, 73, 78, 68, 65, 84, 65}, + `{"a":"UtaitILHMDAAAAfU","b":"日本"}`: []byte{130, 83, 78, 65, 80, 80, 89, 0, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 39, 37, 144, 123, 34, 97, 34, 58, 34, 85, 116, 97, 105, 116, 73, 76, 72, 77, 68, 65, 65, 65, 65, 102, 85, 34, 44, 34, 98, 34, 58, 34, 230, 151, 165, 230, 156, 172, 34, 125}, + `Sed ut perspiciatis unde omnis iste natus error sit voluptatem accusantium doloremque laudantium, totam rem aperiam, eaque ipsa quae ab illo inventore veritatis et quasi architecto beatae vitae dicta sunt explicabo. Nemo enim ipsam voluptatem quia voluptas sit aspernatur aut odit aut fugit, sed quia consequuntur magni dolores eos qui ratione voluptatem sequi nesciunt. Neque porro quisquam est, qui dolorem ipsum quia dolor sit amet, consectetur, adipisci velit, sed quia non numquam eius modi tempora incidunt ut labore et dolore magnam aliquam quaerat voluptatem. Ut enim ad minima veniam, quis nostrum exercitationem ullam corporis suscipit laboriosam, nisi ut aliquid ex ea commodi consequatur? Quis autem vel eum iure reprehenderit qui in ea voluptate velit esse quam nihil molestiae consequatur, vel illum qui dolorem eum fugiat quo voluptas nulla pariatur? At vero eos et accusamus et iusto odio dignissimos ducimus qui blanditiis praesentium voluptatum deleniti atque corrupti quos dolores et quas molestias except`: []byte{130, 83, 78, 65, 80, 80, 89, 0, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 3, 89, 128, 8, 240, 90, 83, 101, 100, 32, 117, 116, 32, 112, 101, 114, 115, 112, 105, 99, 105, 97, 116, 105, 115, 32, 117, 110, 100, 101, 32, 111, 109, 110, 105, 115, 32, 105, 115, 116, 101, 32, 110, 97, 116, 117, 115, 32, 101, 114, 114, 111, 114, 32, 115, 105, 116, 32, 118, 111, 108, 117, 112, 116, 97, 116, 101, 109, 32, 97, 99, 99, 117, 115, 97, 110, 116, 105, 117, 109, 32, 100, 111, 108, 111, 114, 101, 109, 113, 117, 101, 32, 108, 97, 117, 100, 97, 5, 22, 240, 60, 44, 32, 116, 111, 116, 97, 109, 32, 114, 101, 109, 32, 97, 112, 101, 114, 105, 97, 109, 44, 32, 101, 97, 113, 117, 101, 32, 105, 112, 115, 97, 32, 113, 117, 97, 101, 32, 97, 98, 32, 105, 108, 108, 111, 32, 105, 110, 118, 101, 110, 116, 111, 114, 101, 32, 118, 101, 114, 105, 116, 97, 1, 141, 4, 101, 116, 1, 36, 88, 115, 105, 32, 97, 114, 99, 104, 105, 116, 101, 99, 116, 111, 32, 98, 101, 97, 116, 97, 101, 32, 118, 105, 1, 6, 120, 100, 105, 99, 116, 97, 32, 115, 117, 110, 116, 32, 101, 120, 112, 108, 105, 99, 97, 98, 111, 46, 32, 78, 101, 109, 111, 32, 101, 110, 105, 109, 5, 103, 0, 109, 46, 180, 0, 12, 113, 117, 105, 97, 17, 16, 0, 115, 5, 209, 72, 97, 115, 112, 101, 114, 110, 97, 116, 117, 114, 32, 97, 117, 116, 32, 111, 100, 105, 116, 5, 9, 36, 102, 117, 103, 105, 116, 44, 32, 115, 101, 100, 9, 53, 32, 99, 111, 110, 115, 101, 113, 117, 117, 110, 1, 42, 20, 109, 97, 103, 110, 105, 32, 9, 245, 16, 115, 32, 101, 111, 115, 1, 36, 28, 32, 114, 97, 116, 105, 111, 110, 101, 17, 96, 33, 36, 1, 51, 36, 105, 32, 110, 101, 115, 99, 105, 117, 110, 116, 1, 155, 1, 254, 16, 112, 111, 114, 114, 111, 1, 51, 36, 115, 113, 117, 97, 109, 32, 101, 115, 116, 44, 1, 14, 13, 81, 5, 183, 4, 117, 109, 1, 18, 0, 97, 9, 19, 4, 32, 115, 1, 149, 12, 109, 101, 116, 44, 9, 135, 76, 99, 116, 101, 116, 117, 114, 44, 32, 97, 100, 105, 112, 105, 115, 99, 105, 32, 118, 101, 108, 50, 173, 0, 24, 110, 111, 110, 32, 110, 117, 109, 9, 94, 84, 105, 117, 115, 32, 109, 111, 100, 105, 32, 116, 101, 109, 112, 111, 114, 97, 32, 105, 110, 99, 105, 100, 33, 52, 20, 117, 116, 32, 108, 97, 98, 33, 116, 4, 101, 116, 9, 106, 0, 101, 5, 219, 20, 97, 109, 32, 97, 108, 105, 5, 62, 33, 164, 8, 114, 97, 116, 29, 212, 12, 46, 32, 85, 116, 41, 94, 52, 97, 100, 32, 109, 105, 110, 105, 109, 97, 32, 118, 101, 110, 105, 33, 221, 72, 113, 117, 105, 115, 32, 110, 111, 115, 116, 114, 117, 109, 32, 101, 120, 101, 114, 99, 105, 33, 202, 104, 111, 110, 101, 109, 32, 117, 108, 108, 97, 109, 32, 99, 111, 114, 112, 111, 114, 105, 115, 32, 115, 117, 115, 99, 105, 112, 105, 13, 130, 8, 105, 111, 115, 1, 64, 12, 110, 105, 115, 105, 1, 150, 5, 126, 44, 105, 100, 32, 101, 120, 32, 101, 97, 32, 99, 111, 109, 5, 192, 0, 99, 41, 131, 33, 172, 8, 63, 32, 81, 1, 107, 4, 97, 117, 33, 101, 96, 118, 101, 108, 32, 101, 117, 109, 32, 105, 117, 114, 101, 32, 114, 101, 112, 114, 101, 104, 101, 110, 100, 101, 114, 105, 65, 63, 12, 105, 32, 105, 110, 1, 69, 16, 118, 111, 108, 117, 112, 65, 185, 1, 47, 24, 105, 116, 32, 101, 115, 115, 101, 1, 222, 64, 109, 32, 110, 105, 104, 105, 108, 32, 109, 111, 108, 101, 115, 116, 105, 97, 101, 46, 103, 0, 0, 44, 1, 45, 16, 32, 105, 108, 108, 117, 37, 143, 45, 36, 0, 109, 5, 110, 65, 33, 20, 97, 116, 32, 113, 117, 111, 17, 92, 44, 115, 32, 110, 117, 108, 108, 97, 32, 112, 97, 114, 105, 9, 165, 24, 65, 116, 32, 118, 101, 114, 111, 69, 34, 44, 101, 116, 32, 97, 99, 99, 117, 115, 97, 109, 117, 115, 1, 13, 104, 105, 117, 115, 116, 111, 32, 111, 100, 105, 111, 32, 100, 105, 103, 110, 105, 115, 115, 105, 109, 111, 115, 32, 100, 117, 99, 105, 1, 34, 80, 113, 117, 105, 32, 98, 108, 97, 110, 100, 105, 116, 105, 105, 115, 32, 112, 114, 97, 101, 115, 101, 101, 87, 17, 111, 56, 116, 117, 109, 32, 100, 101, 108, 101, 110, 105, 116, 105, 32, 97, 116, 65, 89, 28, 99, 111, 114, 114, 117, 112, 116, 105, 1, 150, 0, 115, 13, 174, 5, 109, 8, 113, 117, 97, 65, 5, 52, 108, 101, 115, 116, 105, 97, 115, 32, 101, 120, 99, 101, 112, 116, 0, 0, 0, 1, 0}, +} + +func TestSnappyEncode(t *testing.T) { + for src, exp := range snappyTestCases { + dst := snappyEncode([]byte(src)) + if !bytes.Equal(dst, exp) { + t.Errorf("Expected %s to generate %v, but was %v", src, exp, dst) + } + } +} + +func TestSnappyDecode(t *testing.T) { + for exp, src := range snappyTestCases { + dst, err := snappyDecode(src) + if err != nil { + t.Error("Encoding error: ", err) + } else if !bytes.Equal(dst, []byte(exp)) { + t.Errorf("Expected %s to be generated from %v, but was %s", exp, src, string(dst)) + } + } +} + +func TestSnappyDecodeStreams(t *testing.T) { + for exp, src := range snappyStreamTestCases { + dst, err := snappyDecode(src) + if err != nil { + t.Error("Encoding error: ", err) + } else if !bytes.Equal(dst, []byte(exp)) { + t.Errorf("Expected %s to be generated from [%d]byte, but was %s", exp, len(src), string(dst)) + } + } +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/sync_producer.go b/Godeps/_workspace/src/github.com/Shopify/sarama/sync_producer.go new file mode 100644 index 0000000000000..b59d74a201479 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/sync_producer.go @@ -0,0 +1,94 @@ +package sarama + +import "sync" + +// SyncProducer publishes Kafka messages. It routes messages to the correct broker, refreshing metadata as appropriate, +// and parses responses for errors. You must call Close() on a producer to avoid leaks, it may not be garbage-collected automatically when +// it passes out of scope. +type SyncProducer interface { + + // SendMessage produces a given message, and returns only when it either has succeeded or failed to produce. + // It will return the partition and the offset of the produced message, or an error if the message + // failed to produce. + SendMessage(msg *ProducerMessage) (partition int32, offset int64, err error) + + // Close shuts down the producer and flushes any messages it may have buffered. You must call this function before + // a producer object passes out of scope, as it may otherwise leak memory. You must call this before calling Close + // on the underlying client. + Close() error +} + +type syncProducer struct { + producer *asyncProducer + wg sync.WaitGroup +} + +// NewSyncProducer creates a new SyncProducer using the given broker addresses and configuration. +func NewSyncProducer(addrs []string, config *Config) (SyncProducer, error) { + p, err := NewAsyncProducer(addrs, config) + if err != nil { + return nil, err + } + return newSyncProducerFromAsyncProducer(p.(*asyncProducer)), nil +} + +// NewSyncProducerFromClient creates a new SyncProducer using the given client. It is still +// necessary to call Close() on the underlying client when shutting down this producer. +func NewSyncProducerFromClient(client Client) (SyncProducer, error) { + p, err := NewAsyncProducerFromClient(client) + if err != nil { + return nil, err + } + return newSyncProducerFromAsyncProducer(p.(*asyncProducer)), nil +} + +func newSyncProducerFromAsyncProducer(p *asyncProducer) *syncProducer { + p.conf.Producer.Return.Successes = true + p.conf.Producer.Return.Errors = true + sp := &syncProducer{producer: p} + + sp.wg.Add(2) + go withRecover(sp.handleSuccesses) + go withRecover(sp.handleErrors) + + return sp +} + +func (sp *syncProducer) SendMessage(msg *ProducerMessage) (partition int32, offset int64, err error) { + oldMetadata := msg.Metadata + defer func() { + msg.Metadata = oldMetadata + }() + + expectation := make(chan error, 1) + msg.Metadata = expectation + sp.producer.Input() <- msg + + if err := <-expectation; err != nil { + return -1, -1, err + } else { + return msg.Partition, msg.Offset, nil + } +} + +func (sp *syncProducer) handleSuccesses() { + defer sp.wg.Done() + for msg := range sp.producer.Successes() { + expectation := msg.Metadata.(chan error) + expectation <- nil + } +} + +func (sp *syncProducer) handleErrors() { + defer sp.wg.Done() + for err := range sp.producer.Errors() { + expectation := err.Msg.Metadata.(chan error) + expectation <- err.Err + } +} + +func (sp *syncProducer) Close() error { + sp.producer.AsyncClose() + sp.wg.Wait() + return nil +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/sync_producer_test.go b/Godeps/_workspace/src/github.com/Shopify/sarama/sync_producer_test.go new file mode 100644 index 0000000000000..d378949b18d34 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/sync_producer_test.go @@ -0,0 +1,149 @@ +package sarama + +import ( + "log" + "sync" + "testing" +) + +func TestSyncProducer(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader := newMockBroker(t, 2) + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker(leader.Addr(), leader.BrokerID()) + metadataResponse.AddTopicPartition("my_topic", 0, leader.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataResponse) + + prodSuccess := new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", 0, ErrNoError) + for i := 0; i < 10; i++ { + leader.Returns(prodSuccess) + } + + producer, err := NewSyncProducer([]string{seedBroker.Addr()}, nil) + if err != nil { + t.Fatal(err) + } + + for i := 0; i < 10; i++ { + msg := &ProducerMessage{ + Topic: "my_topic", + Value: StringEncoder(TestMessage), + Metadata: "test", + } + + partition, offset, err := producer.SendMessage(msg) + + if partition != 0 || msg.Partition != partition { + t.Error("Unexpected partition") + } + if offset != 0 || msg.Offset != offset { + t.Error("Unexpected offset") + } + if str, ok := msg.Metadata.(string); !ok || str != "test" { + t.Error("Unexpected metadata") + } + if err != nil { + t.Error(err) + } + } + + safeClose(t, producer) + leader.Close() + seedBroker.Close() +} + +func TestConcurrentSyncProducer(t *testing.T) { + seedBroker := newMockBroker(t, 1) + leader := newMockBroker(t, 2) + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker(leader.Addr(), leader.BrokerID()) + metadataResponse.AddTopicPartition("my_topic", 0, leader.BrokerID(), nil, nil, ErrNoError) + seedBroker.Returns(metadataResponse) + + prodSuccess := new(ProduceResponse) + prodSuccess.AddTopicPartition("my_topic", 0, ErrNoError) + leader.Returns(prodSuccess) + + config := NewConfig() + config.Producer.Flush.Messages = 100 + producer, err := NewSyncProducer([]string{seedBroker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + wg := sync.WaitGroup{} + + for i := 0; i < 100; i++ { + wg.Add(1) + go func() { + msg := &ProducerMessage{Topic: "my_topic", Value: StringEncoder(TestMessage)} + partition, _, err := producer.SendMessage(msg) + if partition != 0 { + t.Error("Unexpected partition") + } + if err != nil { + t.Error(err) + } + wg.Done() + }() + } + wg.Wait() + + safeClose(t, producer) + leader.Close() + seedBroker.Close() +} + +func TestSyncProducerToNonExistingTopic(t *testing.T) { + broker := newMockBroker(t, 1) + + metadataResponse := new(MetadataResponse) + metadataResponse.AddBroker(broker.Addr(), broker.BrokerID()) + metadataResponse.AddTopicPartition("my_topic", 0, broker.BrokerID(), nil, nil, ErrNoError) + broker.Returns(metadataResponse) + + config := NewConfig() + config.Metadata.Retry.Max = 0 + config.Producer.Retry.Max = 0 + + producer, err := NewSyncProducer([]string{broker.Addr()}, config) + if err != nil { + t.Fatal(err) + } + + metadataResponse = new(MetadataResponse) + metadataResponse.AddTopic("unknown", ErrUnknownTopicOrPartition) + broker.Returns(metadataResponse) + + _, _, err = producer.SendMessage(&ProducerMessage{Topic: "unknown"}) + if err != ErrUnknownTopicOrPartition { + t.Error("Uxpected ErrUnknownTopicOrPartition, found:", err) + } + + safeClose(t, producer) + broker.Close() +} + +// This example shows the basic usage pattern of the SyncProducer. +func ExampleSyncProducer() { + producer, err := NewSyncProducer([]string{"localhost:9092"}, nil) + if err != nil { + log.Fatalln(err) + } + defer func() { + if err := producer.Close(); err != nil { + log.Fatalln(err) + } + }() + + msg := &ProducerMessage{Topic: "my_topic", Value: StringEncoder("testing 123")} + partition, offset, err := producer.SendMessage(msg) + if err != nil { + log.Printf("FAILED to send message: %s\n", err) + } else { + log.Printf("> message sent to partition %d at offset %d\n", partition, offset) + } +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/tools/README.md b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/README.md new file mode 100644 index 0000000000000..3464c4ad8040c --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/README.md @@ -0,0 +1,10 @@ +# Sarama tools + +This folder contains applications that are useful for exploration of your Kafka cluster, or instrumentation. +Some of these tools mirror tools that ship with Kafka, but these tools won't require installing the JVM to function. + +- [kafka-console-producer](./kafka-console-producer): a command line tool to produce a single message to your Kafka custer. +- [kafka-console-partitionconsumer](./kafka-console-partitionconsumer): (deprecated) a command line tool to consume a single partition of a topic on your Kafka cluster. +- [kafka-console-consumer](./kafka-console-consumer): a command line tool to consume arbitrary partitions of a topic on your Kafka cluster. + +To install all tools, run `go get github.com/Shopify/sarama/tools/...` diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-consumer/.gitignore b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-consumer/.gitignore new file mode 100644 index 0000000000000..67da9dfa9f298 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-consumer/.gitignore @@ -0,0 +1,2 @@ +kafka-console-consumer +kafka-console-consumer.test diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-consumer/README.md b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-consumer/README.md new file mode 100644 index 0000000000000..4e77f0b705085 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-consumer/README.md @@ -0,0 +1,29 @@ +# kafka-console-consumer + +A simple command line tool to consume partitions of a topic and print the +messages on the standard output. + +### Installation + + go get github.com/Shopify/sarama/tools/kafka-console-consumer + +### Usage + + # Minimum invocation + kafka-console-consumer -topic=test -brokers=kafka1:9092 + + # It will pick up a KAFKA_PEERS environment variable + export KAFKA_PEERS=kafka1:9092,kafka2:9092,kafka3:9092 + kafka-console-consumer -topic=test + + # You can specify the offset you want to start at. It can be either + # `oldest`, `newest`. The default is `newest`. + kafka-console-consumer -topic=test -offset=oldest + kafka-console-consumer -topic=test -offset=newest + + # You can specify the partition(s) you want to consume as a comma-separated + # list. The default is `all`. + kafka-console-consumer -topic=test -partitions=1,2,3 + + # Display all command line options + kafka-console-consumer -help diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-consumer/kafka-console-consumer.go b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-consumer/kafka-console-consumer.go new file mode 100644 index 0000000000000..0f1eb89a906bc --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-consumer/kafka-console-consumer.go @@ -0,0 +1,145 @@ +package main + +import ( + "flag" + "fmt" + "log" + "os" + "os/signal" + "strconv" + "strings" + "sync" + + "github.com/Shopify/sarama" +) + +var ( + brokerList = flag.String("brokers", os.Getenv("KAFKA_PEERS"), "The comma separated list of brokers in the Kafka cluster") + topic = flag.String("topic", "", "REQUIRED: the topic to consume") + partitions = flag.String("partitions", "all", "The partitions to consume, can be 'all' or comma-separated numbers") + offset = flag.String("offset", "newest", "The offset to start with. Can be `oldest`, `newest`") + verbose = flag.Bool("verbose", false, "Whether to turn on sarama logging") + bufferSize = flag.Int("buffer-size", 256, "The buffer size of the message channel.") + + logger = log.New(os.Stderr, "", log.LstdFlags) +) + +func main() { + flag.Parse() + + if *brokerList == "" { + printUsageErrorAndExit("You have to provide -brokers as a comma-separated list, or set the KAFKA_PEERS environment variable.") + } + + if *topic == "" { + printUsageErrorAndExit("-topic is required") + } + + if *verbose { + sarama.Logger = logger + } + + var initialOffset int64 + switch *offset { + case "oldest": + initialOffset = sarama.OffsetOldest + case "newest": + initialOffset = sarama.OffsetNewest + default: + printUsageErrorAndExit("-offset should be `oldest` or `newest`") + } + + c, err := sarama.NewConsumer(strings.Split(*brokerList, ","), nil) + if err != nil { + printErrorAndExit(69, "Failed to start consumer: %s", err) + } + + partitionList, err := getPartitions(c) + if err != nil { + printErrorAndExit(69, "Failed to get the list of partitions: %s", err) + } + + var ( + messages = make(chan *sarama.ConsumerMessage, *bufferSize) + closing = make(chan struct{}) + wg sync.WaitGroup + ) + + go func() { + signals := make(chan os.Signal, 1) + signal.Notify(signals, os.Kill, os.Interrupt) + <-signals + logger.Println("Initiating shutdown of consumer...") + close(closing) + }() + + for _, partition := range partitionList { + pc, err := c.ConsumePartition(*topic, partition, initialOffset) + if err != nil { + printErrorAndExit(69, "Failed to start consumer for partition %d: %s", partition, err) + } + + go func(pc sarama.PartitionConsumer) { + <-closing + pc.AsyncClose() + }(pc) + + wg.Add(1) + go func(pc sarama.PartitionConsumer) { + defer wg.Done() + for message := range pc.Messages() { + messages <- message + } + }(pc) + } + + go func() { + for msg := range messages { + fmt.Printf("Partition:\t%d\n", msg.Partition) + fmt.Printf("Offset:\t%d\n", msg.Offset) + fmt.Printf("Key:\t%s\n", string(msg.Key)) + fmt.Printf("Value:\t%s\n", string(msg.Value)) + fmt.Println() + } + }() + + wg.Wait() + logger.Println("Done consuming topic", *topic) + close(messages) + + if err := c.Close(); err != nil { + logger.Println("Failed to close consumer: ", err) + } +} + +func getPartitions(c sarama.Consumer) ([]int32, error) { + if *partitions == "all" { + return c.Partitions(*topic) + } + + tmp := strings.Split(*partitions, ",") + var pList []int32 + for i := range tmp { + val, err := strconv.ParseInt(tmp[i], 10, 32) + if err != nil { + return nil, err + } + pList = append(pList, int32(val)) + } + + return pList, nil +} + +func printErrorAndExit(code int, format string, values ...interface{}) { + fmt.Fprintf(os.Stderr, "ERROR: %s\n", fmt.Sprintf(format, values...)) + fmt.Fprintln(os.Stderr) + os.Exit(code) +} + +func printUsageErrorAndExit(format string, values ...interface{}) { + fmt.Fprintf(os.Stderr, "ERROR: %s\n", fmt.Sprintf(format, values...)) + fmt.Fprintln(os.Stderr) + fmt.Fprintln(os.Stderr, "Available command line options:") + flag.PrintDefaults() + os.Exit(64) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-partitionconsumer/.gitignore b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-partitionconsumer/.gitignore new file mode 100644 index 0000000000000..5837fe8ca69fd --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-partitionconsumer/.gitignore @@ -0,0 +1,2 @@ +kafka-console-partitionconsumer +kafka-console-partitionconsumer.test diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-partitionconsumer/README.md b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-partitionconsumer/README.md new file mode 100644 index 0000000000000..646dd5f5c2960 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-partitionconsumer/README.md @@ -0,0 +1,28 @@ +# kafka-console-partitionconsumer + +NOTE: this tool is deprecated in favour of the more general and more powerful +`kafka-console-consumer`. + +A simple command line tool to consume a partition of a topic and print the messages +on the standard output. + +### Installation + + go get github.com/Shopify/sarama/tools/kafka-console-partitionconsumer + +### Usage + + # Minimum invocation + kafka-console-partitionconsumer -topic=test -partition=4 -brokers=kafka1:9092 + + # It will pick up a KAFKA_PEERS environment variable + export KAFKA_PEERS=kafka1:9092,kafka2:9092,kafka3:9092 + kafka-console-partitionconsumer -topic=test -partition=4 + + # You can specify the offset you want to start at. It can be either + # `oldest`, `newest`, or a specific offset number + kafka-console-partitionconsumer -topic=test -partition=3 -offset=oldest + kafka-console-partitionconsumer -topic=test -partition=2 -offset=1337 + + # Display all command line options + kafka-console-partitionconsumer -help diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-partitionconsumer/kafka-console-partitionconsumer.go b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-partitionconsumer/kafka-console-partitionconsumer.go new file mode 100644 index 0000000000000..d5e4464de1843 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-partitionconsumer/kafka-console-partitionconsumer.go @@ -0,0 +1,102 @@ +package main + +import ( + "flag" + "fmt" + "log" + "os" + "os/signal" + "strconv" + "strings" + + "github.com/Shopify/sarama" +) + +var ( + brokerList = flag.String("brokers", os.Getenv("KAFKA_PEERS"), "The comma separated list of brokers in the Kafka cluster") + topic = flag.String("topic", "", "REQUIRED: the topic to consume") + partition = flag.Int("partition", -1, "REQUIRED: the partition to consume") + offset = flag.String("offset", "newest", "The offset to start with. Can be `oldest`, `newest`, or an actual offset") + verbose = flag.Bool("verbose", false, "Whether to turn on sarama logging") + + logger = log.New(os.Stderr, "", log.LstdFlags) +) + +func main() { + flag.Parse() + + if *brokerList == "" { + printUsageErrorAndExit("You have to provide -brokers as a comma-separated list, or set the KAFKA_PEERS environment variable.") + } + + if *topic == "" { + printUsageErrorAndExit("-topic is required") + } + + if *partition == -1 { + printUsageErrorAndExit("-partition is required") + } + + if *verbose { + sarama.Logger = logger + } + + var ( + initialOffset int64 + offsetError error + ) + switch *offset { + case "oldest": + initialOffset = sarama.OffsetOldest + case "newest": + initialOffset = sarama.OffsetNewest + default: + initialOffset, offsetError = strconv.ParseInt(*offset, 10, 64) + } + + if offsetError != nil { + printUsageErrorAndExit("Invalid initial offset: %s", *offset) + } + + c, err := sarama.NewConsumer(strings.Split(*brokerList, ","), nil) + if err != nil { + printErrorAndExit(69, "Failed to start consumer: %s", err) + } + + pc, err := c.ConsumePartition(*topic, int32(*partition), initialOffset) + if err != nil { + printErrorAndExit(69, "Failed to start partition consumer: %s", err) + } + + go func() { + signals := make(chan os.Signal, 1) + signal.Notify(signals, os.Kill, os.Interrupt) + <-signals + pc.AsyncClose() + }() + + for msg := range pc.Messages() { + fmt.Printf("Offset:\t%d\n", msg.Offset) + fmt.Printf("Key:\t%s\n", string(msg.Key)) + fmt.Printf("Value:\t%s\n", string(msg.Value)) + fmt.Println() + } + + if err := c.Close(); err != nil { + logger.Println("Failed to close consumer: ", err) + } +} + +func printErrorAndExit(code int, format string, values ...interface{}) { + fmt.Fprintf(os.Stderr, "ERROR: %s\n", fmt.Sprintf(format, values...)) + fmt.Fprintln(os.Stderr) + os.Exit(code) +} + +func printUsageErrorAndExit(format string, values ...interface{}) { + fmt.Fprintf(os.Stderr, "ERROR: %s\n", fmt.Sprintf(format, values...)) + fmt.Fprintln(os.Stderr) + fmt.Fprintln(os.Stderr, "Available command line options:") + flag.PrintDefaults() + os.Exit(64) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-producer/.gitignore b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-producer/.gitignore new file mode 100644 index 0000000000000..2b9e563a1c62b --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-producer/.gitignore @@ -0,0 +1,2 @@ +kafka-console-producer +kafka-console-producer.test diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-producer/README.md b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-producer/README.md new file mode 100644 index 0000000000000..6b3a65f211ab7 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-producer/README.md @@ -0,0 +1,34 @@ +# kafka-console-producer + +A simple command line tool to produce a single message to Kafka. + +### Installation + + go get github.com/Shopify/sarama/tools/kafka-console-producer + + +### Usage + + # Minimum invocation + kafka-console-producer -topic=test -value=value -brokers=kafka1:9092 + + # It will pick up a KAFKA_PEERS environment variable + export KAFKA_PEERS=kafka1:9092,kafka2:9092,kafka3:9092 + kafka-console-producer -topic=test -value=value + + # It will read the value from stdin by using pipes + echo "hello world" | kafka-console-producer -topic=test + + # Specify a key: + echo "hello world" | kafka-console-producer -topic=test -key=key + + # Partitioning: by default, kafka-console-producer will partition as follows: + # - manual partitioning if a -partition is provided + # - hash partitioning by key if a -key is provided + # - random partioning otherwise. + # + # You can override this using the -partitioner argument: + echo "hello world" | kafka-console-producer -topic=test -key=key -partitioner=random + + # Display all command line options + kafka-console-producer -help diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-producer/kafka-console-producer.go b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-producer/kafka-console-producer.go new file mode 100644 index 0000000000000..6a1765d7c9f1c --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/tools/kafka-console-producer/kafka-console-producer.go @@ -0,0 +1,118 @@ +package main + +import ( + "flag" + "fmt" + "io/ioutil" + "log" + "os" + "strings" + + "github.com/Shopify/sarama" +) + +var ( + brokerList = flag.String("brokers", os.Getenv("KAFKA_PEERS"), "The comma separated list of brokers in the Kafka cluster. You can also set the KAFKA_PEERS environment variable") + topic = flag.String("topic", "", "REQUIRED: the topic to produce to") + key = flag.String("key", "", "The key of the message to produce. Can be empty.") + value = flag.String("value", "", "REQUIRED: the value of the message to produce. You can also provide the value on stdin.") + partitioner = flag.String("partitioner", "", "The partitioning scheme to use. Can be `hash`, `manual`, or `random`") + partition = flag.Int("partition", -1, "The partition to produce to.") + verbose = flag.Bool("verbose", false, "Turn on sarama logging to stderr") + silent = flag.Bool("silent", false, "Turn off printing the message's topic, partition, and offset to stdout") + + logger = log.New(os.Stderr, "", log.LstdFlags) +) + +func main() { + flag.Parse() + + if *brokerList == "" { + printUsageErrorAndExit("no -brokers specified. Alternatively, set the KAFKA_PEERS environment variable") + } + + if *topic == "" { + printUsageErrorAndExit("no -topic specified") + } + + if *verbose { + sarama.Logger = logger + } + + config := sarama.NewConfig() + config.Producer.RequiredAcks = sarama.WaitForAll + + switch *partitioner { + case "": + if *partition >= 0 { + config.Producer.Partitioner = sarama.NewManualPartitioner + } else { + config.Producer.Partitioner = sarama.NewHashPartitioner + } + case "hash": + config.Producer.Partitioner = sarama.NewHashPartitioner + case "random": + config.Producer.Partitioner = sarama.NewRandomPartitioner + case "manual": + config.Producer.Partitioner = sarama.NewManualPartitioner + if *partition == -1 { + printUsageErrorAndExit("-partition is required when partitioning manually") + } + default: + printUsageErrorAndExit(fmt.Sprintf("Partitioner %s not supported.", *partitioner)) + } + + message := &sarama.ProducerMessage{Topic: *topic, Partition: int32(*partition)} + + if *key != "" { + message.Key = sarama.StringEncoder(*key) + } + + if *value != "" { + message.Value = sarama.StringEncoder(*value) + } else if stdinAvailable() { + bytes, err := ioutil.ReadAll(os.Stdin) + if err != nil { + printErrorAndExit(66, "Failed to read data from the standard input: %s", err) + } + message.Value = sarama.ByteEncoder(bytes) + } else { + printUsageErrorAndExit("-value is required, or you have to provide the value on stdin") + } + + producer, err := sarama.NewSyncProducer(strings.Split(*brokerList, ","), config) + if err != nil { + printErrorAndExit(69, "Failed to open Kafka producer: %s", err) + } + defer func() { + if err := producer.Close(); err != nil { + logger.Println("Failed to close Kafka producer cleanly:", err) + } + }() + + partition, offset, err := producer.SendMessage(message) + if err != nil { + printErrorAndExit(69, "Failed to produce message: %s", err) + } else if !*silent { + fmt.Printf("topic=%s\tpartition=%d\toffset=%d\n", *topic, partition, offset) + } +} + +func printErrorAndExit(code int, format string, values ...interface{}) { + fmt.Fprintf(os.Stderr, "ERROR: %s\n", fmt.Sprintf(format, values...)) + fmt.Fprintln(os.Stderr) + os.Exit(code) +} + +func printUsageErrorAndExit(message string) { + fmt.Fprintln(os.Stderr, "ERROR:", message) + fmt.Fprintln(os.Stderr) + fmt.Fprintln(os.Stderr, "Available command line options:") + flag.PrintDefaults() + os.Exit(64) +} + +func stdinAvailable() bool { + stat, _ := os.Stdin.Stat() + return (stat.Mode() & os.ModeCharDevice) == 0 +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/utils.go b/Godeps/_workspace/src/github.com/Shopify/sarama/utils.go new file mode 100644 index 0000000000000..fef7c73816ec9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/utils.go @@ -0,0 +1,89 @@ +package sarama + +import "sort" + +type none struct{} + +// make []int32 sortable so we can sort partition numbers +type int32Slice []int32 + +func (slice int32Slice) Len() int { + return len(slice) +} + +func (slice int32Slice) Less(i, j int) bool { + return slice[i] < slice[j] +} + +func (slice int32Slice) Swap(i, j int) { + slice[i], slice[j] = slice[j], slice[i] +} + +func dupeAndSort(input []int32) []int32 { + ret := make([]int32, 0, len(input)) + for _, val := range input { + ret = append(ret, val) + } + + sort.Sort(int32Slice(ret)) + return ret +} + +func withRecover(fn func()) { + defer func() { + handler := PanicHandler + if handler != nil { + if err := recover(); err != nil { + handler(err) + } + } + }() + + fn() +} + +func safeAsyncClose(b *Broker) { + tmp := b // local var prevents clobbering in goroutine + go withRecover(func() { + if connected, _ := tmp.Connected(); connected { + if err := tmp.Close(); err != nil { + Logger.Println("Error closing broker", tmp.ID(), ":", err) + } + } + }) +} + +// Encoder is a simple interface for any type that can be encoded as an array of bytes +// in order to be sent as the key or value of a Kafka message. Length() is provided as an +// optimization, and must return the same as len() on the result of Encode(). +type Encoder interface { + Encode() ([]byte, error) + Length() int +} + +// make strings and byte slices encodable for convenience so they can be used as keys +// and/or values in kafka messages + +// StringEncoder implements the Encoder interface for Go strings so that they can be used +// as the Key or Value in a ProducerMessage. +type StringEncoder string + +func (s StringEncoder) Encode() ([]byte, error) { + return []byte(s), nil +} + +func (s StringEncoder) Length() int { + return len(s) +} + +// ByteEncoder implements the Encoder interface for Go byte slices so that they can be used +// as the Key or Value in a ProducerMessage. +type ByteEncoder []byte + +func (b ByteEncoder) Encode() ([]byte, error) { + return b, nil +} + +func (b ByteEncoder) Length() int { + return len(b) +} diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/boot_cluster.sh b/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/boot_cluster.sh new file mode 100644 index 0000000000000..95e47dde43443 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/boot_cluster.sh @@ -0,0 +1,22 @@ +#!/bin/sh + +set -ex + +# Launch and wait for toxiproxy +${REPOSITORY_ROOT}/vagrant/run_toxiproxy.sh & +while ! nc -q 1 localhost 2181 ${KAFKA_INSTALL_ROOT}/zookeeper-${ZK_PORT}/myid +done diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/kafka.conf b/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/kafka.conf new file mode 100644 index 0000000000000..d975de4387d4e --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/kafka.conf @@ -0,0 +1,5 @@ +start on started zookeeper-ZK_PORT +stop on stopping zookeeper-ZK_PORT + +pre-start exec sleep 2 +exec /opt/kafka-KAFKAID/bin/kafka-server-start.sh /opt/kafka-KAFKAID/config/server.properties diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/provision.sh b/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/provision.sh new file mode 100644 index 0000000000000..0a2d7778500fe --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/provision.sh @@ -0,0 +1,15 @@ +#!/bin/sh + +set -ex + +apt-get update +yes | apt-get install default-jre + +export KAFKA_INSTALL_ROOT=/opt +export KAFKA_HOSTNAME=192.168.100.67 +export KAFKA_VERSION=0.8.2.1 +export REPOSITORY_ROOT=/vagrant + +sh /vagrant/vagrant/install_cluster.sh +sh /vagrant/vagrant/setup_services.sh +sh /vagrant/vagrant/create_topics.sh diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/run_toxiproxy.sh b/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/run_toxiproxy.sh new file mode 100644 index 0000000000000..e52c00e7b5a86 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/run_toxiproxy.sh @@ -0,0 +1,22 @@ +#!/bin/sh + +set -ex + +${KAFKA_INSTALL_ROOT}/toxiproxy -port 8474 -host 0.0.0.0 & +PID=$! + +while ! nc -q 1 localhost 8474 + +# The number of threads handling network requests +num.network.threads=2 + +# The number of threads doing disk I/O +num.io.threads=8 + +# The send buffer (SO_SNDBUF) used by the socket server +socket.send.buffer.bytes=1048576 + +# The receive buffer (SO_RCVBUF) used by the socket server +socket.receive.buffer.bytes=1048576 + +# The maximum size of a request that the socket server will accept (protection against OOM) +socket.request.max.bytes=104857600 + + +############################# Log Basics ############################# + +# A comma seperated list of directories under which to store log files +log.dirs=KAFKA_DATADIR + +# The default number of log partitions per topic. More partitions allow greater +# parallelism for consumption, but this will also result in more files across +# the brokers. +num.partitions=2 + +# Create new topics with a replication factor of 2 so failover can be tested +# more easily. +default.replication.factor=2 + +auto.create.topics.enable=false +delete.topic.enable=true + +############################# Log Flush Policy ############################# + +# Messages are immediately written to the filesystem but by default we only fsync() to sync +# the OS cache lazily. The following configurations control the flush of data to disk. +# There are a few important trade-offs here: +# 1. Durability: Unflushed data may be lost if you are not using replication. +# 2. Latency: Very large flush intervals may lead to latency spikes when the flush does occur as there will be a lot of data to flush. +# 3. Throughput: The flush is generally the most expensive operation, and a small flush interval may lead to exceessive seeks. +# The settings below allow one to configure the flush policy to flush data after a period of time or +# every N messages (or both). This can be done globally and overridden on a per-topic basis. + +# The number of messages to accept before forcing a flush of data to disk +#log.flush.interval.messages=10000 + +# The maximum amount of time a message can sit in a log before we force a flush +#log.flush.interval.ms=1000 + +############################# Log Retention Policy ############################# + +# The following configurations control the disposal of log segments. The policy can +# be set to delete segments after a period of time, or after a given size has accumulated. +# A segment will be deleted whenever *either* of these criteria are met. Deletion always happens +# from the end of the log. + +# The minimum age of a log file to be eligible for deletion +log.retention.hours=168 + +# A size-based retention policy for logs. Segments are pruned from the log as long as the remaining +# segments don't drop below log.retention.bytes. +log.retention.bytes=268435456 + +# The maximum size of a log segment file. When this size is reached a new log segment will be created. +log.segment.bytes=268435456 + +# The interval at which log segments are checked to see if they can be deleted according +# to the retention policies +log.retention.check.interval.ms=60000 + +# By default the log cleaner is disabled and the log retention policy will default to just delete segments after their retention expires. +# If log.cleaner.enable=true is set the cleaner will be enabled and individual logs can then be marked for log compaction. +log.cleaner.enable=false + +############################# Zookeeper ############################# + +# Zookeeper connection string (see zookeeper docs for details). +# This is a comma separated host:port pairs, each corresponding to a zk +# server. e.g. "127.0.0.1:3000,127.0.0.1:3001,127.0.0.1:3002". +# You can also append an optional chroot string to the urls to specify the +# root directory for all kafka znodes. +zookeeper.connect=localhost:ZK_PORT + +# Timeout in ms for connecting to zookeeper +zookeeper.session.timeout.ms=3000 +zookeeper.connection.timeout.ms=3000 diff --git a/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/setup_services.sh b/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/setup_services.sh new file mode 100644 index 0000000000000..81d8ea05d3b14 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Shopify/sarama/vagrant/setup_services.sh @@ -0,0 +1,29 @@ +#!/bin/sh + +set -ex + +stop toxiproxy || true +cp ${REPOSITORY_ROOT}/vagrant/toxiproxy.conf /etc/init/toxiproxy.conf +cp ${REPOSITORY_ROOT}/vagrant/run_toxiproxy.sh ${KAFKA_INSTALL_ROOT}/ +start toxiproxy + +for i in 1 2 3 4 5; do + ZK_PORT=`expr $i + 2180` + KAFKA_PORT=`expr $i + 9090` + + stop zookeeper-${ZK_PORT} || true + + # set up zk service + cp ${REPOSITORY_ROOT}/vagrant/zookeeper.conf /etc/init/zookeeper-${ZK_PORT}.conf + sed -i s/KAFKAID/${KAFKA_PORT}/g /etc/init/zookeeper-${ZK_PORT}.conf + + # set up kafka service + cp ${REPOSITORY_ROOT}/vagrant/kafka.conf /etc/init/kafka-${KAFKA_PORT}.conf + sed -i s/KAFKAID/${KAFKA_PORT}/g /etc/init/kafka-${KAFKA_PORT}.conf + sed -i s/ZK_PORT/${ZK_PORT}/g /etc/init/kafka-${KAFKA_PORT}.conf + + start zookeeper-${ZK_PORT} +done + +# Wait for the last kafka node to finish booting +while ! nc -q 1 localhost 29095  [![Build Status](https://travis-ci.org/Sirupsen/logrus.svg?branch=master)](https://travis-ci.org/Sirupsen/logrus) [![godoc reference](https://godoc.org/github.com/Sirupsen/logrus?status.png)][godoc] + +Logrus is a structured logger for Go (golang), completely API compatible with +the standard library logger. [Godoc][godoc]. **Please note the Logrus API is not +yet stable (pre 1.0). Logrus itself is completely stable and has been used in +many large deployments. The core API is unlikely to change much but please +version control your Logrus to make sure you aren't fetching latest `master` on +every build.** + +Nicely color-coded in development (when a TTY is attached, otherwise just +plain text): + +![Colored](http://i.imgur.com/PY7qMwd.png) + +With `log.Formatter = new(logrus.JSONFormatter)`, for easy parsing by logstash +or Splunk: + +```json +{"animal":"walrus","level":"info","msg":"A group of walrus emerges from the +ocean","size":10,"time":"2014-03-10 19:57:38.562264131 -0400 EDT"} + +{"level":"warning","msg":"The group's number increased tremendously!", +"number":122,"omg":true,"time":"2014-03-10 19:57:38.562471297 -0400 EDT"} + +{"animal":"walrus","level":"info","msg":"A giant walrus appears!", +"size":10,"time":"2014-03-10 19:57:38.562500591 -0400 EDT"} + +{"animal":"walrus","level":"info","msg":"Tremendously sized cow enters the ocean.", +"size":9,"time":"2014-03-10 19:57:38.562527896 -0400 EDT"} + +{"level":"fatal","msg":"The ice breaks!","number":100,"omg":true, +"time":"2014-03-10 19:57:38.562543128 -0400 EDT"} +``` + +With the default `log.Formatter = new(&log.TextFormatter{})` when a TTY is not +attached, the output is compatible with the +[logfmt](http://godoc.org/github.com/kr/logfmt) format: + +```text +time="2015-03-26T01:27:38-04:00" level=debug msg="Started observing beach" animal=walrus number=8 +time="2015-03-26T01:27:38-04:00" level=info msg="A group of walrus emerges from the ocean" animal=walrus size=10 +time="2015-03-26T01:27:38-04:00" level=warning msg="The group's number increased tremendously!" number=122 omg=true +time="2015-03-26T01:27:38-04:00" level=debug msg="Temperature changes" temperature=-4 +time="2015-03-26T01:27:38-04:00" level=panic msg="It's over 9000!" animal=orca size=9009 +time="2015-03-26T01:27:38-04:00" level=fatal msg="The ice breaks!" err=&{0x2082280c0 map[animal:orca size:9009] 2015-03-26 01:27:38.441574009 -0400 EDT panic It's over 9000!} number=100 omg=true +exit status 1 +``` + +#### Example + +The simplest way to use Logrus is simply the package-level exported logger: + +```go +package main + +import ( + log "github.com/Sirupsen/logrus" +) + +func main() { + log.WithFields(log.Fields{ + "animal": "walrus", + }).Info("A walrus appears") +} +``` + +Note that it's completely api-compatible with the stdlib logger, so you can +replace your `log` imports everywhere with `log "github.com/Sirupsen/logrus"` +and you'll now have the flexibility of Logrus. You can customize it all you +want: + +```go +package main + +import ( + "os" + log "github.com/Sirupsen/logrus" + "github.com/Sirupsen/logrus/hooks/airbrake" +) + +func init() { + // Log as JSON instead of the default ASCII formatter. + log.SetFormatter(&log.JSONFormatter{}) + + // Use the Airbrake hook to report errors that have Error severity or above to + // an exception tracker. You can create custom hooks, see the Hooks section. + log.AddHook(airbrake.NewHook("https://example.com", "xyz", "development")) + + // Output to stderr instead of stdout, could also be a file. + log.SetOutput(os.Stderr) + + // Only log the warning severity or above. + log.SetLevel(log.WarnLevel) +} + +func main() { + log.WithFields(log.Fields{ + "animal": "walrus", + "size": 10, + }).Info("A group of walrus emerges from the ocean") + + log.WithFields(log.Fields{ + "omg": true, + "number": 122, + }).Warn("The group's number increased tremendously!") + + log.WithFields(log.Fields{ + "omg": true, + "number": 100, + }).Fatal("The ice breaks!") + + // A common pattern is to re-use fields between logging statements by re-using + // the logrus.Entry returned from WithFields() + contextLogger := log.WithFields(log.Fields{ + "common": "this is a common field", + "other": "I also should be logged always", + }) + + contextLogger.Info("I'll be logged with common and other field") + contextLogger.Info("Me too") +} +``` + +For more advanced usage such as logging to multiple locations from the same +application, you can also create an instance of the `logrus` Logger: + +```go +package main + +import ( + "github.com/Sirupsen/logrus" +) + +// Create a new instance of the logger. You can have any number of instances. +var log = logrus.New() + +func main() { + // The API for setting attributes is a little different than the package level + // exported logger. See Godoc. + log.Out = os.Stderr + + log.WithFields(logrus.Fields{ + "animal": "walrus", + "size": 10, + }).Info("A group of walrus emerges from the ocean") +} +``` + +#### Fields + +Logrus encourages careful, structured logging though logging fields instead of +long, unparseable error messages. For example, instead of: `log.Fatalf("Failed +to send event %s to topic %s with key %d")`, you should log the much more +discoverable: + +```go +log.WithFields(log.Fields{ + "event": event, + "topic": topic, + "key": key, +}).Fatal("Failed to send event") +``` + +We've found this API forces you to think about logging in a way that produces +much more useful logging messages. We've been in countless situations where just +a single added field to a log statement that was already there would've saved us +hours. The `WithFields` call is optional. + +In general, with Logrus using any of the `printf`-family functions should be +seen as a hint you should add a field, however, you can still use the +`printf`-family functions with Logrus. + +#### Hooks + +You can add hooks for logging levels. For example to send errors to an exception +tracking service on `Error`, `Fatal` and `Panic`, info to StatsD or log to +multiple places simultaneously, e.g. syslog. + +Logrus comes with [built-in hooks](hooks/). Add those, or your custom hook, in +`init`: + +```go +import ( + log "github.com/Sirupsen/logrus" + "github.com/Sirupsen/logrus/hooks/airbrake" + logrus_syslog "github.com/Sirupsen/logrus/hooks/syslog" + "log/syslog" +) + +func init() { + log.AddHook(airbrake.NewHook("https://example.com", "xyz", "development")) + + hook, err := logrus_syslog.NewSyslogHook("udp", "localhost:514", syslog.LOG_INFO, "") + if err != nil { + log.Error("Unable to connect to local syslog daemon") + } else { + log.AddHook(hook) + } +} +``` + + +| Hook | Description | +| ----- | ----------- | +| [Airbrake](https://github.com/Sirupsen/logrus/blob/master/hooks/airbrake/airbrake.go) | Send errors to an exception tracking service compatible with the Airbrake API. Uses [`airbrake-go`](https://github.com/tobi/airbrake-go) behind the scenes. | +| [Papertrail](https://github.com/Sirupsen/logrus/blob/master/hooks/papertrail/papertrail.go) | Send errors to the Papertrail hosted logging service via UDP. | +| [Syslog](https://github.com/Sirupsen/logrus/blob/master/hooks/syslog/syslog.go) | Send errors to remote syslog server. Uses standard library `log/syslog` behind the scenes. | +| [BugSnag](https://github.com/Sirupsen/logrus/blob/master/hooks/bugsnag/bugsnag.go) | Send errors to the Bugsnag exception tracking service. | +| [Sentry](https://github.com/Sirupsen/logrus/blob/master/hooks/sentry/sentry.go) | Send errors to the Sentry error logging and aggregation service. | +| [Hiprus](https://github.com/nubo/hiprus) | Send errors to a channel in hipchat. | +| [Logrusly](https://github.com/sebest/logrusly) | Send logs to [Loggly](https://www.loggly.com/) | +| [Slackrus](https://github.com/johntdyer/slackrus) | Hook for Slack chat. | +| [Journalhook](https://github.com/wercker/journalhook) | Hook for logging to `systemd-journald` | +| [Graylog](https://github.com/gemnasium/logrus-hooks/tree/master/graylog) | Hook for logging to [Graylog](http://graylog2.org/) | +| [Raygun](https://github.com/squirkle/logrus-raygun-hook) | Hook for logging to [Raygun.io](http://raygun.io/) | +| [LFShook](https://github.com/rifflock/lfshook) | Hook for logging to the local filesystem | +| [Honeybadger](https://github.com/agonzalezro/logrus_honeybadger) | Hook for sending exceptions to Honeybadger | +| [Mail](https://github.com/zbindenren/logrus_mail) | Hook for sending exceptions via mail | +| [Rollrus](https://github.com/heroku/rollrus) | Hook for sending errors to rollbar | +| [Fluentd](https://github.com/evalphobia/logrus_fluent) | Hook for logging to fluentd | + +#### Level logging + +Logrus has six logging levels: Debug, Info, Warning, Error, Fatal and Panic. + +```go +log.Debug("Useful debugging information.") +log.Info("Something noteworthy happened!") +log.Warn("You should probably take a look at this.") +log.Error("Something failed but I'm not quitting.") +// Calls os.Exit(1) after logging +log.Fatal("Bye.") +// Calls panic() after logging +log.Panic("I'm bailing.") +``` + +You can set the logging level on a `Logger`, then it will only log entries with +that severity or anything above it: + +```go +// Will log anything that is info or above (warn, error, fatal, panic). Default. +log.SetLevel(log.InfoLevel) +``` + +It may be useful to set `log.Level = logrus.DebugLevel` in a debug or verbose +environment if your application has that. + +#### Entries + +Besides the fields added with `WithField` or `WithFields` some fields are +automatically added to all logging events: + +1. `time`. The timestamp when the entry was created. +2. `msg`. The logging message passed to `{Info,Warn,Error,Fatal,Panic}` after + the `AddFields` call. E.g. `Failed to send event.` +3. `level`. The logging level. E.g. `info`. + +#### Environments + +Logrus has no notion of environment. + +If you wish for hooks and formatters to only be used in specific environments, +you should handle that yourself. For example, if your application has a global +variable `Environment`, which is a string representation of the environment you +could do: + +```go +import ( + log "github.com/Sirupsen/logrus" +) + +init() { + // do something here to set environment depending on an environment variable + // or command-line flag + if Environment == "production" { + log.SetFormatter(&logrus.JSONFormatter{}) + } else { + // The TextFormatter is default, you don't actually have to do this. + log.SetFormatter(&log.TextFormatter{}) + } +} +``` + +This configuration is how `logrus` was intended to be used, but JSON in +production is mostly only useful if you do log aggregation with tools like +Splunk or Logstash. + +#### Formatters + +The built-in logging formatters are: + +* `logrus.TextFormatter`. Logs the event in colors if stdout is a tty, otherwise + without colors. + * *Note:* to force colored output when there is no TTY, set the `ForceColors` + field to `true`. To force no colored output even if there is a TTY set the + `DisableColors` field to `true` +* `logrus.JSONFormatter`. Logs fields as JSON. +* `logrus_logstash.LogstashFormatter`. Logs fields as Logstash Events (http://logstash.net). + + ```go + logrus.SetFormatter(&logrus_logstash.LogstashFormatter{Type: “application_name"}) + ``` + +Third party logging formatters: + +* [`zalgo`](https://github.com/aybabtme/logzalgo): invoking the P͉̫o̳̼̊w̖͈̰͎e̬͔̭͂r͚̼̹̲ ̫͓͉̳͈ō̠͕͖̚f̝͍̠ ͕̲̞͖͑Z̖̫̤̫ͪa͉̬͈̗l͖͎g̳̥o̰̥̅!̣͔̲̻͊̄ ̙̘̦̹̦. + +You can define your formatter by implementing the `Formatter` interface, +requiring a `Format` method. `Format` takes an `*Entry`. `entry.Data` is a +`Fields` type (`map[string]interface{}`) with all your fields as well as the +default ones (see Entries section above): + +```go +type MyJSONFormatter struct { +} + +log.SetFormatter(new(MyJSONFormatter)) + +func (f *JSONFormatter) Format(entry *Entry) ([]byte, error) { + // Note this doesn't include Time, Level and Message which are available on + // the Entry. Consult `godoc` on information about those fields or read the + // source of the official loggers. + serialized, err := json.Marshal(entry.Data) + if err != nil { + return nil, fmt.Errorf("Failed to marshal fields to JSON, %v", err) + } + return append(serialized, '\n'), nil +} +``` + +#### Logger as an `io.Writer` + +Logrus can be transformed into an `io.Writer`. That writer is the end of an `io.Pipe` and it is your responsibility to close it. + +```go +w := logger.Writer() +defer w.Close() + +srv := http.Server{ + // create a stdlib log.Logger that writes to + // logrus.Logger. + ErrorLog: log.New(w, "", 0), +} +``` + +Each line written to that writer will be printed the usual way, using formatters +and hooks. The level for those entries is `info`. + +#### Rotation + +Log rotation is not provided with Logrus. Log rotation should be done by an +external program (like `logrotate(8)`) that can compress and delete old log +entries. It should not be a feature of the application-level logger. + + +[godoc]: https://godoc.org/github.com/Sirupsen/logrus diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/entry.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/entry.go new file mode 100644 index 0000000000000..04673a075db21 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/entry.go @@ -0,0 +1,256 @@ +package logrus + +import ( + "bytes" + "fmt" + "io" + "os" + "time" +) + +// An entry is the final or intermediate Logrus logging entry. It contains all +// the fields passed with WithField{,s}. It's finally logged when Debug, Info, +// Warn, Error, Fatal or Panic is called on it. These objects can be reused and +// passed around as much as you wish to avoid field duplication. +type Entry struct { + Logger *Logger + + // Contains all the fields set by the user. + Data Fields + + // Time at which the log entry was created + Time time.Time + + // Level the log entry was logged at: Debug, Info, Warn, Error, Fatal or Panic + Level Level + + // Message passed to Debug, Info, Warn, Error, Fatal or Panic + Message string +} + +func NewEntry(logger *Logger) *Entry { + return &Entry{ + Logger: logger, + // Default is three fields, give a little extra room + Data: make(Fields, 5), + } +} + +// Returns a reader for the entry, which is a proxy to the formatter. +func (entry *Entry) Reader() (*bytes.Buffer, error) { + serialized, err := entry.Logger.Formatter.Format(entry) + return bytes.NewBuffer(serialized), err +} + +// Returns the string representation from the reader and ultimately the +// formatter. +func (entry *Entry) String() (string, error) { + reader, err := entry.Reader() + if err != nil { + return "", err + } + + return reader.String(), err +} + +// Add a single field to the Entry. +func (entry *Entry) WithField(key string, value interface{}) *Entry { + return entry.WithFields(Fields{key: value}) +} + +// Add a map of fields to the Entry. +func (entry *Entry) WithFields(fields Fields) *Entry { + data := Fields{} + for k, v := range entry.Data { + data[k] = v + } + for k, v := range fields { + data[k] = v + } + return &Entry{Logger: entry.Logger, Data: data} +} + +// This function is not declared with a pointer value because otherwise +// race conditions will occur when using multiple goroutines +func (entry Entry) log(level Level, msg string) { + entry.Time = time.Now() + entry.Level = level + entry.Message = msg + + if err := entry.Logger.Hooks.Fire(level, &entry); err != nil { + entry.Logger.mu.Lock() + fmt.Fprintf(os.Stderr, "Failed to fire hook: %v\n", err) + entry.Logger.mu.Unlock() + } + + reader, err := entry.Reader() + if err != nil { + entry.Logger.mu.Lock() + fmt.Fprintf(os.Stderr, "Failed to obtain reader, %v\n", err) + entry.Logger.mu.Unlock() + } + + entry.Logger.mu.Lock() + defer entry.Logger.mu.Unlock() + + _, err = io.Copy(entry.Logger.Out, reader) + if err != nil { + fmt.Fprintf(os.Stderr, "Failed to write to log, %v\n", err) + } + + // To avoid Entry#log() returning a value that only would make sense for + // panic() to use in Entry#Panic(), we avoid the allocation by checking + // directly here. + if level <= PanicLevel { + panic(&entry) + } +} + +func (entry *Entry) Debug(args ...interface{}) { + if entry.Logger.Level >= DebugLevel { + entry.log(DebugLevel, fmt.Sprint(args...)) + } +} + +func (entry *Entry) Print(args ...interface{}) { + entry.Info(args...) +} + +func (entry *Entry) Info(args ...interface{}) { + if entry.Logger.Level >= InfoLevel { + entry.log(InfoLevel, fmt.Sprint(args...)) + } +} + +func (entry *Entry) Warn(args ...interface{}) { + if entry.Logger.Level >= WarnLevel { + entry.log(WarnLevel, fmt.Sprint(args...)) + } +} + +func (entry *Entry) Warning(args ...interface{}) { + entry.Warn(args...) +} + +func (entry *Entry) Error(args ...interface{}) { + if entry.Logger.Level >= ErrorLevel { + entry.log(ErrorLevel, fmt.Sprint(args...)) + } +} + +func (entry *Entry) Fatal(args ...interface{}) { + if entry.Logger.Level >= FatalLevel { + entry.log(FatalLevel, fmt.Sprint(args...)) + } + os.Exit(1) +} + +func (entry *Entry) Panic(args ...interface{}) { + if entry.Logger.Level >= PanicLevel { + entry.log(PanicLevel, fmt.Sprint(args...)) + } + panic(fmt.Sprint(args...)) +} + +// Entry Printf family functions + +func (entry *Entry) Debugf(format string, args ...interface{}) { + if entry.Logger.Level >= DebugLevel { + entry.Debug(fmt.Sprintf(format, args...)) + } +} + +func (entry *Entry) Infof(format string, args ...interface{}) { + if entry.Logger.Level >= InfoLevel { + entry.Info(fmt.Sprintf(format, args...)) + } +} + +func (entry *Entry) Printf(format string, args ...interface{}) { + entry.Infof(format, args...) +} + +func (entry *Entry) Warnf(format string, args ...interface{}) { + if entry.Logger.Level >= WarnLevel { + entry.Warn(fmt.Sprintf(format, args...)) + } +} + +func (entry *Entry) Warningf(format string, args ...interface{}) { + entry.Warnf(format, args...) +} + +func (entry *Entry) Errorf(format string, args ...interface{}) { + if entry.Logger.Level >= ErrorLevel { + entry.Error(fmt.Sprintf(format, args...)) + } +} + +func (entry *Entry) Fatalf(format string, args ...interface{}) { + if entry.Logger.Level >= FatalLevel { + entry.Fatal(fmt.Sprintf(format, args...)) + } + os.Exit(1) +} + +func (entry *Entry) Panicf(format string, args ...interface{}) { + if entry.Logger.Level >= PanicLevel { + entry.Panic(fmt.Sprintf(format, args...)) + } +} + +// Entry Println family functions + +func (entry *Entry) Debugln(args ...interface{}) { + if entry.Logger.Level >= DebugLevel { + entry.Debug(entry.sprintlnn(args...)) + } +} + +func (entry *Entry) Infoln(args ...interface{}) { + if entry.Logger.Level >= InfoLevel { + entry.Info(entry.sprintlnn(args...)) + } +} + +func (entry *Entry) Println(args ...interface{}) { + entry.Infoln(args...) +} + +func (entry *Entry) Warnln(args ...interface{}) { + if entry.Logger.Level >= WarnLevel { + entry.Warn(entry.sprintlnn(args...)) + } +} + +func (entry *Entry) Warningln(args ...interface{}) { + entry.Warnln(args...) +} + +func (entry *Entry) Errorln(args ...interface{}) { + if entry.Logger.Level >= ErrorLevel { + entry.Error(entry.sprintlnn(args...)) + } +} + +func (entry *Entry) Fatalln(args ...interface{}) { + if entry.Logger.Level >= FatalLevel { + entry.Fatal(entry.sprintlnn(args...)) + } + os.Exit(1) +} + +func (entry *Entry) Panicln(args ...interface{}) { + if entry.Logger.Level >= PanicLevel { + entry.Panic(entry.sprintlnn(args...)) + } +} + +// Sprintlnn => Sprint no newline. This is to get the behavior of how +// fmt.Sprintln where spaces are always added between operands, regardless of +// their type. Instead of vendoring the Sprintln implementation to spare a +// string allocation, we do the simplest thing. +func (entry *Entry) sprintlnn(args ...interface{}) string { + msg := fmt.Sprintln(args...) + return msg[:len(msg)-1] +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/entry_test.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/entry_test.go new file mode 100644 index 0000000000000..98717df4901bd --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/entry_test.go @@ -0,0 +1,53 @@ +package logrus + +import ( + "bytes" + "fmt" + "testing" + + "github.com/stretchr/testify/assert" +) + +func TestEntryPanicln(t *testing.T) { + errBoom := fmt.Errorf("boom time") + + defer func() { + p := recover() + assert.NotNil(t, p) + + switch pVal := p.(type) { + case *Entry: + assert.Equal(t, "kaboom", pVal.Message) + assert.Equal(t, errBoom, pVal.Data["err"]) + default: + t.Fatalf("want type *Entry, got %T: %#v", pVal, pVal) + } + }() + + logger := New() + logger.Out = &bytes.Buffer{} + entry := NewEntry(logger) + entry.WithField("err", errBoom).Panicln("kaboom") +} + +func TestEntryPanicf(t *testing.T) { + errBoom := fmt.Errorf("boom again") + + defer func() { + p := recover() + assert.NotNil(t, p) + + switch pVal := p.(type) { + case *Entry: + assert.Equal(t, "kaboom true", pVal.Message) + assert.Equal(t, errBoom, pVal.Data["err"]) + default: + t.Fatalf("want type *Entry, got %T: %#v", pVal, pVal) + } + }() + + logger := New() + logger.Out = &bytes.Buffer{} + entry := NewEntry(logger) + entry.WithField("err", errBoom).Panicf("kaboom %v", true) +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/examples/basic/basic.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/examples/basic/basic.go new file mode 100644 index 0000000000000..a1623ec003f5a --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/examples/basic/basic.go @@ -0,0 +1,50 @@ +package main + +import ( + "github.com/Sirupsen/logrus" +) + +var log = logrus.New() + +func init() { + log.Formatter = new(logrus.JSONFormatter) + log.Formatter = new(logrus.TextFormatter) // default + log.Level = logrus.DebugLevel +} + +func main() { + defer func() { + err := recover() + if err != nil { + log.WithFields(logrus.Fields{ + "omg": true, + "err": err, + "number": 100, + }).Fatal("The ice breaks!") + } + }() + + log.WithFields(logrus.Fields{ + "animal": "walrus", + "number": 8, + }).Debug("Started observing beach") + + log.WithFields(logrus.Fields{ + "animal": "walrus", + "size": 10, + }).Info("A group of walrus emerges from the ocean") + + log.WithFields(logrus.Fields{ + "omg": true, + "number": 122, + }).Warn("The group's number increased tremendously!") + + log.WithFields(logrus.Fields{ + "temperature": -4, + }).Debug("Temperature changes") + + log.WithFields(logrus.Fields{ + "animal": "orca", + "size": 9009, + }).Panic("It's over 9000!") +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/examples/hook/hook.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/examples/hook/hook.go new file mode 100644 index 0000000000000..cb5759a35cf52 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/examples/hook/hook.go @@ -0,0 +1,30 @@ +package main + +import ( + "github.com/Sirupsen/logrus" + "github.com/Sirupsen/logrus/hooks/airbrake" +) + +var log = logrus.New() + +func init() { + log.Formatter = new(logrus.TextFormatter) // default + log.Hooks.Add(airbrake.NewHook("https://example.com", "xyz", "development")) +} + +func main() { + log.WithFields(logrus.Fields{ + "animal": "walrus", + "size": 10, + }).Info("A group of walrus emerges from the ocean") + + log.WithFields(logrus.Fields{ + "omg": true, + "number": 122, + }).Warn("The group's number increased tremendously!") + + log.WithFields(logrus.Fields{ + "omg": true, + "number": 100, + }).Fatal("The ice breaks!") +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/exported.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/exported.go new file mode 100644 index 0000000000000..a67e1b802d914 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/exported.go @@ -0,0 +1,188 @@ +package logrus + +import ( + "io" +) + +var ( + // std is the name of the standard logger in stdlib `log` + std = New() +) + +func StandardLogger() *Logger { + return std +} + +// SetOutput sets the standard logger output. +func SetOutput(out io.Writer) { + std.mu.Lock() + defer std.mu.Unlock() + std.Out = out +} + +// SetFormatter sets the standard logger formatter. +func SetFormatter(formatter Formatter) { + std.mu.Lock() + defer std.mu.Unlock() + std.Formatter = formatter +} + +// SetLevel sets the standard logger level. +func SetLevel(level Level) { + std.mu.Lock() + defer std.mu.Unlock() + std.Level = level +} + +// GetLevel returns the standard logger level. +func GetLevel() Level { + std.mu.Lock() + defer std.mu.Unlock() + return std.Level +} + +// AddHook adds a hook to the standard logger hooks. +func AddHook(hook Hook) { + std.mu.Lock() + defer std.mu.Unlock() + std.Hooks.Add(hook) +} + +// WithField creates an entry from the standard logger and adds a field to +// it. If you want multiple fields, use `WithFields`. +// +// Note that it doesn't log until you call Debug, Print, Info, Warn, Fatal +// or Panic on the Entry it returns. +func WithField(key string, value interface{}) *Entry { + return std.WithField(key, value) +} + +// WithFields creates an entry from the standard logger and adds multiple +// fields to it. This is simply a helper for `WithField`, invoking it +// once for each field. +// +// Note that it doesn't log until you call Debug, Print, Info, Warn, Fatal +// or Panic on the Entry it returns. +func WithFields(fields Fields) *Entry { + return std.WithFields(fields) +} + +// Debug logs a message at level Debug on the standard logger. +func Debug(args ...interface{}) { + std.Debug(args...) +} + +// Print logs a message at level Info on the standard logger. +func Print(args ...interface{}) { + std.Print(args...) +} + +// Info logs a message at level Info on the standard logger. +func Info(args ...interface{}) { + std.Info(args...) +} + +// Warn logs a message at level Warn on the standard logger. +func Warn(args ...interface{}) { + std.Warn(args...) +} + +// Warning logs a message at level Warn on the standard logger. +func Warning(args ...interface{}) { + std.Warning(args...) +} + +// Error logs a message at level Error on the standard logger. +func Error(args ...interface{}) { + std.Error(args...) +} + +// Panic logs a message at level Panic on the standard logger. +func Panic(args ...interface{}) { + std.Panic(args...) +} + +// Fatal logs a message at level Fatal on the standard logger. +func Fatal(args ...interface{}) { + std.Fatal(args...) +} + +// Debugf logs a message at level Debug on the standard logger. +func Debugf(format string, args ...interface{}) { + std.Debugf(format, args...) +} + +// Printf logs a message at level Info on the standard logger. +func Printf(format string, args ...interface{}) { + std.Printf(format, args...) +} + +// Infof logs a message at level Info on the standard logger. +func Infof(format string, args ...interface{}) { + std.Infof(format, args...) +} + +// Warnf logs a message at level Warn on the standard logger. +func Warnf(format string, args ...interface{}) { + std.Warnf(format, args...) +} + +// Warningf logs a message at level Warn on the standard logger. +func Warningf(format string, args ...interface{}) { + std.Warningf(format, args...) +} + +// Errorf logs a message at level Error on the standard logger. +func Errorf(format string, args ...interface{}) { + std.Errorf(format, args...) +} + +// Panicf logs a message at level Panic on the standard logger. +func Panicf(format string, args ...interface{}) { + std.Panicf(format, args...) +} + +// Fatalf logs a message at level Fatal on the standard logger. +func Fatalf(format string, args ...interface{}) { + std.Fatalf(format, args...) +} + +// Debugln logs a message at level Debug on the standard logger. +func Debugln(args ...interface{}) { + std.Debugln(args...) +} + +// Println logs a message at level Info on the standard logger. +func Println(args ...interface{}) { + std.Println(args...) +} + +// Infoln logs a message at level Info on the standard logger. +func Infoln(args ...interface{}) { + std.Infoln(args...) +} + +// Warnln logs a message at level Warn on the standard logger. +func Warnln(args ...interface{}) { + std.Warnln(args...) +} + +// Warningln logs a message at level Warn on the standard logger. +func Warningln(args ...interface{}) { + std.Warningln(args...) +} + +// Errorln logs a message at level Error on the standard logger. +func Errorln(args ...interface{}) { + std.Errorln(args...) +} + +// Panicln logs a message at level Panic on the standard logger. +func Panicln(args ...interface{}) { + std.Panicln(args...) +} + +// Fatalln logs a message at level Fatal on the standard logger. +func Fatalln(args ...interface{}) { + std.Fatalln(args...) +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/formatter.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/formatter.go new file mode 100644 index 0000000000000..104d689f187eb --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/formatter.go @@ -0,0 +1,48 @@ +package logrus + +import "time" + +const DefaultTimestampFormat = time.RFC3339 + +// The Formatter interface is used to implement a custom Formatter. It takes an +// `Entry`. It exposes all the fields, including the default ones: +// +// * `entry.Data["msg"]`. The message passed from Info, Warn, Error .. +// * `entry.Data["time"]`. The timestamp. +// * `entry.Data["level"]. The level the entry was logged at. +// +// Any additional fields added with `WithField` or `WithFields` are also in +// `entry.Data`. Format is expected to return an array of bytes which are then +// logged to `logger.Out`. +type Formatter interface { + Format(*Entry) ([]byte, error) +} + +// This is to not silently overwrite `time`, `msg` and `level` fields when +// dumping it. If this code wasn't there doing: +// +// logrus.WithField("level", 1).Info("hello") +// +// Would just silently drop the user provided level. Instead with this code +// it'll logged as: +// +// {"level": "info", "fields.level": 1, "msg": "hello", "time": "..."} +// +// It's not exported because it's still using Data in an opinionated way. It's to +// avoid code duplication between the two default formatters. +func prefixFieldClashes(data Fields) { + _, ok := data["time"] + if ok { + data["fields.time"] = data["time"] + } + + _, ok = data["msg"] + if ok { + data["fields.msg"] = data["msg"] + } + + _, ok = data["level"] + if ok { + data["fields.level"] = data["level"] + } +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/formatter_bench_test.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/formatter_bench_test.go new file mode 100644 index 0000000000000..c6d290c77f09b --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/formatter_bench_test.go @@ -0,0 +1,98 @@ +package logrus + +import ( + "fmt" + "testing" + "time" +) + +// smallFields is a small size data set for benchmarking +var smallFields = Fields{ + "foo": "bar", + "baz": "qux", + "one": "two", + "three": "four", +} + +// largeFields is a large size data set for benchmarking +var largeFields = Fields{ + "foo": "bar", + "baz": "qux", + "one": "two", + "three": "four", + "five": "six", + "seven": "eight", + "nine": "ten", + "eleven": "twelve", + "thirteen": "fourteen", + "fifteen": "sixteen", + "seventeen": "eighteen", + "nineteen": "twenty", + "a": "b", + "c": "d", + "e": "f", + "g": "h", + "i": "j", + "k": "l", + "m": "n", + "o": "p", + "q": "r", + "s": "t", + "u": "v", + "w": "x", + "y": "z", + "this": "will", + "make": "thirty", + "entries": "yeah", +} + +var errorFields = Fields{ + "foo": fmt.Errorf("bar"), + "baz": fmt.Errorf("qux"), +} + +func BenchmarkErrorTextFormatter(b *testing.B) { + doBenchmark(b, &TextFormatter{DisableColors: true}, errorFields) +} + +func BenchmarkSmallTextFormatter(b *testing.B) { + doBenchmark(b, &TextFormatter{DisableColors: true}, smallFields) +} + +func BenchmarkLargeTextFormatter(b *testing.B) { + doBenchmark(b, &TextFormatter{DisableColors: true}, largeFields) +} + +func BenchmarkSmallColoredTextFormatter(b *testing.B) { + doBenchmark(b, &TextFormatter{ForceColors: true}, smallFields) +} + +func BenchmarkLargeColoredTextFormatter(b *testing.B) { + doBenchmark(b, &TextFormatter{ForceColors: true}, largeFields) +} + +func BenchmarkSmallJSONFormatter(b *testing.B) { + doBenchmark(b, &JSONFormatter{}, smallFields) +} + +func BenchmarkLargeJSONFormatter(b *testing.B) { + doBenchmark(b, &JSONFormatter{}, largeFields) +} + +func doBenchmark(b *testing.B, formatter Formatter, fields Fields) { + entry := &Entry{ + Time: time.Time{}, + Level: InfoLevel, + Message: "message", + Data: fields, + } + var d []byte + var err error + for i := 0; i < b.N; i++ { + d, err = formatter.Format(entry) + if err != nil { + b.Fatal(err) + } + b.SetBytes(int64(len(d))) + } +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/formatters/logstash/logstash.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/formatters/logstash/logstash.go new file mode 100644 index 0000000000000..8ea93ddf20a03 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/formatters/logstash/logstash.go @@ -0,0 +1,56 @@ +package logstash + +import ( + "encoding/json" + "fmt" + + "github.com/Sirupsen/logrus" +) + +// Formatter generates json in logstash format. +// Logstash site: http://logstash.net/ +type LogstashFormatter struct { + Type string // if not empty use for logstash type field. + + // TimestampFormat sets the format used for timestamps. + TimestampFormat string +} + +func (f *LogstashFormatter) Format(entry *logrus.Entry) ([]byte, error) { + entry.Data["@version"] = 1 + + if f.TimestampFormat == "" { + f.TimestampFormat = logrus.DefaultTimestampFormat + } + + entry.Data["@timestamp"] = entry.Time.Format(f.TimestampFormat) + + // set message field + v, ok := entry.Data["message"] + if ok { + entry.Data["fields.message"] = v + } + entry.Data["message"] = entry.Message + + // set level field + v, ok = entry.Data["level"] + if ok { + entry.Data["fields.level"] = v + } + entry.Data["level"] = entry.Level.String() + + // set type field + if f.Type != "" { + v, ok = entry.Data["type"] + if ok { + entry.Data["fields.type"] = v + } + entry.Data["type"] = f.Type + } + + serialized, err := json.Marshal(entry.Data) + if err != nil { + return nil, fmt.Errorf("Failed to marshal fields to JSON, %v", err) + } + return append(serialized, '\n'), nil +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/formatters/logstash/logstash_test.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/formatters/logstash/logstash_test.go new file mode 100644 index 0000000000000..d8814a0eae438 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/formatters/logstash/logstash_test.go @@ -0,0 +1,52 @@ +package logstash + +import ( + "bytes" + "encoding/json" + "github.com/Sirupsen/logrus" + "github.com/stretchr/testify/assert" + "testing" +) + +func TestLogstashFormatter(t *testing.T) { + assert := assert.New(t) + + lf := LogstashFormatter{Type: "abc"} + + fields := logrus.Fields{ + "message": "def", + "level": "ijk", + "type": "lmn", + "one": 1, + "pi": 3.14, + "bool": true, + } + + entry := logrus.WithFields(fields) + entry.Message = "msg" + entry.Level = logrus.InfoLevel + + b, _ := lf.Format(entry) + + var data map[string]interface{} + dec := json.NewDecoder(bytes.NewReader(b)) + dec.UseNumber() + dec.Decode(&data) + + // base fields + assert.Equal(json.Number("1"), data["@version"]) + assert.NotEmpty(data["@timestamp"]) + assert.Equal("abc", data["type"]) + assert.Equal("msg", data["message"]) + assert.Equal("info", data["level"]) + + // substituted fields + assert.Equal("def", data["fields.message"]) + assert.Equal("ijk", data["fields.level"]) + assert.Equal("lmn", data["fields.type"]) + + // formats + assert.Equal(json.Number("1"), data["one"]) + assert.Equal(json.Number("3.14"), data["pi"]) + assert.Equal(true, data["bool"]) +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hook_test.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hook_test.go new file mode 100644 index 0000000000000..13f34cb6f8150 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hook_test.go @@ -0,0 +1,122 @@ +package logrus + +import ( + "testing" + + "github.com/stretchr/testify/assert" +) + +type TestHook struct { + Fired bool +} + +func (hook *TestHook) Fire(entry *Entry) error { + hook.Fired = true + return nil +} + +func (hook *TestHook) Levels() []Level { + return []Level{ + DebugLevel, + InfoLevel, + WarnLevel, + ErrorLevel, + FatalLevel, + PanicLevel, + } +} + +func TestHookFires(t *testing.T) { + hook := new(TestHook) + + LogAndAssertJSON(t, func(log *Logger) { + log.Hooks.Add(hook) + assert.Equal(t, hook.Fired, false) + + log.Print("test") + }, func(fields Fields) { + assert.Equal(t, hook.Fired, true) + }) +} + +type ModifyHook struct { +} + +func (hook *ModifyHook) Fire(entry *Entry) error { + entry.Data["wow"] = "whale" + return nil +} + +func (hook *ModifyHook) Levels() []Level { + return []Level{ + DebugLevel, + InfoLevel, + WarnLevel, + ErrorLevel, + FatalLevel, + PanicLevel, + } +} + +func TestHookCanModifyEntry(t *testing.T) { + hook := new(ModifyHook) + + LogAndAssertJSON(t, func(log *Logger) { + log.Hooks.Add(hook) + log.WithField("wow", "elephant").Print("test") + }, func(fields Fields) { + assert.Equal(t, fields["wow"], "whale") + }) +} + +func TestCanFireMultipleHooks(t *testing.T) { + hook1 := new(ModifyHook) + hook2 := new(TestHook) + + LogAndAssertJSON(t, func(log *Logger) { + log.Hooks.Add(hook1) + log.Hooks.Add(hook2) + + log.WithField("wow", "elephant").Print("test") + }, func(fields Fields) { + assert.Equal(t, fields["wow"], "whale") + assert.Equal(t, hook2.Fired, true) + }) +} + +type ErrorHook struct { + Fired bool +} + +func (hook *ErrorHook) Fire(entry *Entry) error { + hook.Fired = true + return nil +} + +func (hook *ErrorHook) Levels() []Level { + return []Level{ + ErrorLevel, + } +} + +func TestErrorHookShouldntFireOnInfo(t *testing.T) { + hook := new(ErrorHook) + + LogAndAssertJSON(t, func(log *Logger) { + log.Hooks.Add(hook) + log.Info("test") + }, func(fields Fields) { + assert.Equal(t, hook.Fired, false) + }) +} + +func TestErrorHookShouldFireOnError(t *testing.T) { + hook := new(ErrorHook) + + LogAndAssertJSON(t, func(log *Logger) { + log.Hooks.Add(hook) + log.Error("test") + }, func(fields Fields) { + assert.Equal(t, hook.Fired, true) + }) +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks.go new file mode 100644 index 0000000000000..3f151cdc39275 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks.go @@ -0,0 +1,34 @@ +package logrus + +// A hook to be fired when logging on the logging levels returned from +// `Levels()` on your implementation of the interface. Note that this is not +// fired in a goroutine or a channel with workers, you should handle such +// functionality yourself if your call is non-blocking and you don't wish for +// the logging calls for levels returned from `Levels()` to block. +type Hook interface { + Levels() []Level + Fire(*Entry) error +} + +// Internal type for storing the hooks on a logger instance. +type LevelHooks map[Level][]Hook + +// Add a hook to an instance of logger. This is called with +// `log.Hooks.Add(new(MyHook))` where `MyHook` implements the `Hook` interface. +func (hooks LevelHooks) Add(hook Hook) { + for _, level := range hook.Levels() { + hooks[level] = append(hooks[level], hook) + } +} + +// Fire all the hooks for the passed level. Used by `entry.log` to fire +// appropriate hooks for a log entry. +func (hooks LevelHooks) Fire(level Level, entry *Entry) error { + for _, hook := range hooks[level] { + if err := hook.Fire(entry); err != nil { + return err + } + } + + return nil +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/airbrake/airbrake.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/airbrake/airbrake.go new file mode 100644 index 0000000000000..b0502c335a96d --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/airbrake/airbrake.go @@ -0,0 +1,54 @@ +package airbrake + +import ( + "errors" + "fmt" + + "github.com/Sirupsen/logrus" + "github.com/tobi/airbrake-go" +) + +// AirbrakeHook to send exceptions to an exception-tracking service compatible +// with the Airbrake API. +type airbrakeHook struct { + APIKey string + Endpoint string + Environment string +} + +func NewHook(endpoint, apiKey, env string) *airbrakeHook { + return &airbrakeHook{ + APIKey: apiKey, + Endpoint: endpoint, + Environment: env, + } +} + +func (hook *airbrakeHook) Fire(entry *logrus.Entry) error { + airbrake.ApiKey = hook.APIKey + airbrake.Endpoint = hook.Endpoint + airbrake.Environment = hook.Environment + + var notifyErr error + err, ok := entry.Data["error"].(error) + if ok { + notifyErr = err + } else { + notifyErr = errors.New(entry.Message) + } + + airErr := airbrake.Notify(notifyErr) + if airErr != nil { + return fmt.Errorf("Failed to send error to Airbrake: %s", airErr) + } + + return nil +} + +func (hook *airbrakeHook) Levels() []logrus.Level { + return []logrus.Level{ + logrus.ErrorLevel, + logrus.FatalLevel, + logrus.PanicLevel, + } +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/airbrake/airbrake_test.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/airbrake/airbrake_test.go new file mode 100644 index 0000000000000..058a91e343820 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/airbrake/airbrake_test.go @@ -0,0 +1,133 @@ +package airbrake + +import ( + "encoding/xml" + "net/http" + "net/http/httptest" + "testing" + "time" + + "github.com/Sirupsen/logrus" +) + +type notice struct { + Error NoticeError `xml:"error"` +} +type NoticeError struct { + Class string `xml:"class"` + Message string `xml:"message"` +} + +type customErr struct { + msg string +} + +func (e *customErr) Error() string { + return e.msg +} + +const ( + testAPIKey = "abcxyz" + testEnv = "development" + expectedClass = "*airbrake.customErr" + expectedMsg = "foo" + unintendedMsg = "Airbrake will not see this string" +) + +var ( + noticeError = make(chan NoticeError, 1) +) + +// TestLogEntryMessageReceived checks if invoking Logrus' log.Error +// method causes an XML payload containing the log entry message is received +// by a HTTP server emulating an Airbrake-compatible endpoint. +func TestLogEntryMessageReceived(t *testing.T) { + log := logrus.New() + ts := startAirbrakeServer(t) + defer ts.Close() + + hook := NewHook(ts.URL, testAPIKey, "production") + log.Hooks.Add(hook) + + log.Error(expectedMsg) + + select { + case received := <-noticeError: + if received.Message != expectedMsg { + t.Errorf("Unexpected message received: %s", received.Message) + } + case <-time.After(time.Second): + t.Error("Timed out; no notice received by Airbrake API") + } +} + +// TestLogEntryMessageReceived confirms that, when passing an error type using +// logrus.Fields, a HTTP server emulating an Airbrake endpoint receives the +// error message returned by the Error() method on the error interface +// rather than the logrus.Entry.Message string. +func TestLogEntryWithErrorReceived(t *testing.T) { + log := logrus.New() + ts := startAirbrakeServer(t) + defer ts.Close() + + hook := NewHook(ts.URL, testAPIKey, "production") + log.Hooks.Add(hook) + + log.WithFields(logrus.Fields{ + "error": &customErr{expectedMsg}, + }).Error(unintendedMsg) + + select { + case received := <-noticeError: + if received.Message != expectedMsg { + t.Errorf("Unexpected message received: %s", received.Message) + } + if received.Class != expectedClass { + t.Errorf("Unexpected error class: %s", received.Class) + } + case <-time.After(time.Second): + t.Error("Timed out; no notice received by Airbrake API") + } +} + +// TestLogEntryWithNonErrorTypeNotReceived confirms that, when passing a +// non-error type using logrus.Fields, a HTTP server emulating an Airbrake +// endpoint receives the logrus.Entry.Message string. +// +// Only error types are supported when setting the 'error' field using +// logrus.WithFields(). +func TestLogEntryWithNonErrorTypeNotReceived(t *testing.T) { + log := logrus.New() + ts := startAirbrakeServer(t) + defer ts.Close() + + hook := NewHook(ts.URL, testAPIKey, "production") + log.Hooks.Add(hook) + + log.WithFields(logrus.Fields{ + "error": expectedMsg, + }).Error(unintendedMsg) + + select { + case received := <-noticeError: + if received.Message != unintendedMsg { + t.Errorf("Unexpected message received: %s", received.Message) + } + case <-time.After(time.Second): + t.Error("Timed out; no notice received by Airbrake API") + } +} + +func startAirbrakeServer(t *testing.T) *httptest.Server { + ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + var notice notice + if err := xml.NewDecoder(r.Body).Decode(¬ice); err != nil { + t.Error(err) + } + r.Body.Close() + + noticeError <- notice.Error + })) + + return ts +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/bugsnag/bugsnag.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/bugsnag/bugsnag.go new file mode 100644 index 0000000000000..d20a0f54ab770 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/bugsnag/bugsnag.go @@ -0,0 +1,68 @@ +package logrus_bugsnag + +import ( + "errors" + + "github.com/Sirupsen/logrus" + "github.com/bugsnag/bugsnag-go" +) + +type bugsnagHook struct{} + +// ErrBugsnagUnconfigured is returned if NewBugsnagHook is called before +// bugsnag.Configure. Bugsnag must be configured before the hook. +var ErrBugsnagUnconfigured = errors.New("bugsnag must be configured before installing this logrus hook") + +// ErrBugsnagSendFailed indicates that the hook failed to submit an error to +// bugsnag. The error was successfully generated, but `bugsnag.Notify()` +// failed. +type ErrBugsnagSendFailed struct { + err error +} + +func (e ErrBugsnagSendFailed) Error() string { + return "failed to send error to Bugsnag: " + e.err.Error() +} + +// NewBugsnagHook initializes a logrus hook which sends exceptions to an +// exception-tracking service compatible with the Bugsnag API. Before using +// this hook, you must call bugsnag.Configure(). The returned object should be +// registered with a log via `AddHook()` +// +// Entries that trigger an Error, Fatal or Panic should now include an "error" +// field to send to Bugsnag. +func NewBugsnagHook() (*bugsnagHook, error) { + if bugsnag.Config.APIKey == "" { + return nil, ErrBugsnagUnconfigured + } + return &bugsnagHook{}, nil +} + +// Fire forwards an error to Bugsnag. Given a logrus.Entry, it extracts the +// "error" field (or the Message if the error isn't present) and sends it off. +func (hook *bugsnagHook) Fire(entry *logrus.Entry) error { + var notifyErr error + err, ok := entry.Data["error"].(error) + if ok { + notifyErr = err + } else { + notifyErr = errors.New(entry.Message) + } + + bugsnagErr := bugsnag.Notify(notifyErr) + if bugsnagErr != nil { + return ErrBugsnagSendFailed{bugsnagErr} + } + + return nil +} + +// Levels enumerates the log levels on which the error should be forwarded to +// bugsnag: everything at or above the "Error" level. +func (hook *bugsnagHook) Levels() []logrus.Level { + return []logrus.Level{ + logrus.ErrorLevel, + logrus.FatalLevel, + logrus.PanicLevel, + } +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/bugsnag/bugsnag_test.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/bugsnag/bugsnag_test.go new file mode 100644 index 0000000000000..e9ea298d89b43 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/bugsnag/bugsnag_test.go @@ -0,0 +1,64 @@ +package logrus_bugsnag + +import ( + "encoding/json" + "errors" + "io/ioutil" + "net/http" + "net/http/httptest" + "testing" + "time" + + "github.com/Sirupsen/logrus" + "github.com/bugsnag/bugsnag-go" +) + +type notice struct { + Events []struct { + Exceptions []struct { + Message string `json:"message"` + } `json:"exceptions"` + } `json:"events"` +} + +func TestNoticeReceived(t *testing.T) { + msg := make(chan string, 1) + expectedMsg := "foo" + + ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + var notice notice + data, _ := ioutil.ReadAll(r.Body) + if err := json.Unmarshal(data, ¬ice); err != nil { + t.Error(err) + } + _ = r.Body.Close() + + msg <- notice.Events[0].Exceptions[0].Message + })) + defer ts.Close() + + hook := &bugsnagHook{} + + bugsnag.Configure(bugsnag.Configuration{ + Endpoint: ts.URL, + ReleaseStage: "production", + APIKey: "12345678901234567890123456789012", + Synchronous: true, + }) + + log := logrus.New() + log.Hooks.Add(hook) + + log.WithFields(logrus.Fields{ + "error": errors.New(expectedMsg), + }).Error("Bugsnag will not see this string") + + select { + case received := <-msg: + if received != expectedMsg { + t.Errorf("Unexpected message received: %s", received) + } + case <-time.After(time.Second): + t.Error("Timed out; no notice received by Bugsnag API") + } +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/papertrail/README.md b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/papertrail/README.md new file mode 100644 index 0000000000000..ae61e9229ab72 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/papertrail/README.md @@ -0,0 +1,28 @@ +# Papertrail Hook for Logrus :walrus: + +[Papertrail](https://papertrailapp.com) provides hosted log management. Once stored in Papertrail, you can [group](http://help.papertrailapp.com/kb/how-it-works/groups/) your logs on various dimensions, [search](http://help.papertrailapp.com/kb/how-it-works/search-syntax) them, and trigger [alerts](http://help.papertrailapp.com/kb/how-it-works/alerts). + +In most deployments, you'll want to send logs to Papertrail via their [remote_syslog](http://help.papertrailapp.com/kb/configuration/configuring-centralized-logging-from-text-log-files-in-unix/) daemon, which requires no application-specific configuration. This hook is intended for relatively low-volume logging, likely in managed cloud hosting deployments where installing `remote_syslog` is not possible. + +## Usage + +You can find your Papertrail UDP port on your [Papertrail account page](https://papertrailapp.com/account/destinations). Substitute it below for `YOUR_PAPERTRAIL_UDP_PORT`. + +For `YOUR_APP_NAME`, substitute a short string that will readily identify your application or service in the logs. + +```go +import ( + "log/syslog" + "github.com/Sirupsen/logrus" + "github.com/Sirupsen/logrus/hooks/papertrail" +) + +func main() { + log := logrus.New() + hook, err := logrus_papertrail.NewPapertrailHook("logs.papertrailapp.com", YOUR_PAPERTRAIL_UDP_PORT, YOUR_APP_NAME) + + if err == nil { + log.Hooks.Add(hook) + } +} +``` diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/papertrail/papertrail.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/papertrail/papertrail.go new file mode 100644 index 0000000000000..c0f10c1bda23f --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/papertrail/papertrail.go @@ -0,0 +1,55 @@ +package logrus_papertrail + +import ( + "fmt" + "net" + "os" + "time" + + "github.com/Sirupsen/logrus" +) + +const ( + format = "Jan 2 15:04:05" +) + +// PapertrailHook to send logs to a logging service compatible with the Papertrail API. +type PapertrailHook struct { + Host string + Port int + AppName string + UDPConn net.Conn +} + +// NewPapertrailHook creates a hook to be added to an instance of logger. +func NewPapertrailHook(host string, port int, appName string) (*PapertrailHook, error) { + conn, err := net.Dial("udp", fmt.Sprintf("%s:%d", host, port)) + return &PapertrailHook{host, port, appName, conn}, err +} + +// Fire is called when a log event is fired. +func (hook *PapertrailHook) Fire(entry *logrus.Entry) error { + date := time.Now().Format(format) + msg, _ := entry.String() + payload := fmt.Sprintf("<22> %s %s: %s", date, hook.AppName, msg) + + bytesWritten, err := hook.UDPConn.Write([]byte(payload)) + if err != nil { + fmt.Fprintf(os.Stderr, "Unable to send log line to Papertrail via UDP. Wrote %d bytes before error: %v", bytesWritten, err) + return err + } + + return nil +} + +// Levels returns the available logging levels. +func (hook *PapertrailHook) Levels() []logrus.Level { + return []logrus.Level{ + logrus.PanicLevel, + logrus.FatalLevel, + logrus.ErrorLevel, + logrus.WarnLevel, + logrus.InfoLevel, + logrus.DebugLevel, + } +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/papertrail/papertrail_test.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/papertrail/papertrail_test.go new file mode 100644 index 0000000000000..96318d0030491 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/papertrail/papertrail_test.go @@ -0,0 +1,26 @@ +package logrus_papertrail + +import ( + "fmt" + "testing" + + "github.com/Sirupsen/logrus" + "github.com/stvp/go-udp-testing" +) + +func TestWritingToUDP(t *testing.T) { + port := 16661 + udp.SetAddr(fmt.Sprintf(":%d", port)) + + hook, err := NewPapertrailHook("localhost", port, "test") + if err != nil { + t.Errorf("Unable to connect to local UDP server.") + } + + log := logrus.New() + log.Hooks.Add(hook) + + udp.ShouldReceive(t, "foo", func() { + log.Info("foo") + }) +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/sentry/README.md b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/sentry/README.md new file mode 100644 index 0000000000000..31de6540adbf9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/sentry/README.md @@ -0,0 +1,111 @@ +# Sentry Hook for Logrus :walrus: + +[Sentry](https://getsentry.com) provides both self-hosted and hosted +solutions for exception tracking. +Both client and server are +[open source](https://github.com/getsentry/sentry). + +## Usage + +Every sentry application defined on the server gets a different +[DSN](https://www.getsentry.com/docs/). In the example below replace +`YOUR_DSN` with the one created for your application. + +```go +import ( + "github.com/Sirupsen/logrus" + "github.com/Sirupsen/logrus/hooks/sentry" +) + +func main() { + log := logrus.New() + hook, err := logrus_sentry.NewSentryHook(YOUR_DSN, []logrus.Level{ + logrus.PanicLevel, + logrus.FatalLevel, + logrus.ErrorLevel, + }) + + if err == nil { + log.Hooks.Add(hook) + } +} +``` + +If you wish to initialize a SentryHook with tags, you can use the `NewWithTagsSentryHook` constructor to provide default tags: + +```go +tags := map[string]string{ + "site": "example.com", +} +levels := []logrus.Level{ + logrus.PanicLevel, + logrus.FatalLevel, + logrus.ErrorLevel, +} +hook, err := logrus_sentry.NewWithTagsSentryHook(YOUR_DSN, tags, levels) + +``` + +If you wish to initialize a SentryHook with an already initialized raven client, you can use +the `NewWithClientSentryHook` constructor: + +```go +import ( + "github.com/Sirupsen/logrus" + "github.com/Sirupsen/logrus/hooks/sentry" + "github.com/getsentry/raven-go" +) + +func main() { + log := logrus.New() + + client, err := raven.New(YOUR_DSN) + if err != nil { + log.Fatal(err) + } + + hook, err := logrus_sentry.NewWithClientSentryHook(client, []logrus.Level{ + logrus.PanicLevel, + logrus.FatalLevel, + logrus.ErrorLevel, + }) + + if err == nil { + log.Hooks.Add(hook) + } +} + +hook, err := NewWithClientSentryHook(client, []logrus.Level{ + logrus.ErrorLevel, +}) +``` + +## Special fields + +Some logrus fields have a special meaning in this hook, +these are `server_name`, `logger` and `http_request`. +When logs are sent to sentry these fields are treated differently. +- `server_name` (also known as hostname) is the name of the server which +is logging the event (hostname.example.com) +- `logger` is the part of the application which is logging the event. +In go this usually means setting it to the name of the package. +- `http_request` is the in-coming request(*http.Request). The detailed request data are sent to Sentry. + +## Timeout + +`Timeout` is the time the sentry hook will wait for a response +from the sentry server. + +If this time elapses with no response from +the server an error will be returned. + +If `Timeout` is set to 0 the SentryHook will not wait for a reply +and will assume a correct delivery. + +The SentryHook has a default timeout of `100 milliseconds` when created +with a call to `NewSentryHook`. This can be changed by assigning a value to the `Timeout` field: + +```go +hook, _ := logrus_sentry.NewSentryHook(...) +hook.Timeout = 20*time.Second +``` diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/sentry/sentry.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/sentry/sentry.go new file mode 100644 index 0000000000000..cf88098a8892f --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/sentry/sentry.go @@ -0,0 +1,137 @@ +package logrus_sentry + +import ( + "fmt" + "net/http" + "time" + + "github.com/Sirupsen/logrus" + "github.com/getsentry/raven-go" +) + +var ( + severityMap = map[logrus.Level]raven.Severity{ + logrus.DebugLevel: raven.DEBUG, + logrus.InfoLevel: raven.INFO, + logrus.WarnLevel: raven.WARNING, + logrus.ErrorLevel: raven.ERROR, + logrus.FatalLevel: raven.FATAL, + logrus.PanicLevel: raven.FATAL, + } +) + +func getAndDel(d logrus.Fields, key string) (string, bool) { + var ( + ok bool + v interface{} + val string + ) + if v, ok = d[key]; !ok { + return "", false + } + + if val, ok = v.(string); !ok { + return "", false + } + delete(d, key) + return val, true +} + +func getAndDelRequest(d logrus.Fields, key string) (*http.Request, bool) { + var ( + ok bool + v interface{} + req *http.Request + ) + if v, ok = d[key]; !ok { + return nil, false + } + if req, ok = v.(*http.Request); !ok || req == nil { + return nil, false + } + delete(d, key) + return req, true +} + +// SentryHook delivers logs to a sentry server. +type SentryHook struct { + // Timeout sets the time to wait for a delivery error from the sentry server. + // If this is set to zero the server will not wait for any response and will + // consider the message correctly sent + Timeout time.Duration + + client *raven.Client + levels []logrus.Level +} + +// NewSentryHook creates a hook to be added to an instance of logger +// and initializes the raven client. +// This method sets the timeout to 100 milliseconds. +func NewSentryHook(DSN string, levels []logrus.Level) (*SentryHook, error) { + client, err := raven.New(DSN) + if err != nil { + return nil, err + } + return &SentryHook{100 * time.Millisecond, client, levels}, nil +} + +// NewWithTagsSentryHook creates a hook with tags to be added to an instance +// of logger and initializes the raven client. This method sets the timeout to +// 100 milliseconds. +func NewWithTagsSentryHook(DSN string, tags map[string]string, levels []logrus.Level) (*SentryHook, error) { + client, err := raven.NewWithTags(DSN, tags) + if err != nil { + return nil, err + } + return &SentryHook{100 * time.Millisecond, client, levels}, nil +} + +// NewWithClientSentryHook creates a hook using an initialized raven client. +// This method sets the timeout to 100 milliseconds. +func NewWithClientSentryHook(client *raven.Client, levels []logrus.Level) (*SentryHook, error) { + return &SentryHook{100 * time.Millisecond, client, levels}, nil +} + +// Called when an event should be sent to sentry +// Special fields that sentry uses to give more information to the server +// are extracted from entry.Data (if they are found) +// These fields are: logger, server_name and http_request +func (hook *SentryHook) Fire(entry *logrus.Entry) error { + packet := &raven.Packet{ + Message: entry.Message, + Timestamp: raven.Timestamp(entry.Time), + Level: severityMap[entry.Level], + Platform: "go", + } + + d := entry.Data + + if logger, ok := getAndDel(d, "logger"); ok { + packet.Logger = logger + } + if serverName, ok := getAndDel(d, "server_name"); ok { + packet.ServerName = serverName + } + if req, ok := getAndDelRequest(d, "http_request"); ok { + packet.Interfaces = append(packet.Interfaces, raven.NewHttp(req)) + } + packet.Extra = map[string]interface{}(d) + + _, errCh := hook.client.Capture(packet, nil) + timeout := hook.Timeout + if timeout != 0 { + timeoutCh := time.After(timeout) + select { + case err := <-errCh: + return err + case <-timeoutCh: + return fmt.Errorf("no response from sentry server in %s", timeout) + } + } + return nil +} + +// Levels returns the available logging levels. +func (hook *SentryHook) Levels() []logrus.Level { + return hook.levels +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/sentry/sentry_test.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/sentry/sentry_test.go new file mode 100644 index 0000000000000..4a97bc63e28c0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/sentry/sentry_test.go @@ -0,0 +1,154 @@ +package logrus_sentry + +import ( + "encoding/json" + "fmt" + "io/ioutil" + "net/http" + "net/http/httptest" + "reflect" + "strings" + "testing" + + "github.com/Sirupsen/logrus" + "github.com/getsentry/raven-go" +) + +const ( + message = "error message" + server_name = "testserver.internal" + logger_name = "test.logger" +) + +func getTestLogger() *logrus.Logger { + l := logrus.New() + l.Out = ioutil.Discard + return l +} + +func WithTestDSN(t *testing.T, tf func(string, <-chan *raven.Packet)) { + pch := make(chan *raven.Packet, 1) + s := httptest.NewServer(http.HandlerFunc(func(rw http.ResponseWriter, req *http.Request) { + defer req.Body.Close() + d := json.NewDecoder(req.Body) + p := &raven.Packet{} + err := d.Decode(p) + if err != nil { + t.Fatal(err.Error()) + } + + pch <- p + })) + defer s.Close() + + fragments := strings.SplitN(s.URL, "://", 2) + dsn := fmt.Sprintf( + "%s://public:secret@%s/sentry/project-id", + fragments[0], + fragments[1], + ) + tf(dsn, pch) +} + +func TestSpecialFields(t *testing.T) { + WithTestDSN(t, func(dsn string, pch <-chan *raven.Packet) { + logger := getTestLogger() + + hook, err := NewSentryHook(dsn, []logrus.Level{ + logrus.ErrorLevel, + }) + + if err != nil { + t.Fatal(err.Error()) + } + logger.Hooks.Add(hook) + + req, _ := http.NewRequest("GET", "url", nil) + logger.WithFields(logrus.Fields{ + "server_name": server_name, + "logger": logger_name, + "http_request": req, + }).Error(message) + + packet := <-pch + if packet.Logger != logger_name { + t.Errorf("logger should have been %s, was %s", logger_name, packet.Logger) + } + + if packet.ServerName != server_name { + t.Errorf("server_name should have been %s, was %s", server_name, packet.ServerName) + } + }) +} + +func TestSentryHandler(t *testing.T) { + WithTestDSN(t, func(dsn string, pch <-chan *raven.Packet) { + logger := getTestLogger() + hook, err := NewSentryHook(dsn, []logrus.Level{ + logrus.ErrorLevel, + }) + if err != nil { + t.Fatal(err.Error()) + } + logger.Hooks.Add(hook) + + logger.Error(message) + packet := <-pch + if packet.Message != message { + t.Errorf("message should have been %s, was %s", message, packet.Message) + } + }) +} + +func TestSentryWithClient(t *testing.T) { + WithTestDSN(t, func(dsn string, pch <-chan *raven.Packet) { + logger := getTestLogger() + + client, _ := raven.New(dsn) + + hook, err := NewWithClientSentryHook(client, []logrus.Level{ + logrus.ErrorLevel, + }) + if err != nil { + t.Fatal(err.Error()) + } + logger.Hooks.Add(hook) + + logger.Error(message) + packet := <-pch + if packet.Message != message { + t.Errorf("message should have been %s, was %s", message, packet.Message) + } + }) +} + +func TestSentryTags(t *testing.T) { + WithTestDSN(t, func(dsn string, pch <-chan *raven.Packet) { + logger := getTestLogger() + tags := map[string]string{ + "site": "test", + } + levels := []logrus.Level{ + logrus.ErrorLevel, + } + + hook, err := NewWithTagsSentryHook(dsn, tags, levels) + if err != nil { + t.Fatal(err.Error()) + } + + logger.Hooks.Add(hook) + + logger.Error(message) + packet := <-pch + expected := raven.Tags{ + raven.Tag{ + Key: "site", + Value: "test", + }, + } + if !reflect.DeepEqual(packet.Tags, expected) { + t.Errorf("message should have been %s, was %s", message, packet.Message) + } + }) +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/syslog/README.md b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/syslog/README.md new file mode 100644 index 0000000000000..4dbb8e729078e --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/syslog/README.md @@ -0,0 +1,20 @@ +# Syslog Hooks for Logrus :walrus: + +## Usage + +```go +import ( + "log/syslog" + "github.com/Sirupsen/logrus" + logrus_syslog "github.com/Sirupsen/logrus/hooks/syslog" +) + +func main() { + log := logrus.New() + hook, err := logrus_syslog.NewSyslogHook("udp", "localhost:514", syslog.LOG_INFO, "") + + if err == nil { + log.Hooks.Add(hook) + } +} +``` diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/syslog/syslog.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/syslog/syslog.go new file mode 100644 index 0000000000000..b6fa3746280c6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/syslog/syslog.go @@ -0,0 +1,59 @@ +package logrus_syslog + +import ( + "fmt" + "github.com/Sirupsen/logrus" + "log/syslog" + "os" +) + +// SyslogHook to send logs via syslog. +type SyslogHook struct { + Writer *syslog.Writer + SyslogNetwork string + SyslogRaddr string +} + +// Creates a hook to be added to an instance of logger. This is called with +// `hook, err := NewSyslogHook("udp", "localhost:514", syslog.LOG_DEBUG, "")` +// `if err == nil { log.Hooks.Add(hook) }` +func NewSyslogHook(network, raddr string, priority syslog.Priority, tag string) (*SyslogHook, error) { + w, err := syslog.Dial(network, raddr, priority, tag) + return &SyslogHook{w, network, raddr}, err +} + +func (hook *SyslogHook) Fire(entry *logrus.Entry) error { + line, err := entry.String() + if err != nil { + fmt.Fprintf(os.Stderr, "Unable to read entry, %v", err) + return err + } + + switch entry.Level { + case logrus.PanicLevel: + return hook.Writer.Crit(line) + case logrus.FatalLevel: + return hook.Writer.Crit(line) + case logrus.ErrorLevel: + return hook.Writer.Err(line) + case logrus.WarnLevel: + return hook.Writer.Warning(line) + case logrus.InfoLevel: + return hook.Writer.Info(line) + case logrus.DebugLevel: + return hook.Writer.Debug(line) + default: + return nil + } +} + +func (hook *SyslogHook) Levels() []logrus.Level { + return []logrus.Level{ + logrus.PanicLevel, + logrus.FatalLevel, + logrus.ErrorLevel, + logrus.WarnLevel, + logrus.InfoLevel, + logrus.DebugLevel, + } +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/syslog/syslog_test.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/syslog/syslog_test.go new file mode 100644 index 0000000000000..42762dc10d715 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/hooks/syslog/syslog_test.go @@ -0,0 +1,26 @@ +package logrus_syslog + +import ( + "github.com/Sirupsen/logrus" + "log/syslog" + "testing" +) + +func TestLocalhostAddAndPrint(t *testing.T) { + log := logrus.New() + hook, err := NewSyslogHook("udp", "localhost:514", syslog.LOG_INFO, "") + + if err != nil { + t.Errorf("Unable to connect to local syslog.") + } + + log.Hooks.Add(hook) + + for _, level := range hook.Levels() { + if len(log.Hooks[level]) != 1 { + t.Errorf("SyslogHook was not added. The length of log.Hooks[%v]: %v", level, len(log.Hooks[level])) + } + } + + log.Info("Congratulations!") +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/json_formatter.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/json_formatter.go new file mode 100644 index 0000000000000..2ad6dc5cf4f04 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/json_formatter.go @@ -0,0 +1,41 @@ +package logrus + +import ( + "encoding/json" + "fmt" +) + +type JSONFormatter struct { + // TimestampFormat sets the format used for marshaling timestamps. + TimestampFormat string +} + +func (f *JSONFormatter) Format(entry *Entry) ([]byte, error) { + data := make(Fields, len(entry.Data)+3) + for k, v := range entry.Data { + switch v := v.(type) { + case error: + // Otherwise errors are ignored by `encoding/json` + // https://github.com/Sirupsen/logrus/issues/137 + data[k] = v.Error() + default: + data[k] = v + } + } + prefixFieldClashes(data) + + timestampFormat := f.TimestampFormat + if timestampFormat == "" { + timestampFormat = DefaultTimestampFormat + } + + data["time"] = entry.Time.Format(timestampFormat) + data["msg"] = entry.Message + data["level"] = entry.Level.String() + + serialized, err := json.Marshal(data) + if err != nil { + return nil, fmt.Errorf("Failed to marshal fields to JSON, %v", err) + } + return append(serialized, '\n'), nil +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/json_formatter_test.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/json_formatter_test.go new file mode 100644 index 0000000000000..1d70873254d7b --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/json_formatter_test.go @@ -0,0 +1,120 @@ +package logrus + +import ( + "encoding/json" + "errors" + + "testing" +) + +func TestErrorNotLost(t *testing.T) { + formatter := &JSONFormatter{} + + b, err := formatter.Format(WithField("error", errors.New("wild walrus"))) + if err != nil { + t.Fatal("Unable to format entry: ", err) + } + + entry := make(map[string]interface{}) + err = json.Unmarshal(b, &entry) + if err != nil { + t.Fatal("Unable to unmarshal formatted entry: ", err) + } + + if entry["error"] != "wild walrus" { + t.Fatal("Error field not set") + } +} + +func TestErrorNotLostOnFieldNotNamedError(t *testing.T) { + formatter := &JSONFormatter{} + + b, err := formatter.Format(WithField("omg", errors.New("wild walrus"))) + if err != nil { + t.Fatal("Unable to format entry: ", err) + } + + entry := make(map[string]interface{}) + err = json.Unmarshal(b, &entry) + if err != nil { + t.Fatal("Unable to unmarshal formatted entry: ", err) + } + + if entry["omg"] != "wild walrus" { + t.Fatal("Error field not set") + } +} + +func TestFieldClashWithTime(t *testing.T) { + formatter := &JSONFormatter{} + + b, err := formatter.Format(WithField("time", "right now!")) + if err != nil { + t.Fatal("Unable to format entry: ", err) + } + + entry := make(map[string]interface{}) + err = json.Unmarshal(b, &entry) + if err != nil { + t.Fatal("Unable to unmarshal formatted entry: ", err) + } + + if entry["fields.time"] != "right now!" { + t.Fatal("fields.time not set to original time field") + } + + if entry["time"] != "0001-01-01T00:00:00Z" { + t.Fatal("time field not set to current time, was: ", entry["time"]) + } +} + +func TestFieldClashWithMsg(t *testing.T) { + formatter := &JSONFormatter{} + + b, err := formatter.Format(WithField("msg", "something")) + if err != nil { + t.Fatal("Unable to format entry: ", err) + } + + entry := make(map[string]interface{}) + err = json.Unmarshal(b, &entry) + if err != nil { + t.Fatal("Unable to unmarshal formatted entry: ", err) + } + + if entry["fields.msg"] != "something" { + t.Fatal("fields.msg not set to original msg field") + } +} + +func TestFieldClashWithLevel(t *testing.T) { + formatter := &JSONFormatter{} + + b, err := formatter.Format(WithField("level", "something")) + if err != nil { + t.Fatal("Unable to format entry: ", err) + } + + entry := make(map[string]interface{}) + err = json.Unmarshal(b, &entry) + if err != nil { + t.Fatal("Unable to unmarshal formatted entry: ", err) + } + + if entry["fields.level"] != "something" { + t.Fatal("fields.level not set to original level field") + } +} + +func TestJSONEntryEndsWithNewline(t *testing.T) { + formatter := &JSONFormatter{} + + b, err := formatter.Format(WithField("level", "something")) + if err != nil { + t.Fatal("Unable to format entry: ", err) + } + + if b[len(b)-1] != '\n' { + t.Fatal("Expected JSON log entry to end with a newline") + } +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/logger.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/logger.go new file mode 100644 index 0000000000000..dd9975931502d --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/logger.go @@ -0,0 +1,206 @@ +package logrus + +import ( + "io" + "os" + "sync" +) + +type Logger struct { + // The logs are `io.Copy`'d to this in a mutex. It's common to set this to a + // file, or leave it default which is `os.Stderr`. You can also set this to + // something more adventorous, such as logging to Kafka. + Out io.Writer + // Hooks for the logger instance. These allow firing events based on logging + // levels and log entries. For example, to send errors to an error tracking + // service, log to StatsD or dump the core on fatal errors. + Hooks LevelHooks + // All log entries pass through the formatter before logged to Out. The + // included formatters are `TextFormatter` and `JSONFormatter` for which + // TextFormatter is the default. In development (when a TTY is attached) it + // logs with colors, but to a file it wouldn't. You can easily implement your + // own that implements the `Formatter` interface, see the `README` or included + // formatters for examples. + Formatter Formatter + // The logging level the logger should log at. This is typically (and defaults + // to) `logrus.Info`, which allows Info(), Warn(), Error() and Fatal() to be + // logged. `logrus.Debug` is useful in + Level Level + // Used to sync writing to the log. + mu sync.Mutex +} + +// Creates a new logger. Configuration should be set by changing `Formatter`, +// `Out` and `Hooks` directly on the default logger instance. You can also just +// instantiate your own: +// +// var log = &Logger{ +// Out: os.Stderr, +// Formatter: new(JSONFormatter), +// Hooks: make(LevelHooks), +// Level: logrus.DebugLevel, +// } +// +// It's recommended to make this a global instance called `log`. +func New() *Logger { + return &Logger{ + Out: os.Stderr, + Formatter: new(TextFormatter), + Hooks: make(LevelHooks), + Level: InfoLevel, + } +} + +// Adds a field to the log entry, note that you it doesn't log until you call +// Debug, Print, Info, Warn, Fatal or Panic. It only creates a log entry. +// Ff you want multiple fields, use `WithFields`. +func (logger *Logger) WithField(key string, value interface{}) *Entry { + return NewEntry(logger).WithField(key, value) +} + +// Adds a struct of fields to the log entry. All it does is call `WithField` for +// each `Field`. +func (logger *Logger) WithFields(fields Fields) *Entry { + return NewEntry(logger).WithFields(fields) +} + +func (logger *Logger) Debugf(format string, args ...interface{}) { + if logger.Level >= DebugLevel { + NewEntry(logger).Debugf(format, args...) + } +} + +func (logger *Logger) Infof(format string, args ...interface{}) { + if logger.Level >= InfoLevel { + NewEntry(logger).Infof(format, args...) + } +} + +func (logger *Logger) Printf(format string, args ...interface{}) { + NewEntry(logger).Printf(format, args...) +} + +func (logger *Logger) Warnf(format string, args ...interface{}) { + if logger.Level >= WarnLevel { + NewEntry(logger).Warnf(format, args...) + } +} + +func (logger *Logger) Warningf(format string, args ...interface{}) { + if logger.Level >= WarnLevel { + NewEntry(logger).Warnf(format, args...) + } +} + +func (logger *Logger) Errorf(format string, args ...interface{}) { + if logger.Level >= ErrorLevel { + NewEntry(logger).Errorf(format, args...) + } +} + +func (logger *Logger) Fatalf(format string, args ...interface{}) { + if logger.Level >= FatalLevel { + NewEntry(logger).Fatalf(format, args...) + } + os.Exit(1) +} + +func (logger *Logger) Panicf(format string, args ...interface{}) { + if logger.Level >= PanicLevel { + NewEntry(logger).Panicf(format, args...) + } +} + +func (logger *Logger) Debug(args ...interface{}) { + if logger.Level >= DebugLevel { + NewEntry(logger).Debug(args...) + } +} + +func (logger *Logger) Info(args ...interface{}) { + if logger.Level >= InfoLevel { + NewEntry(logger).Info(args...) + } +} + +func (logger *Logger) Print(args ...interface{}) { + NewEntry(logger).Info(args...) +} + +func (logger *Logger) Warn(args ...interface{}) { + if logger.Level >= WarnLevel { + NewEntry(logger).Warn(args...) + } +} + +func (logger *Logger) Warning(args ...interface{}) { + if logger.Level >= WarnLevel { + NewEntry(logger).Warn(args...) + } +} + +func (logger *Logger) Error(args ...interface{}) { + if logger.Level >= ErrorLevel { + NewEntry(logger).Error(args...) + } +} + +func (logger *Logger) Fatal(args ...interface{}) { + if logger.Level >= FatalLevel { + NewEntry(logger).Fatal(args...) + } + os.Exit(1) +} + +func (logger *Logger) Panic(args ...interface{}) { + if logger.Level >= PanicLevel { + NewEntry(logger).Panic(args...) + } +} + +func (logger *Logger) Debugln(args ...interface{}) { + if logger.Level >= DebugLevel { + NewEntry(logger).Debugln(args...) + } +} + +func (logger *Logger) Infoln(args ...interface{}) { + if logger.Level >= InfoLevel { + NewEntry(logger).Infoln(args...) + } +} + +func (logger *Logger) Println(args ...interface{}) { + NewEntry(logger).Println(args...) +} + +func (logger *Logger) Warnln(args ...interface{}) { + if logger.Level >= WarnLevel { + NewEntry(logger).Warnln(args...) + } +} + +func (logger *Logger) Warningln(args ...interface{}) { + if logger.Level >= WarnLevel { + NewEntry(logger).Warnln(args...) + } +} + +func (logger *Logger) Errorln(args ...interface{}) { + if logger.Level >= ErrorLevel { + NewEntry(logger).Errorln(args...) + } +} + +func (logger *Logger) Fatalln(args ...interface{}) { + if logger.Level >= FatalLevel { + NewEntry(logger).Fatalln(args...) + } + os.Exit(1) +} + +func (logger *Logger) Panicln(args ...interface{}) { + if logger.Level >= PanicLevel { + NewEntry(logger).Panicln(args...) + } +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/logrus.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/logrus.go new file mode 100644 index 0000000000000..43ee12e90e7b4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/logrus.go @@ -0,0 +1,94 @@ +package logrus + +import ( + "fmt" + "log" +) + +// Fields type, used to pass to `WithFields`. +type Fields map[string]interface{} + +// Level type +type Level uint8 + +// Convert the Level to a string. E.g. PanicLevel becomes "panic". +func (level Level) String() string { + switch level { + case DebugLevel: + return "debug" + case InfoLevel: + return "info" + case WarnLevel: + return "warning" + case ErrorLevel: + return "error" + case FatalLevel: + return "fatal" + case PanicLevel: + return "panic" + } + + return "unknown" +} + +// ParseLevel takes a string level and returns the Logrus log level constant. +func ParseLevel(lvl string) (Level, error) { + switch lvl { + case "panic": + return PanicLevel, nil + case "fatal": + return FatalLevel, nil + case "error": + return ErrorLevel, nil + case "warn", "warning": + return WarnLevel, nil + case "info": + return InfoLevel, nil + case "debug": + return DebugLevel, nil + } + + var l Level + return l, fmt.Errorf("not a valid logrus Level: %q", lvl) +} + +// These are the different logging levels. You can set the logging level to log +// on your instance of logger, obtained with `logrus.New()`. +const ( + // PanicLevel level, highest level of severity. Logs and then calls panic with the + // message passed to Debug, Info, ... + PanicLevel Level = iota + // FatalLevel level. Logs and then calls `os.Exit(1)`. It will exit even if the + // logging level is set to Panic. + FatalLevel + // ErrorLevel level. Logs. Used for errors that should definitely be noted. + // Commonly used for hooks to send errors to an error tracking service. + ErrorLevel + // WarnLevel level. Non-critical entries that deserve eyes. + WarnLevel + // InfoLevel level. General operational entries about what's going on inside the + // application. + InfoLevel + // DebugLevel level. Usually only enabled when debugging. Very verbose logging. + DebugLevel +) + +// Won't compile if StdLogger can't be realized by a log.Logger +var _ StdLogger = &log.Logger{} + +// StdLogger is what your logrus-enabled library should take, that way +// it'll accept a stdlib logger and a logrus logger. There's no standard +// interface, this is the closest we get, unfortunately. +type StdLogger interface { + Print(...interface{}) + Printf(string, ...interface{}) + Println(...interface{}) + + Fatal(...interface{}) + Fatalf(string, ...interface{}) + Fatalln(...interface{}) + + Panic(...interface{}) + Panicf(string, ...interface{}) + Panicln(...interface{}) +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/logrus_test.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/logrus_test.go new file mode 100644 index 0000000000000..efaacea2361a0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/logrus_test.go @@ -0,0 +1,301 @@ +package logrus + +import ( + "bytes" + "encoding/json" + "strconv" + "strings" + "sync" + "testing" + + "github.com/stretchr/testify/assert" +) + +func LogAndAssertJSON(t *testing.T, log func(*Logger), assertions func(fields Fields)) { + var buffer bytes.Buffer + var fields Fields + + logger := New() + logger.Out = &buffer + logger.Formatter = new(JSONFormatter) + + log(logger) + + err := json.Unmarshal(buffer.Bytes(), &fields) + assert.Nil(t, err) + + assertions(fields) +} + +func LogAndAssertText(t *testing.T, log func(*Logger), assertions func(fields map[string]string)) { + var buffer bytes.Buffer + + logger := New() + logger.Out = &buffer + logger.Formatter = &TextFormatter{ + DisableColors: true, + } + + log(logger) + + fields := make(map[string]string) + for _, kv := range strings.Split(buffer.String(), " ") { + if !strings.Contains(kv, "=") { + continue + } + kvArr := strings.Split(kv, "=") + key := strings.TrimSpace(kvArr[0]) + val := kvArr[1] + if kvArr[1][0] == '"' { + var err error + val, err = strconv.Unquote(val) + assert.NoError(t, err) + } + fields[key] = val + } + assertions(fields) +} + +func TestPrint(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Print("test") + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test") + assert.Equal(t, fields["level"], "info") + }) +} + +func TestInfo(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Info("test") + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test") + assert.Equal(t, fields["level"], "info") + }) +} + +func TestWarn(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Warn("test") + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test") + assert.Equal(t, fields["level"], "warning") + }) +} + +func TestInfolnShouldAddSpacesBetweenStrings(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Infoln("test", "test") + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test test") + }) +} + +func TestInfolnShouldAddSpacesBetweenStringAndNonstring(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Infoln("test", 10) + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test 10") + }) +} + +func TestInfolnShouldAddSpacesBetweenTwoNonStrings(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Infoln(10, 10) + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "10 10") + }) +} + +func TestInfoShouldAddSpacesBetweenTwoNonStrings(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Infoln(10, 10) + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "10 10") + }) +} + +func TestInfoShouldNotAddSpacesBetweenStringAndNonstring(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Info("test", 10) + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test10") + }) +} + +func TestInfoShouldNotAddSpacesBetweenStrings(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Info("test", "test") + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "testtest") + }) +} + +func TestWithFieldsShouldAllowAssignments(t *testing.T) { + var buffer bytes.Buffer + var fields Fields + + logger := New() + logger.Out = &buffer + logger.Formatter = new(JSONFormatter) + + localLog := logger.WithFields(Fields{ + "key1": "value1", + }) + + localLog.WithField("key2", "value2").Info("test") + err := json.Unmarshal(buffer.Bytes(), &fields) + assert.Nil(t, err) + + assert.Equal(t, "value2", fields["key2"]) + assert.Equal(t, "value1", fields["key1"]) + + buffer = bytes.Buffer{} + fields = Fields{} + localLog.Info("test") + err = json.Unmarshal(buffer.Bytes(), &fields) + assert.Nil(t, err) + + _, ok := fields["key2"] + assert.Equal(t, false, ok) + assert.Equal(t, "value1", fields["key1"]) +} + +func TestUserSuppliedFieldDoesNotOverwriteDefaults(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.WithField("msg", "hello").Info("test") + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test") + }) +} + +func TestUserSuppliedMsgFieldHasPrefix(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.WithField("msg", "hello").Info("test") + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test") + assert.Equal(t, fields["fields.msg"], "hello") + }) +} + +func TestUserSuppliedTimeFieldHasPrefix(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.WithField("time", "hello").Info("test") + }, func(fields Fields) { + assert.Equal(t, fields["fields.time"], "hello") + }) +} + +func TestUserSuppliedLevelFieldHasPrefix(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.WithField("level", 1).Info("test") + }, func(fields Fields) { + assert.Equal(t, fields["level"], "info") + assert.Equal(t, fields["fields.level"], 1.0) // JSON has floats only + }) +} + +func TestDefaultFieldsAreNotPrefixed(t *testing.T) { + LogAndAssertText(t, func(log *Logger) { + ll := log.WithField("herp", "derp") + ll.Info("hello") + ll.Info("bye") + }, func(fields map[string]string) { + for _, fieldName := range []string{"fields.level", "fields.time", "fields.msg"} { + if _, ok := fields[fieldName]; ok { + t.Fatalf("should not have prefixed %q: %v", fieldName, fields) + } + } + }) +} + +func TestDoubleLoggingDoesntPrefixPreviousFields(t *testing.T) { + + var buffer bytes.Buffer + var fields Fields + + logger := New() + logger.Out = &buffer + logger.Formatter = new(JSONFormatter) + + llog := logger.WithField("context", "eating raw fish") + + llog.Info("looks delicious") + + err := json.Unmarshal(buffer.Bytes(), &fields) + assert.NoError(t, err, "should have decoded first message") + assert.Equal(t, len(fields), 4, "should only have msg/time/level/context fields") + assert.Equal(t, fields["msg"], "looks delicious") + assert.Equal(t, fields["context"], "eating raw fish") + + buffer.Reset() + + llog.Warn("omg it is!") + + err = json.Unmarshal(buffer.Bytes(), &fields) + assert.NoError(t, err, "should have decoded second message") + assert.Equal(t, len(fields), 4, "should only have msg/time/level/context fields") + assert.Equal(t, fields["msg"], "omg it is!") + assert.Equal(t, fields["context"], "eating raw fish") + assert.Nil(t, fields["fields.msg"], "should not have prefixed previous `msg` entry") + +} + +func TestConvertLevelToString(t *testing.T) { + assert.Equal(t, "debug", DebugLevel.String()) + assert.Equal(t, "info", InfoLevel.String()) + assert.Equal(t, "warning", WarnLevel.String()) + assert.Equal(t, "error", ErrorLevel.String()) + assert.Equal(t, "fatal", FatalLevel.String()) + assert.Equal(t, "panic", PanicLevel.String()) +} + +func TestParseLevel(t *testing.T) { + l, err := ParseLevel("panic") + assert.Nil(t, err) + assert.Equal(t, PanicLevel, l) + + l, err = ParseLevel("fatal") + assert.Nil(t, err) + assert.Equal(t, FatalLevel, l) + + l, err = ParseLevel("error") + assert.Nil(t, err) + assert.Equal(t, ErrorLevel, l) + + l, err = ParseLevel("warn") + assert.Nil(t, err) + assert.Equal(t, WarnLevel, l) + + l, err = ParseLevel("warning") + assert.Nil(t, err) + assert.Equal(t, WarnLevel, l) + + l, err = ParseLevel("info") + assert.Nil(t, err) + assert.Equal(t, InfoLevel, l) + + l, err = ParseLevel("debug") + assert.Nil(t, err) + assert.Equal(t, DebugLevel, l) + + l, err = ParseLevel("invalid") + assert.Equal(t, "not a valid logrus Level: \"invalid\"", err.Error()) +} + +func TestGetSetLevelRace(t *testing.T) { + wg := sync.WaitGroup{} + for i := 0; i < 100; i++ { + wg.Add(1) + go func(i int) { + defer wg.Done() + if i%2 == 0 { + SetLevel(InfoLevel) + } else { + GetLevel() + } + }(i) + + } + wg.Wait() +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/terminal_bsd.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/terminal_bsd.go new file mode 100644 index 0000000000000..71f8d67a55d77 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/terminal_bsd.go @@ -0,0 +1,9 @@ +// +build darwin freebsd openbsd netbsd dragonfly + +package logrus + +import "syscall" + +const ioctlReadTermios = syscall.TIOCGETA + +type Termios syscall.Termios diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/terminal_linux.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/terminal_linux.go new file mode 100644 index 0000000000000..a2c0b40db612b --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/terminal_linux.go @@ -0,0 +1,12 @@ +// Based on ssh/terminal: +// Copyright 2013 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package logrus + +import "syscall" + +const ioctlReadTermios = syscall.TCGETS + +type Termios syscall.Termios diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/terminal_notwindows.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/terminal_notwindows.go new file mode 100644 index 0000000000000..4bb5376028a14 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/terminal_notwindows.go @@ -0,0 +1,21 @@ +// Based on ssh/terminal: +// Copyright 2011 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build linux darwin freebsd openbsd netbsd dragonfly + +package logrus + +import ( + "syscall" + "unsafe" +) + +// IsTerminal returns true if the given file descriptor is a terminal. +func IsTerminal() bool { + fd := syscall.Stdout + var termios Termios + _, _, err := syscall.Syscall6(syscall.SYS_IOCTL, uintptr(fd), ioctlReadTermios, uintptr(unsafe.Pointer(&termios)), 0, 0, 0) + return err == 0 +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/terminal_windows.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/terminal_windows.go new file mode 100644 index 0000000000000..2e09f6f7e31d2 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/terminal_windows.go @@ -0,0 +1,27 @@ +// Based on ssh/terminal: +// Copyright 2011 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build windows + +package logrus + +import ( + "syscall" + "unsafe" +) + +var kernel32 = syscall.NewLazyDLL("kernel32.dll") + +var ( + procGetConsoleMode = kernel32.NewProc("GetConsoleMode") +) + +// IsTerminal returns true if the given file descriptor is a terminal. +func IsTerminal() bool { + fd := syscall.Stdout + var st uint32 + r, _, e := syscall.Syscall(procGetConsoleMode.Addr(), 2, uintptr(fd), uintptr(unsafe.Pointer(&st)), 0) + return r != 0 && e == 0 +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/text_formatter.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/text_formatter.go new file mode 100644 index 0000000000000..17cc29848477d --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/text_formatter.go @@ -0,0 +1,159 @@ +package logrus + +import ( + "bytes" + "fmt" + "runtime" + "sort" + "strings" + "time" +) + +const ( + nocolor = 0 + red = 31 + green = 32 + yellow = 33 + blue = 34 + gray = 37 +) + +var ( + baseTimestamp time.Time + isTerminal bool +) + +func init() { + baseTimestamp = time.Now() + isTerminal = IsTerminal() +} + +func miniTS() int { + return int(time.Since(baseTimestamp) / time.Second) +} + +type TextFormatter struct { + // Set to true to bypass checking for a TTY before outputting colors. + ForceColors bool + + // Force disabling colors. + DisableColors bool + + // Disable timestamp logging. useful when output is redirected to logging + // system that already adds timestamps. + DisableTimestamp bool + + // Enable logging the full timestamp when a TTY is attached instead of just + // the time passed since beginning of execution. + FullTimestamp bool + + // TimestampFormat to use for display when a full timestamp is printed + TimestampFormat string + + // The fields are sorted by default for a consistent output. For applications + // that log extremely frequently and don't use the JSON formatter this may not + // be desired. + DisableSorting bool +} + +func (f *TextFormatter) Format(entry *Entry) ([]byte, error) { + var keys []string = make([]string, 0, len(entry.Data)) + for k := range entry.Data { + keys = append(keys, k) + } + + if !f.DisableSorting { + sort.Strings(keys) + } + + b := &bytes.Buffer{} + + prefixFieldClashes(entry.Data) + + isColorTerminal := isTerminal && (runtime.GOOS != "windows") + isColored := (f.ForceColors || isColorTerminal) && !f.DisableColors + + timestampFormat := f.TimestampFormat + if timestampFormat == "" { + timestampFormat = DefaultTimestampFormat + } + if isColored { + f.printColored(b, entry, keys, timestampFormat) + } else { + if !f.DisableTimestamp { + f.appendKeyValue(b, "time", entry.Time.Format(timestampFormat)) + } + f.appendKeyValue(b, "level", entry.Level.String()) + f.appendKeyValue(b, "msg", entry.Message) + for _, key := range keys { + f.appendKeyValue(b, key, entry.Data[key]) + } + } + + b.WriteByte('\n') + return b.Bytes(), nil +} + +func (f *TextFormatter) printColored(b *bytes.Buffer, entry *Entry, keys []string, timestampFormat string) { + var levelColor int + switch entry.Level { + case DebugLevel: + levelColor = gray + case WarnLevel: + levelColor = yellow + case ErrorLevel, FatalLevel, PanicLevel: + levelColor = red + default: + levelColor = blue + } + + levelText := strings.ToUpper(entry.Level.String())[0:4] + + if !f.FullTimestamp { + fmt.Fprintf(b, "\x1b[%dm%s\x1b[0m[%04d] %-44s ", levelColor, levelText, miniTS(), entry.Message) + } else { + fmt.Fprintf(b, "\x1b[%dm%s\x1b[0m[%s] %-44s ", levelColor, levelText, entry.Time.Format(timestampFormat), entry.Message) + } + for _, k := range keys { + v := entry.Data[k] + fmt.Fprintf(b, " \x1b[%dm%s\x1b[0m=%+v", levelColor, k, v) + } +} + +func needsQuoting(text string) bool { + for _, ch := range text { + if !((ch >= 'a' && ch <= 'z') || + (ch >= 'A' && ch <= 'Z') || + (ch >= '0' && ch <= '9') || + ch == '-' || ch == '.') { + return false + } + } + return true +} + +func (f *TextFormatter) appendKeyValue(b *bytes.Buffer, key string, value interface{}) { + + b.WriteString(key) + b.WriteByte('=') + + switch value := value.(type) { + case string: + if needsQuoting(value) { + b.WriteString(value) + } else { + fmt.Fprintf(b, "%q", value) + } + case error: + errmsg := value.Error() + if needsQuoting(errmsg) { + b.WriteString(errmsg) + } else { + fmt.Fprintf(b, "%q", value) + } + default: + fmt.Fprint(b, value) + } + + b.WriteByte(' ') +} diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/text_formatter_test.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/text_formatter_test.go new file mode 100644 index 0000000000000..e25a44f67bfe3 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/text_formatter_test.go @@ -0,0 +1,61 @@ +package logrus + +import ( + "bytes" + "errors" + "testing" + "time" +) + +func TestQuoting(t *testing.T) { + tf := &TextFormatter{DisableColors: true} + + checkQuoting := func(q bool, value interface{}) { + b, _ := tf.Format(WithField("test", value)) + idx := bytes.Index(b, ([]byte)("test=")) + cont := bytes.Contains(b[idx+5:], []byte{'"'}) + if cont != q { + if q { + t.Errorf("quoting expected for: %#v", value) + } else { + t.Errorf("quoting not expected for: %#v", value) + } + } + } + + checkQuoting(false, "abcd") + checkQuoting(false, "v1.0") + checkQuoting(false, "1234567890") + checkQuoting(true, "/foobar") + checkQuoting(true, "x y") + checkQuoting(true, "x,y") + checkQuoting(false, errors.New("invalid")) + checkQuoting(true, errors.New("invalid argument")) +} + +func TestTimestampFormat(t *testing.T) { + checkTimeStr := func(format string) { + customFormatter := &TextFormatter{DisableColors: true, TimestampFormat: format} + customStr, _ := customFormatter.Format(WithField("test", "test")) + timeStart := bytes.Index(customStr, ([]byte)("time=")) + timeEnd := bytes.Index(customStr, ([]byte)("level=")) + timeStr := customStr[timeStart+5 : timeEnd-1] + if timeStr[0] == '"' && timeStr[len(timeStr)-1] == '"' { + timeStr = timeStr[1 : len(timeStr)-1] + } + if format == "" { + format = time.RFC3339 + } + _, e := time.Parse(format, (string)(timeStr)) + if e != nil { + t.Errorf("time string \"%s\" did not match provided time format \"%s\": %s", timeStr, format, e) + } + } + + checkTimeStr("2006-01-02T15:04:05.000000000Z07:00") + checkTimeStr("Mon Jan _2 15:04:05 2006") + checkTimeStr("") +} + +// TODO add tests for sorting etc., this requires a parser for the text +// formatter output. diff --git a/Godeps/_workspace/src/github.com/Sirupsen/logrus/writer.go b/Godeps/_workspace/src/github.com/Sirupsen/logrus/writer.go new file mode 100644 index 0000000000000..1e30b1c753a74 --- /dev/null +++ b/Godeps/_workspace/src/github.com/Sirupsen/logrus/writer.go @@ -0,0 +1,31 @@ +package logrus + +import ( + "bufio" + "io" + "runtime" +) + +func (logger *Logger) Writer() *io.PipeWriter { + reader, writer := io.Pipe() + + go logger.writerScanner(reader) + runtime.SetFinalizer(writer, writerFinalizer) + + return writer +} + +func (logger *Logger) writerScanner(reader *io.PipeReader) { + scanner := bufio.NewScanner(reader) + for scanner.Scan() { + logger.Print(scanner.Text()) + } + if err := scanner.Err(); err != nil { + logger.Errorf("Error while reading from Writer: %s", err) + } + reader.Close() +} + +func writerFinalizer(writer *io.PipeWriter) { + writer.Close() +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/.gitignore b/Godeps/_workspace/src/github.com/armon/go-metrics/.gitignore new file mode 100644 index 0000000000000..00268614f0456 --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/.gitignore @@ -0,0 +1,22 @@ +# Compiled Object files, Static and Dynamic libs (Shared Objects) +*.o +*.a +*.so + +# Folders +_obj +_test + +# Architecture specific extensions/prefixes +*.[568vq] +[568vq].out + +*.cgo1.go +*.cgo2.c +_cgo_defun.c +_cgo_gotypes.go +_cgo_export.* + +_testmain.go + +*.exe diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/LICENSE b/Godeps/_workspace/src/github.com/armon/go-metrics/LICENSE new file mode 100644 index 0000000000000..106569e542b0d --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/LICENSE @@ -0,0 +1,20 @@ +The MIT License (MIT) + +Copyright (c) 2013 Armon Dadgar + +Permission is hereby granted, free of charge, to any person obtaining a copy of +this software and associated documentation files (the "Software"), to deal in +the Software without restriction, including without limitation the rights to +use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of +the Software, and to permit persons to whom the Software is furnished to do so, +subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS +FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR +COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER +IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN +CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/README.md b/Godeps/_workspace/src/github.com/armon/go-metrics/README.md new file mode 100644 index 0000000000000..7b6f23e29f839 --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/README.md @@ -0,0 +1,71 @@ +go-metrics +========== + +This library provides a `metrics` package which can be used to instrument code, +expose application metrics, and profile runtime performance in a flexible manner. + +Current API: [![GoDoc](https://godoc.org/github.com/armon/go-metrics?status.svg)](https://godoc.org/github.com/armon/go-metrics) + +Sinks +===== + +The `metrics` package makes use of a `MetricSink` interface to support delivery +to any type of backend. Currently the following sinks are provided: + +* StatsiteSink : Sinks to a [statsite](https://github.com/armon/statsite/) instance (TCP) +* StatsdSink: Sinks to a [StatsD](https://github.com/etsy/statsd/) / statsite instance (UDP) +* PrometheusSink: Sinks to a [Prometheus](http://prometheus.io/) metrics endpoint (exposed via HTTP for scrapes) +* InmemSink : Provides in-memory aggregation, can be used to export stats +* FanoutSink : Sinks to multiple sinks. Enables writing to multiple statsite instances for example. +* BlackholeSink : Sinks to nowhere + +In addition to the sinks, the `InmemSignal` can be used to catch a signal, +and dump a formatted output of recent metrics. For example, when a process gets +a SIGUSR1, it can dump to stderr recent performance metrics for debugging. + +Examples +======== + +Here is an example of using the package: + + func SlowMethod() { + // Profiling the runtime of a method + defer metrics.MeasureSince([]string{"SlowMethod"}, time.Now()) + } + + // Configure a statsite sink as the global metrics sink + sink, _ := metrics.NewStatsiteSink("statsite:8125") + metrics.NewGlobal(metrics.DefaultConfig("service-name"), sink) + + // Emit a Key/Value pair + metrics.EmitKey([]string{"questions", "meaning of life"}, 42) + + +Here is an example of setting up an signal handler: + + // Setup the inmem sink and signal handler + inm := metrics.NewInmemSink(10*time.Second, time.Minute) + sig := metrics.DefaultInmemSignal(inm) + metrics.NewGlobal(metrics.DefaultConfig("service-name"), inm) + + // Run some code + inm.SetGauge([]string{"foo"}, 42) + inm.EmitKey([]string{"bar"}, 30) + + inm.IncrCounter([]string{"baz"}, 42) + inm.IncrCounter([]string{"baz"}, 1) + inm.IncrCounter([]string{"baz"}, 80) + + inm.AddSample([]string{"method", "wow"}, 42) + inm.AddSample([]string{"method", "wow"}, 100) + inm.AddSample([]string{"method", "wow"}, 22) + + .... + +When a signal comes in, output like the following will be dumped to stderr: + + [2014-01-28 14:57:33.04 -0800 PST][G] 'foo': 42.000 + [2014-01-28 14:57:33.04 -0800 PST][P] 'bar': 30.000 + [2014-01-28 14:57:33.04 -0800 PST][C] 'baz': Count: 3 Min: 1.000 Mean: 41.000 Max: 80.000 Stddev: 39.509 + [2014-01-28 14:57:33.04 -0800 PST][S] 'method.wow': Count: 3 Min: 22.000 Mean: 54.667 Max: 100.000 Stddev: 40.513 + diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/const_unix.go b/Godeps/_workspace/src/github.com/armon/go-metrics/const_unix.go new file mode 100644 index 0000000000000..31098dd57e555 --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/const_unix.go @@ -0,0 +1,12 @@ +// +build !windows + +package metrics + +import ( + "syscall" +) + +const ( + // DefaultSignal is used with DefaultInmemSignal + DefaultSignal = syscall.SIGUSR1 +) diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/const_windows.go b/Godeps/_workspace/src/github.com/armon/go-metrics/const_windows.go new file mode 100644 index 0000000000000..38136af3e4237 --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/const_windows.go @@ -0,0 +1,13 @@ +// +build windows + +package metrics + +import ( + "syscall" +) + +const ( + // DefaultSignal is used with DefaultInmemSignal + // Windows has no SIGUSR1, use SIGBREAK + DefaultSignal = syscall.Signal(21) +) diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/inmem.go b/Godeps/_workspace/src/github.com/armon/go-metrics/inmem.go new file mode 100644 index 0000000000000..da503296060ab --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/inmem.go @@ -0,0 +1,241 @@ +package metrics + +import ( + "fmt" + "math" + "strings" + "sync" + "time" +) + +// InmemSink provides a MetricSink that does in-memory aggregation +// without sending metrics over a network. It can be embedded within +// an application to provide profiling information. +type InmemSink struct { + // How long is each aggregation interval + interval time.Duration + + // Retain controls how many metrics interval we keep + retain time.Duration + + // maxIntervals is the maximum length of intervals. + // It is retain / interval. + maxIntervals int + + // intervals is a slice of the retained intervals + intervals []*IntervalMetrics + intervalLock sync.RWMutex +} + +// IntervalMetrics stores the aggregated metrics +// for a specific interval +type IntervalMetrics struct { + sync.RWMutex + + // The start time of the interval + Interval time.Time + + // Gauges maps the key to the last set value + Gauges map[string]float32 + + // Points maps the string to the list of emitted values + // from EmitKey + Points map[string][]float32 + + // Counters maps the string key to a sum of the counter + // values + Counters map[string]*AggregateSample + + // Samples maps the key to an AggregateSample, + // which has the rolled up view of a sample + Samples map[string]*AggregateSample +} + +// NewIntervalMetrics creates a new IntervalMetrics for a given interval +func NewIntervalMetrics(intv time.Time) *IntervalMetrics { + return &IntervalMetrics{ + Interval: intv, + Gauges: make(map[string]float32), + Points: make(map[string][]float32), + Counters: make(map[string]*AggregateSample), + Samples: make(map[string]*AggregateSample), + } +} + +// AggregateSample is used to hold aggregate metrics +// about a sample +type AggregateSample struct { + Count int // The count of emitted pairs + Sum float64 // The sum of values + SumSq float64 // The sum of squared values + Min float64 // Minimum value + Max float64 // Maximum value + LastUpdated time.Time // When value was last updated +} + +// Computes a Stddev of the values +func (a *AggregateSample) Stddev() float64 { + num := (float64(a.Count) * a.SumSq) - math.Pow(a.Sum, 2) + div := float64(a.Count * (a.Count - 1)) + if div == 0 { + return 0 + } + return math.Sqrt(num / div) +} + +// Computes a mean of the values +func (a *AggregateSample) Mean() float64 { + if a.Count == 0 { + return 0 + } + return a.Sum / float64(a.Count) +} + +// Ingest is used to update a sample +func (a *AggregateSample) Ingest(v float64) { + a.Count++ + a.Sum += v + a.SumSq += (v * v) + if v < a.Min || a.Count == 1 { + a.Min = v + } + if v > a.Max || a.Count == 1 { + a.Max = v + } + a.LastUpdated = time.Now() +} + +func (a *AggregateSample) String() string { + if a.Count == 0 { + return "Count: 0" + } else if a.Stddev() == 0 { + return fmt.Sprintf("Count: %d Sum: %0.3f LastUpdated: %s", a.Count, a.Sum, a.LastUpdated) + } else { + return fmt.Sprintf("Count: %d Min: %0.3f Mean: %0.3f Max: %0.3f Stddev: %0.3f Sum: %0.3f LastUpdated: %s", + a.Count, a.Min, a.Mean(), a.Max, a.Stddev(), a.Sum, a.LastUpdated) + } +} + +// NewInmemSink is used to construct a new in-memory sink. +// Uses an aggregation interval and maximum retention period. +func NewInmemSink(interval, retain time.Duration) *InmemSink { + i := &InmemSink{ + interval: interval, + retain: retain, + maxIntervals: int(retain / interval), + } + i.intervals = make([]*IntervalMetrics, 0, i.maxIntervals) + return i +} + +func (i *InmemSink) SetGauge(key []string, val float32) { + k := i.flattenKey(key) + intv := i.getInterval() + + intv.Lock() + defer intv.Unlock() + intv.Gauges[k] = val +} + +func (i *InmemSink) EmitKey(key []string, val float32) { + k := i.flattenKey(key) + intv := i.getInterval() + + intv.Lock() + defer intv.Unlock() + vals := intv.Points[k] + intv.Points[k] = append(vals, val) +} + +func (i *InmemSink) IncrCounter(key []string, val float32) { + k := i.flattenKey(key) + intv := i.getInterval() + + intv.Lock() + defer intv.Unlock() + + agg := intv.Counters[k] + if agg == nil { + agg = &AggregateSample{} + intv.Counters[k] = agg + } + agg.Ingest(float64(val)) +} + +func (i *InmemSink) AddSample(key []string, val float32) { + k := i.flattenKey(key) + intv := i.getInterval() + + intv.Lock() + defer intv.Unlock() + + agg := intv.Samples[k] + if agg == nil { + agg = &AggregateSample{} + intv.Samples[k] = agg + } + agg.Ingest(float64(val)) +} + +// Data is used to retrieve all the aggregated metrics +// Intervals may be in use, and a read lock should be acquired +func (i *InmemSink) Data() []*IntervalMetrics { + // Get the current interval, forces creation + i.getInterval() + + i.intervalLock.RLock() + defer i.intervalLock.RUnlock() + + intervals := make([]*IntervalMetrics, len(i.intervals)) + copy(intervals, i.intervals) + return intervals +} + +func (i *InmemSink) getExistingInterval(intv time.Time) *IntervalMetrics { + i.intervalLock.RLock() + defer i.intervalLock.RUnlock() + + n := len(i.intervals) + if n > 0 && i.intervals[n-1].Interval == intv { + return i.intervals[n-1] + } + return nil +} + +func (i *InmemSink) createInterval(intv time.Time) *IntervalMetrics { + i.intervalLock.Lock() + defer i.intervalLock.Unlock() + + // Check for an existing interval + n := len(i.intervals) + if n > 0 && i.intervals[n-1].Interval == intv { + return i.intervals[n-1] + } + + // Add the current interval + current := NewIntervalMetrics(intv) + i.intervals = append(i.intervals, current) + n++ + + // Truncate the intervals if they are too long + if n >= i.maxIntervals { + copy(i.intervals[0:], i.intervals[n-i.maxIntervals:]) + i.intervals = i.intervals[:i.maxIntervals] + } + return current +} + +// getInterval returns the current interval to write to +func (i *InmemSink) getInterval() *IntervalMetrics { + intv := time.Now().Truncate(i.interval) + if m := i.getExistingInterval(intv); m != nil { + return m + } + return i.createInterval(intv) +} + +// Flattens the key for formatting, removes spaces +func (i *InmemSink) flattenKey(parts []string) string { + joined := strings.Join(parts, ".") + return strings.Replace(joined, " ", "_", -1) +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/inmem_signal.go b/Godeps/_workspace/src/github.com/armon/go-metrics/inmem_signal.go new file mode 100644 index 0000000000000..95d08ee10f0b1 --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/inmem_signal.go @@ -0,0 +1,100 @@ +package metrics + +import ( + "bytes" + "fmt" + "io" + "os" + "os/signal" + "sync" + "syscall" +) + +// InmemSignal is used to listen for a given signal, and when received, +// to dump the current metrics from the InmemSink to an io.Writer +type InmemSignal struct { + signal syscall.Signal + inm *InmemSink + w io.Writer + sigCh chan os.Signal + + stop bool + stopCh chan struct{} + stopLock sync.Mutex +} + +// NewInmemSignal creates a new InmemSignal which listens for a given signal, +// and dumps the current metrics out to a writer +func NewInmemSignal(inmem *InmemSink, sig syscall.Signal, w io.Writer) *InmemSignal { + i := &InmemSignal{ + signal: sig, + inm: inmem, + w: w, + sigCh: make(chan os.Signal, 1), + stopCh: make(chan struct{}), + } + signal.Notify(i.sigCh, sig) + go i.run() + return i +} + +// DefaultInmemSignal returns a new InmemSignal that responds to SIGUSR1 +// and writes output to stderr. Windows uses SIGBREAK +func DefaultInmemSignal(inmem *InmemSink) *InmemSignal { + return NewInmemSignal(inmem, DefaultSignal, os.Stderr) +} + +// Stop is used to stop the InmemSignal from listening +func (i *InmemSignal) Stop() { + i.stopLock.Lock() + defer i.stopLock.Unlock() + + if i.stop { + return + } + i.stop = true + close(i.stopCh) + signal.Stop(i.sigCh) +} + +// run is a long running routine that handles signals +func (i *InmemSignal) run() { + for { + select { + case <-i.sigCh: + i.dumpStats() + case <-i.stopCh: + return + } + } +} + +// dumpStats is used to dump the data to output writer +func (i *InmemSignal) dumpStats() { + buf := bytes.NewBuffer(nil) + + data := i.inm.Data() + // Skip the last period which is still being aggregated + for i := 0; i < len(data)-1; i++ { + intv := data[i] + intv.RLock() + for name, val := range intv.Gauges { + fmt.Fprintf(buf, "[%v][G] '%s': %0.3f\n", intv.Interval, name, val) + } + for name, vals := range intv.Points { + for _, val := range vals { + fmt.Fprintf(buf, "[%v][P] '%s': %0.3f\n", intv.Interval, name, val) + } + } + for name, agg := range intv.Counters { + fmt.Fprintf(buf, "[%v][C] '%s': %s\n", intv.Interval, name, agg) + } + for name, agg := range intv.Samples { + fmt.Fprintf(buf, "[%v][S] '%s': %s\n", intv.Interval, name, agg) + } + intv.RUnlock() + } + + // Write out the bytes + i.w.Write(buf.Bytes()) +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/inmem_signal_test.go b/Godeps/_workspace/src/github.com/armon/go-metrics/inmem_signal_test.go new file mode 100644 index 0000000000000..9bbca5f254b49 --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/inmem_signal_test.go @@ -0,0 +1,46 @@ +package metrics + +import ( + "bytes" + "os" + "strings" + "syscall" + "testing" + "time" +) + +func TestInmemSignal(t *testing.T) { + buf := bytes.NewBuffer(nil) + inm := NewInmemSink(10*time.Millisecond, 50*time.Millisecond) + sig := NewInmemSignal(inm, syscall.SIGUSR1, buf) + defer sig.Stop() + + inm.SetGauge([]string{"foo"}, 42) + inm.EmitKey([]string{"bar"}, 42) + inm.IncrCounter([]string{"baz"}, 42) + inm.AddSample([]string{"wow"}, 42) + + // Wait for period to end + time.Sleep(15 * time.Millisecond) + + // Send signal! + syscall.Kill(os.Getpid(), syscall.SIGUSR1) + + // Wait for flush + time.Sleep(10 * time.Millisecond) + + // Check the output + out := string(buf.Bytes()) + if !strings.Contains(out, "[G] 'foo': 42") { + t.Fatalf("bad: %v", out) + } + if !strings.Contains(out, "[P] 'bar': 42") { + t.Fatalf("bad: %v", out) + } + if !strings.Contains(out, "[C] 'baz': Count: 1 Sum: 42") { + t.Fatalf("bad: %v", out) + } + if !strings.Contains(out, "[S] 'wow': Count: 1 Sum: 42") { + t.Fatalf("bad: %v", out) + } +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/inmem_test.go b/Godeps/_workspace/src/github.com/armon/go-metrics/inmem_test.go new file mode 100644 index 0000000000000..228a2fc1af91c --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/inmem_test.go @@ -0,0 +1,104 @@ +package metrics + +import ( + "math" + "testing" + "time" +) + +func TestInmemSink(t *testing.T) { + inm := NewInmemSink(10*time.Millisecond, 50*time.Millisecond) + + data := inm.Data() + if len(data) != 1 { + t.Fatalf("bad: %v", data) + } + + // Add data points + inm.SetGauge([]string{"foo", "bar"}, 42) + inm.EmitKey([]string{"foo", "bar"}, 42) + inm.IncrCounter([]string{"foo", "bar"}, 20) + inm.IncrCounter([]string{"foo", "bar"}, 22) + inm.AddSample([]string{"foo", "bar"}, 20) + inm.AddSample([]string{"foo", "bar"}, 22) + + data = inm.Data() + if len(data) != 1 { + t.Fatalf("bad: %v", data) + } + + intvM := data[0] + intvM.RLock() + + if time.Now().Sub(intvM.Interval) > 10*time.Millisecond { + t.Fatalf("interval too old") + } + if intvM.Gauges["foo.bar"] != 42 { + t.Fatalf("bad val: %v", intvM.Gauges) + } + if intvM.Points["foo.bar"][0] != 42 { + t.Fatalf("bad val: %v", intvM.Points) + } + + agg := intvM.Counters["foo.bar"] + if agg.Count != 2 { + t.Fatalf("bad val: %v", agg) + } + if agg.Sum != 42 { + t.Fatalf("bad val: %v", agg) + } + if agg.SumSq != 884 { + t.Fatalf("bad val: %v", agg) + } + if agg.Min != 20 { + t.Fatalf("bad val: %v", agg) + } + if agg.Max != 22 { + t.Fatalf("bad val: %v", agg) + } + if agg.Mean() != 21 { + t.Fatalf("bad val: %v", agg) + } + if agg.Stddev() != math.Sqrt(2) { + t.Fatalf("bad val: %v", agg) + } + + if agg.LastUpdated.IsZero() { + t.Fatalf("agg.LastUpdated is not set: %v", agg) + } + + diff := time.Now().Sub(agg.LastUpdated).Seconds() + if diff > 1 { + t.Fatalf("time diff too great: %f", diff) + } + + if agg = intvM.Samples["foo.bar"]; agg == nil { + t.Fatalf("missing sample") + } + + intvM.RUnlock() + + for i := 1; i < 10; i++ { + time.Sleep(10 * time.Millisecond) + inm.SetGauge([]string{"foo", "bar"}, 42) + data = inm.Data() + if len(data) != min(i+1, 5) { + t.Fatalf("bad: %v", data) + } + } + + // Should not exceed 5 intervals! + time.Sleep(10 * time.Millisecond) + inm.SetGauge([]string{"foo", "bar"}, 42) + data = inm.Data() + if len(data) != 5 { + t.Fatalf("bad: %v", data) + } +} + +func min(a, b int) int { + if a < b { + return a + } + return b +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/metrics.go b/Godeps/_workspace/src/github.com/armon/go-metrics/metrics.go new file mode 100644 index 0000000000000..b818e4182c0cb --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/metrics.go @@ -0,0 +1,115 @@ +package metrics + +import ( + "runtime" + "time" +) + +func (m *Metrics) SetGauge(key []string, val float32) { + if m.HostName != "" && m.EnableHostname { + key = insert(0, m.HostName, key) + } + if m.EnableTypePrefix { + key = insert(0, "gauge", key) + } + if m.ServiceName != "" { + key = insert(0, m.ServiceName, key) + } + m.sink.SetGauge(key, val) +} + +func (m *Metrics) EmitKey(key []string, val float32) { + if m.EnableTypePrefix { + key = insert(0, "kv", key) + } + if m.ServiceName != "" { + key = insert(0, m.ServiceName, key) + } + m.sink.EmitKey(key, val) +} + +func (m *Metrics) IncrCounter(key []string, val float32) { + if m.EnableTypePrefix { + key = insert(0, "counter", key) + } + if m.ServiceName != "" { + key = insert(0, m.ServiceName, key) + } + m.sink.IncrCounter(key, val) +} + +func (m *Metrics) AddSample(key []string, val float32) { + if m.EnableTypePrefix { + key = insert(0, "sample", key) + } + if m.ServiceName != "" { + key = insert(0, m.ServiceName, key) + } + m.sink.AddSample(key, val) +} + +func (m *Metrics) MeasureSince(key []string, start time.Time) { + if m.EnableTypePrefix { + key = insert(0, "timer", key) + } + if m.ServiceName != "" { + key = insert(0, m.ServiceName, key) + } + now := time.Now() + elapsed := now.Sub(start) + msec := float32(elapsed.Nanoseconds()) / float32(m.TimerGranularity) + m.sink.AddSample(key, msec) +} + +// Periodically collects runtime stats to publish +func (m *Metrics) collectStats() { + for { + time.Sleep(m.ProfileInterval) + m.emitRuntimeStats() + } +} + +// Emits various runtime statsitics +func (m *Metrics) emitRuntimeStats() { + // Export number of Goroutines + numRoutines := runtime.NumGoroutine() + m.SetGauge([]string{"runtime", "num_goroutines"}, float32(numRoutines)) + + // Export memory stats + var stats runtime.MemStats + runtime.ReadMemStats(&stats) + m.SetGauge([]string{"runtime", "alloc_bytes"}, float32(stats.Alloc)) + m.SetGauge([]string{"runtime", "sys_bytes"}, float32(stats.Sys)) + m.SetGauge([]string{"runtime", "malloc_count"}, float32(stats.Mallocs)) + m.SetGauge([]string{"runtime", "free_count"}, float32(stats.Frees)) + m.SetGauge([]string{"runtime", "heap_objects"}, float32(stats.HeapObjects)) + m.SetGauge([]string{"runtime", "total_gc_pause_ns"}, float32(stats.PauseTotalNs)) + m.SetGauge([]string{"runtime", "total_gc_runs"}, float32(stats.NumGC)) + + // Export info about the last few GC runs + num := stats.NumGC + + // Handle wrap around + if num < m.lastNumGC { + m.lastNumGC = 0 + } + + // Ensure we don't scan more than 256 + if num-m.lastNumGC >= 256 { + m.lastNumGC = num - 255 + } + + for i := m.lastNumGC; i < num; i++ { + pause := stats.PauseNs[i%256] + m.AddSample([]string{"runtime", "gc_pause_ns"}, float32(pause)) + } + m.lastNumGC = num +} + +// Inserts a string value at an index into the slice +func insert(i int, v string, s []string) []string { + s = append(s, "") + copy(s[i+1:], s[i:]) + s[i] = v + return s +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/metrics_test.go b/Godeps/_workspace/src/github.com/armon/go-metrics/metrics_test.go new file mode 100644 index 0000000000000..c7baf22bf4db6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/metrics_test.go @@ -0,0 +1,262 @@ +package metrics + +import ( + "reflect" + "runtime" + "testing" + "time" +) + +func mockMetric() (*MockSink, *Metrics) { + m := &MockSink{} + met := &Metrics{sink: m} + return m, met +} + +func TestMetrics_SetGauge(t *testing.T) { + m, met := mockMetric() + met.SetGauge([]string{"key"}, float32(1)) + if m.keys[0][0] != "key" { + t.Fatalf("") + } + if m.vals[0] != 1 { + t.Fatalf("") + } + + m, met = mockMetric() + met.HostName = "test" + met.EnableHostname = true + met.SetGauge([]string{"key"}, float32(1)) + if m.keys[0][0] != "test" || m.keys[0][1] != "key" { + t.Fatalf("") + } + if m.vals[0] != 1 { + t.Fatalf("") + } + + m, met = mockMetric() + met.EnableTypePrefix = true + met.SetGauge([]string{"key"}, float32(1)) + if m.keys[0][0] != "gauge" || m.keys[0][1] != "key" { + t.Fatalf("") + } + if m.vals[0] != 1 { + t.Fatalf("") + } + + m, met = mockMetric() + met.ServiceName = "service" + met.SetGauge([]string{"key"}, float32(1)) + if m.keys[0][0] != "service" || m.keys[0][1] != "key" { + t.Fatalf("") + } + if m.vals[0] != 1 { + t.Fatalf("") + } +} + +func TestMetrics_EmitKey(t *testing.T) { + m, met := mockMetric() + met.EmitKey([]string{"key"}, float32(1)) + if m.keys[0][0] != "key" { + t.Fatalf("") + } + if m.vals[0] != 1 { + t.Fatalf("") + } + + m, met = mockMetric() + met.EnableTypePrefix = true + met.EmitKey([]string{"key"}, float32(1)) + if m.keys[0][0] != "kv" || m.keys[0][1] != "key" { + t.Fatalf("") + } + if m.vals[0] != 1 { + t.Fatalf("") + } + + m, met = mockMetric() + met.ServiceName = "service" + met.EmitKey([]string{"key"}, float32(1)) + if m.keys[0][0] != "service" || m.keys[0][1] != "key" { + t.Fatalf("") + } + if m.vals[0] != 1 { + t.Fatalf("") + } +} + +func TestMetrics_IncrCounter(t *testing.T) { + m, met := mockMetric() + met.IncrCounter([]string{"key"}, float32(1)) + if m.keys[0][0] != "key" { + t.Fatalf("") + } + if m.vals[0] != 1 { + t.Fatalf("") + } + + m, met = mockMetric() + met.EnableTypePrefix = true + met.IncrCounter([]string{"key"}, float32(1)) + if m.keys[0][0] != "counter" || m.keys[0][1] != "key" { + t.Fatalf("") + } + if m.vals[0] != 1 { + t.Fatalf("") + } + + m, met = mockMetric() + met.ServiceName = "service" + met.IncrCounter([]string{"key"}, float32(1)) + if m.keys[0][0] != "service" || m.keys[0][1] != "key" { + t.Fatalf("") + } + if m.vals[0] != 1 { + t.Fatalf("") + } +} + +func TestMetrics_AddSample(t *testing.T) { + m, met := mockMetric() + met.AddSample([]string{"key"}, float32(1)) + if m.keys[0][0] != "key" { + t.Fatalf("") + } + if m.vals[0] != 1 { + t.Fatalf("") + } + + m, met = mockMetric() + met.EnableTypePrefix = true + met.AddSample([]string{"key"}, float32(1)) + if m.keys[0][0] != "sample" || m.keys[0][1] != "key" { + t.Fatalf("") + } + if m.vals[0] != 1 { + t.Fatalf("") + } + + m, met = mockMetric() + met.ServiceName = "service" + met.AddSample([]string{"key"}, float32(1)) + if m.keys[0][0] != "service" || m.keys[0][1] != "key" { + t.Fatalf("") + } + if m.vals[0] != 1 { + t.Fatalf("") + } +} + +func TestMetrics_MeasureSince(t *testing.T) { + m, met := mockMetric() + met.TimerGranularity = time.Millisecond + n := time.Now() + met.MeasureSince([]string{"key"}, n) + if m.keys[0][0] != "key" { + t.Fatalf("") + } + if m.vals[0] > 0.1 { + t.Fatalf("") + } + + m, met = mockMetric() + met.TimerGranularity = time.Millisecond + met.EnableTypePrefix = true + met.MeasureSince([]string{"key"}, n) + if m.keys[0][0] != "timer" || m.keys[0][1] != "key" { + t.Fatalf("") + } + if m.vals[0] > 0.1 { + t.Fatalf("") + } + + m, met = mockMetric() + met.TimerGranularity = time.Millisecond + met.ServiceName = "service" + met.MeasureSince([]string{"key"}, n) + if m.keys[0][0] != "service" || m.keys[0][1] != "key" { + t.Fatalf("") + } + if m.vals[0] > 0.1 { + t.Fatalf("") + } +} + +func TestMetrics_EmitRuntimeStats(t *testing.T) { + runtime.GC() + m, met := mockMetric() + met.emitRuntimeStats() + + if m.keys[0][0] != "runtime" || m.keys[0][1] != "num_goroutines" { + t.Fatalf("bad key %v", m.keys) + } + if m.vals[0] <= 1 { + t.Fatalf("bad val: %v", m.vals) + } + + if m.keys[1][0] != "runtime" || m.keys[1][1] != "alloc_bytes" { + t.Fatalf("bad key %v", m.keys) + } + if m.vals[1] <= 40000 { + t.Fatalf("bad val: %v", m.vals) + } + + if m.keys[2][0] != "runtime" || m.keys[2][1] != "sys_bytes" { + t.Fatalf("bad key %v", m.keys) + } + if m.vals[2] <= 100000 { + t.Fatalf("bad val: %v", m.vals) + } + + if m.keys[3][0] != "runtime" || m.keys[3][1] != "malloc_count" { + t.Fatalf("bad key %v", m.keys) + } + if m.vals[3] <= 100 { + t.Fatalf("bad val: %v", m.vals) + } + + if m.keys[4][0] != "runtime" || m.keys[4][1] != "free_count" { + t.Fatalf("bad key %v", m.keys) + } + if m.vals[4] <= 100 { + t.Fatalf("bad val: %v", m.vals) + } + + if m.keys[5][0] != "runtime" || m.keys[5][1] != "heap_objects" { + t.Fatalf("bad key %v", m.keys) + } + if m.vals[5] <= 100 { + t.Fatalf("bad val: %v", m.vals) + } + + if m.keys[6][0] != "runtime" || m.keys[6][1] != "total_gc_pause_ns" { + t.Fatalf("bad key %v", m.keys) + } + if m.vals[6] <= 100000 { + t.Fatalf("bad val: %v", m.vals) + } + + if m.keys[7][0] != "runtime" || m.keys[7][1] != "total_gc_runs" { + t.Fatalf("bad key %v", m.keys) + } + if m.vals[7] <= 1 { + t.Fatalf("bad val: %v", m.vals) + } + + if m.keys[8][0] != "runtime" || m.keys[8][1] != "gc_pause_ns" { + t.Fatalf("bad key %v", m.keys) + } + if m.vals[8] <= 1000 { + t.Fatalf("bad val: %v", m.vals) + } +} + +func TestInsert(t *testing.T) { + k := []string{"hi", "bob"} + exp := []string{"hi", "there", "bob"} + out := insert(1, "there", k) + if !reflect.DeepEqual(exp, out) { + t.Fatalf("bad insert %v %v", exp, out) + } +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/prometheus/prometheus.go b/Godeps/_workspace/src/github.com/armon/go-metrics/prometheus/prometheus.go new file mode 100644 index 0000000000000..362dbfb623d29 --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/prometheus/prometheus.go @@ -0,0 +1,88 @@ +// +build go1.3 +package prometheus + +import ( + "strings" + "sync" + "time" + + "github.com/prometheus/client_golang/prometheus" +) + +type PrometheusSink struct { + mu sync.Mutex + gauges map[string]prometheus.Gauge + summaries map[string]prometheus.Summary + counters map[string]prometheus.Counter +} + +func NewPrometheusSink() (*PrometheusSink, error) { + return &PrometheusSink{ + gauges: make(map[string]prometheus.Gauge), + summaries: make(map[string]prometheus.Summary), + counters: make(map[string]prometheus.Counter), + }, nil +} + +func (p *PrometheusSink) flattenKey(parts []string) string { + joined := strings.Join(parts, "_") + joined = strings.Replace(joined, " ", "_", -1) + joined = strings.Replace(joined, ".", "_", -1) + joined = strings.Replace(joined, "-", "_", -1) + return joined +} + +func (p *PrometheusSink) SetGauge(parts []string, val float32) { + p.mu.Lock() + defer p.mu.Unlock() + key := p.flattenKey(parts) + g, ok := p.gauges[key] + if !ok { + g = prometheus.NewGauge(prometheus.GaugeOpts{ + Name: key, + Help: key, + }) + prometheus.MustRegister(g) + p.gauges[key] = g + } + g.Set(float64(val)) +} + +func (p *PrometheusSink) AddSample(parts []string, val float32) { + p.mu.Lock() + defer p.mu.Unlock() + key := p.flattenKey(parts) + g, ok := p.summaries[key] + if !ok { + g = prometheus.NewSummary(prometheus.SummaryOpts{ + Name: key, + Help: key, + MaxAge: 10 * time.Second, + }) + prometheus.MustRegister(g) + p.summaries[key] = g + } + g.Observe(float64(val)) +} + +// EmitKey is not implemented. Prometheus doesn’t offer a type for which an +// arbitrary number of values is retained, as Prometheus works with a pull +// model, rather than a push model. +func (p *PrometheusSink) EmitKey(key []string, val float32) { +} + +func (p *PrometheusSink) IncrCounter(parts []string, val float32) { + p.mu.Lock() + defer p.mu.Unlock() + key := p.flattenKey(parts) + g, ok := p.counters[key] + if !ok { + g = prometheus.NewCounter(prometheus.CounterOpts{ + Name: key, + Help: key, + }) + prometheus.MustRegister(g) + p.counters[key] = g + } + g.Add(float64(val)) +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/sink.go b/Godeps/_workspace/src/github.com/armon/go-metrics/sink.go new file mode 100644 index 0000000000000..0c240c2c47ee8 --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/sink.go @@ -0,0 +1,52 @@ +package metrics + +// The MetricSink interface is used to transmit metrics information +// to an external system +type MetricSink interface { + // A Gauge should retain the last value it is set to + SetGauge(key []string, val float32) + + // Should emit a Key/Value pair for each call + EmitKey(key []string, val float32) + + // Counters should accumulate values + IncrCounter(key []string, val float32) + + // Samples are for timing information, where quantiles are used + AddSample(key []string, val float32) +} + +// BlackholeSink is used to just blackhole messages +type BlackholeSink struct{} + +func (*BlackholeSink) SetGauge(key []string, val float32) {} +func (*BlackholeSink) EmitKey(key []string, val float32) {} +func (*BlackholeSink) IncrCounter(key []string, val float32) {} +func (*BlackholeSink) AddSample(key []string, val float32) {} + +// FanoutSink is used to sink to fanout values to multiple sinks +type FanoutSink []MetricSink + +func (fh FanoutSink) SetGauge(key []string, val float32) { + for _, s := range fh { + s.SetGauge(key, val) + } +} + +func (fh FanoutSink) EmitKey(key []string, val float32) { + for _, s := range fh { + s.EmitKey(key, val) + } +} + +func (fh FanoutSink) IncrCounter(key []string, val float32) { + for _, s := range fh { + s.IncrCounter(key, val) + } +} + +func (fh FanoutSink) AddSample(key []string, val float32) { + for _, s := range fh { + s.AddSample(key, val) + } +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/sink_test.go b/Godeps/_workspace/src/github.com/armon/go-metrics/sink_test.go new file mode 100644 index 0000000000000..15c5d771aa9e2 --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/sink_test.go @@ -0,0 +1,120 @@ +package metrics + +import ( + "reflect" + "testing" +) + +type MockSink struct { + keys [][]string + vals []float32 +} + +func (m *MockSink) SetGauge(key []string, val float32) { + m.keys = append(m.keys, key) + m.vals = append(m.vals, val) +} +func (m *MockSink) EmitKey(key []string, val float32) { + m.keys = append(m.keys, key) + m.vals = append(m.vals, val) +} +func (m *MockSink) IncrCounter(key []string, val float32) { + m.keys = append(m.keys, key) + m.vals = append(m.vals, val) +} +func (m *MockSink) AddSample(key []string, val float32) { + m.keys = append(m.keys, key) + m.vals = append(m.vals, val) +} + +func TestFanoutSink_Gauge(t *testing.T) { + m1 := &MockSink{} + m2 := &MockSink{} + fh := &FanoutSink{m1, m2} + + k := []string{"test"} + v := float32(42.0) + fh.SetGauge(k, v) + + if !reflect.DeepEqual(m1.keys[0], k) { + t.Fatalf("key not equal") + } + if !reflect.DeepEqual(m2.keys[0], k) { + t.Fatalf("key not equal") + } + if !reflect.DeepEqual(m1.vals[0], v) { + t.Fatalf("val not equal") + } + if !reflect.DeepEqual(m2.vals[0], v) { + t.Fatalf("val not equal") + } +} + +func TestFanoutSink_Key(t *testing.T) { + m1 := &MockSink{} + m2 := &MockSink{} + fh := &FanoutSink{m1, m2} + + k := []string{"test"} + v := float32(42.0) + fh.EmitKey(k, v) + + if !reflect.DeepEqual(m1.keys[0], k) { + t.Fatalf("key not equal") + } + if !reflect.DeepEqual(m2.keys[0], k) { + t.Fatalf("key not equal") + } + if !reflect.DeepEqual(m1.vals[0], v) { + t.Fatalf("val not equal") + } + if !reflect.DeepEqual(m2.vals[0], v) { + t.Fatalf("val not equal") + } +} + +func TestFanoutSink_Counter(t *testing.T) { + m1 := &MockSink{} + m2 := &MockSink{} + fh := &FanoutSink{m1, m2} + + k := []string{"test"} + v := float32(42.0) + fh.IncrCounter(k, v) + + if !reflect.DeepEqual(m1.keys[0], k) { + t.Fatalf("key not equal") + } + if !reflect.DeepEqual(m2.keys[0], k) { + t.Fatalf("key not equal") + } + if !reflect.DeepEqual(m1.vals[0], v) { + t.Fatalf("val not equal") + } + if !reflect.DeepEqual(m2.vals[0], v) { + t.Fatalf("val not equal") + } +} + +func TestFanoutSink_Sample(t *testing.T) { + m1 := &MockSink{} + m2 := &MockSink{} + fh := &FanoutSink{m1, m2} + + k := []string{"test"} + v := float32(42.0) + fh.AddSample(k, v) + + if !reflect.DeepEqual(m1.keys[0], k) { + t.Fatalf("key not equal") + } + if !reflect.DeepEqual(m2.keys[0], k) { + t.Fatalf("key not equal") + } + if !reflect.DeepEqual(m1.vals[0], v) { + t.Fatalf("val not equal") + } + if !reflect.DeepEqual(m2.vals[0], v) { + t.Fatalf("val not equal") + } +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/start.go b/Godeps/_workspace/src/github.com/armon/go-metrics/start.go new file mode 100644 index 0000000000000..44113f100426b --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/start.go @@ -0,0 +1,95 @@ +package metrics + +import ( + "os" + "time" +) + +// Config is used to configure metrics settings +type Config struct { + ServiceName string // Prefixed with keys to seperate services + HostName string // Hostname to use. If not provided and EnableHostname, it will be os.Hostname + EnableHostname bool // Enable prefixing gauge values with hostname + EnableRuntimeMetrics bool // Enables profiling of runtime metrics (GC, Goroutines, Memory) + EnableTypePrefix bool // Prefixes key with a type ("counter", "gauge", "timer") + TimerGranularity time.Duration // Granularity of timers. + ProfileInterval time.Duration // Interval to profile runtime metrics +} + +// Metrics represents an instance of a metrics sink that can +// be used to emit +type Metrics struct { + Config + lastNumGC uint32 + sink MetricSink +} + +// Shared global metrics instance +var globalMetrics *Metrics + +func init() { + // Initialize to a blackhole sink to avoid errors + globalMetrics = &Metrics{sink: &BlackholeSink{}} +} + +// DefaultConfig provides a sane default configuration +func DefaultConfig(serviceName string) *Config { + c := &Config{ + ServiceName: serviceName, // Use client provided service + HostName: "", + EnableHostname: true, // Enable hostname prefix + EnableRuntimeMetrics: true, // Enable runtime profiling + EnableTypePrefix: false, // Disable type prefix + TimerGranularity: time.Millisecond, // Timers are in milliseconds + ProfileInterval: time.Second, // Poll runtime every second + } + + // Try to get the hostname + name, _ := os.Hostname() + c.HostName = name + return c +} + +// New is used to create a new instance of Metrics +func New(conf *Config, sink MetricSink) (*Metrics, error) { + met := &Metrics{} + met.Config = *conf + met.sink = sink + + // Start the runtime collector + if conf.EnableRuntimeMetrics { + go met.collectStats() + } + return met, nil +} + +// NewGlobal is the same as New, but it assigns the metrics object to be +// used globally as well as returning it. +func NewGlobal(conf *Config, sink MetricSink) (*Metrics, error) { + metrics, err := New(conf, sink) + if err == nil { + globalMetrics = metrics + } + return metrics, err +} + +// Proxy all the methods to the globalMetrics instance +func SetGauge(key []string, val float32) { + globalMetrics.SetGauge(key, val) +} + +func EmitKey(key []string, val float32) { + globalMetrics.EmitKey(key, val) +} + +func IncrCounter(key []string, val float32) { + globalMetrics.IncrCounter(key, val) +} + +func AddSample(key []string, val float32) { + globalMetrics.AddSample(key, val) +} + +func MeasureSince(key []string, start time.Time) { + globalMetrics.MeasureSince(key, start) +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/start_test.go b/Godeps/_workspace/src/github.com/armon/go-metrics/start_test.go new file mode 100644 index 0000000000000..8b3210c15fe2e --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/start_test.go @@ -0,0 +1,110 @@ +package metrics + +import ( + "reflect" + "testing" + "time" +) + +func TestDefaultConfig(t *testing.T) { + conf := DefaultConfig("service") + if conf.ServiceName != "service" { + t.Fatalf("Bad name") + } + if conf.HostName == "" { + t.Fatalf("missing hostname") + } + if !conf.EnableHostname || !conf.EnableRuntimeMetrics { + t.Fatalf("expect true") + } + if conf.EnableTypePrefix { + t.Fatalf("expect false") + } + if conf.TimerGranularity != time.Millisecond { + t.Fatalf("bad granularity") + } + if conf.ProfileInterval != time.Second { + t.Fatalf("bad interval") + } +} + +func Test_GlobalMetrics_SetGauge(t *testing.T) { + m := &MockSink{} + globalMetrics = &Metrics{sink: m} + + k := []string{"test"} + v := float32(42.0) + SetGauge(k, v) + + if !reflect.DeepEqual(m.keys[0], k) { + t.Fatalf("key not equal") + } + if !reflect.DeepEqual(m.vals[0], v) { + t.Fatalf("val not equal") + } +} + +func Test_GlobalMetrics_EmitKey(t *testing.T) { + m := &MockSink{} + globalMetrics = &Metrics{sink: m} + + k := []string{"test"} + v := float32(42.0) + EmitKey(k, v) + + if !reflect.DeepEqual(m.keys[0], k) { + t.Fatalf("key not equal") + } + if !reflect.DeepEqual(m.vals[0], v) { + t.Fatalf("val not equal") + } +} + +func Test_GlobalMetrics_IncrCounter(t *testing.T) { + m := &MockSink{} + globalMetrics = &Metrics{sink: m} + + k := []string{"test"} + v := float32(42.0) + IncrCounter(k, v) + + if !reflect.DeepEqual(m.keys[0], k) { + t.Fatalf("key not equal") + } + if !reflect.DeepEqual(m.vals[0], v) { + t.Fatalf("val not equal") + } +} + +func Test_GlobalMetrics_AddSample(t *testing.T) { + m := &MockSink{} + globalMetrics = &Metrics{sink: m} + + k := []string{"test"} + v := float32(42.0) + AddSample(k, v) + + if !reflect.DeepEqual(m.keys[0], k) { + t.Fatalf("key not equal") + } + if !reflect.DeepEqual(m.vals[0], v) { + t.Fatalf("val not equal") + } +} + +func Test_GlobalMetrics_MeasureSince(t *testing.T) { + m := &MockSink{} + globalMetrics = &Metrics{sink: m} + globalMetrics.TimerGranularity = time.Millisecond + + k := []string{"test"} + now := time.Now() + MeasureSince(k, now) + + if !reflect.DeepEqual(m.keys[0], k) { + t.Fatalf("key not equal") + } + if m.vals[0] > 0.1 { + t.Fatalf("val too large %v", m.vals[0]) + } +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/statsd.go b/Godeps/_workspace/src/github.com/armon/go-metrics/statsd.go new file mode 100644 index 0000000000000..65a5021a05743 --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/statsd.go @@ -0,0 +1,154 @@ +package metrics + +import ( + "bytes" + "fmt" + "log" + "net" + "strings" + "time" +) + +const ( + // statsdMaxLen is the maximum size of a packet + // to send to statsd + statsdMaxLen = 1400 +) + +// StatsdSink provides a MetricSink that can be used +// with a statsite or statsd metrics server. It uses +// only UDP packets, while StatsiteSink uses TCP. +type StatsdSink struct { + addr string + metricQueue chan string +} + +// NewStatsdSink is used to create a new StatsdSink +func NewStatsdSink(addr string) (*StatsdSink, error) { + s := &StatsdSink{ + addr: addr, + metricQueue: make(chan string, 4096), + } + go s.flushMetrics() + return s, nil +} + +// Close is used to stop flushing to statsd +func (s *StatsdSink) Shutdown() { + close(s.metricQueue) +} + +func (s *StatsdSink) SetGauge(key []string, val float32) { + flatKey := s.flattenKey(key) + s.pushMetric(fmt.Sprintf("%s:%f|g\n", flatKey, val)) +} + +func (s *StatsdSink) EmitKey(key []string, val float32) { + flatKey := s.flattenKey(key) + s.pushMetric(fmt.Sprintf("%s:%f|kv\n", flatKey, val)) +} + +func (s *StatsdSink) IncrCounter(key []string, val float32) { + flatKey := s.flattenKey(key) + s.pushMetric(fmt.Sprintf("%s:%f|c\n", flatKey, val)) +} + +func (s *StatsdSink) AddSample(key []string, val float32) { + flatKey := s.flattenKey(key) + s.pushMetric(fmt.Sprintf("%s:%f|ms\n", flatKey, val)) +} + +// Flattens the key for formatting, removes spaces +func (s *StatsdSink) flattenKey(parts []string) string { + joined := strings.Join(parts, ".") + return strings.Map(func(r rune) rune { + switch r { + case ':': + fallthrough + case ' ': + return '_' + default: + return r + } + }, joined) +} + +// Does a non-blocking push to the metrics queue +func (s *StatsdSink) pushMetric(m string) { + select { + case s.metricQueue <- m: + default: + } +} + +// Flushes metrics +func (s *StatsdSink) flushMetrics() { + var sock net.Conn + var err error + var wait <-chan time.Time + ticker := time.NewTicker(flushInterval) + defer ticker.Stop() + +CONNECT: + // Create a buffer + buf := bytes.NewBuffer(nil) + + // Attempt to connect + sock, err = net.Dial("udp", s.addr) + if err != nil { + log.Printf("[ERR] Error connecting to statsd! Err: %s", err) + goto WAIT + } + + for { + select { + case metric, ok := <-s.metricQueue: + // Get a metric from the queue + if !ok { + goto QUIT + } + + // Check if this would overflow the packet size + if len(metric)+buf.Len() > statsdMaxLen { + _, err := sock.Write(buf.Bytes()) + buf.Reset() + if err != nil { + log.Printf("[ERR] Error writing to statsd! Err: %s", err) + goto WAIT + } + } + + // Append to the buffer + buf.WriteString(metric) + + case <-ticker.C: + if buf.Len() == 0 { + continue + } + + _, err := sock.Write(buf.Bytes()) + buf.Reset() + if err != nil { + log.Printf("[ERR] Error flushing to statsd! Err: %s", err) + goto WAIT + } + } + } + +WAIT: + // Wait for a while + wait = time.After(time.Duration(5) * time.Second) + for { + select { + // Dequeue the messages to avoid backlog + case _, ok := <-s.metricQueue: + if !ok { + goto QUIT + } + case <-wait: + goto CONNECT + } + } +QUIT: + s.metricQueue = nil +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/statsd_test.go b/Godeps/_workspace/src/github.com/armon/go-metrics/statsd_test.go new file mode 100644 index 0000000000000..622eb5d3aad2a --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/statsd_test.go @@ -0,0 +1,105 @@ +package metrics + +import ( + "bufio" + "bytes" + "net" + "testing" + "time" +) + +func TestStatsd_Flatten(t *testing.T) { + s := &StatsdSink{} + flat := s.flattenKey([]string{"a", "b", "c", "d"}) + if flat != "a.b.c.d" { + t.Fatalf("Bad flat") + } +} + +func TestStatsd_PushFullQueue(t *testing.T) { + q := make(chan string, 1) + q <- "full" + + s := &StatsdSink{metricQueue: q} + s.pushMetric("omit") + + out := <-q + if out != "full" { + t.Fatalf("bad val %v", out) + } + + select { + case v := <-q: + t.Fatalf("bad val %v", v) + default: + } +} + +func TestStatsd_Conn(t *testing.T) { + addr := "127.0.0.1:7524" + done := make(chan bool) + go func() { + list, err := net.ListenUDP("udp", &net.UDPAddr{IP: net.ParseIP("127.0.0.1"), Port: 7524}) + if err != nil { + panic(err) + } + defer list.Close() + buf := make([]byte, 1500) + n, err := list.Read(buf) + if err != nil { + panic(err) + } + buf = buf[:n] + reader := bufio.NewReader(bytes.NewReader(buf)) + + line, err := reader.ReadString('\n') + if err != nil { + t.Fatalf("unexpected err %s", err) + } + if line != "gauge.val:1.000000|g\n" { + t.Fatalf("bad line %s", line) + } + + line, err = reader.ReadString('\n') + if err != nil { + t.Fatalf("unexpected err %s", err) + } + if line != "key.other:2.000000|kv\n" { + t.Fatalf("bad line %s", line) + } + + line, err = reader.ReadString('\n') + if err != nil { + t.Fatalf("unexpected err %s", err) + } + if line != "counter.me:3.000000|c\n" { + t.Fatalf("bad line %s", line) + } + + line, err = reader.ReadString('\n') + if err != nil { + t.Fatalf("unexpected err %s", err) + } + if line != "sample.slow_thingy:4.000000|ms\n" { + t.Fatalf("bad line %s", line) + } + + done <- true + }() + s, err := NewStatsdSink(addr) + if err != nil { + t.Fatalf("bad error") + } + + s.SetGauge([]string{"gauge", "val"}, float32(1)) + s.EmitKey([]string{"key", "other"}, float32(2)) + s.IncrCounter([]string{"counter", "me"}, float32(3)) + s.AddSample([]string{"sample", "slow thingy"}, float32(4)) + + select { + case <-done: + s.Shutdown() + case <-time.After(3 * time.Second): + t.Fatalf("timeout") + } +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/statsite.go b/Godeps/_workspace/src/github.com/armon/go-metrics/statsite.go new file mode 100644 index 0000000000000..68730139a73a7 --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/statsite.go @@ -0,0 +1,142 @@ +package metrics + +import ( + "bufio" + "fmt" + "log" + "net" + "strings" + "time" +) + +const ( + // We force flush the statsite metrics after this period of + // inactivity. Prevents stats from getting stuck in a buffer + // forever. + flushInterval = 100 * time.Millisecond +) + +// StatsiteSink provides a MetricSink that can be used with a +// statsite metrics server +type StatsiteSink struct { + addr string + metricQueue chan string +} + +// NewStatsiteSink is used to create a new StatsiteSink +func NewStatsiteSink(addr string) (*StatsiteSink, error) { + s := &StatsiteSink{ + addr: addr, + metricQueue: make(chan string, 4096), + } + go s.flushMetrics() + return s, nil +} + +// Close is used to stop flushing to statsite +func (s *StatsiteSink) Shutdown() { + close(s.metricQueue) +} + +func (s *StatsiteSink) SetGauge(key []string, val float32) { + flatKey := s.flattenKey(key) + s.pushMetric(fmt.Sprintf("%s:%f|g\n", flatKey, val)) +} + +func (s *StatsiteSink) EmitKey(key []string, val float32) { + flatKey := s.flattenKey(key) + s.pushMetric(fmt.Sprintf("%s:%f|kv\n", flatKey, val)) +} + +func (s *StatsiteSink) IncrCounter(key []string, val float32) { + flatKey := s.flattenKey(key) + s.pushMetric(fmt.Sprintf("%s:%f|c\n", flatKey, val)) +} + +func (s *StatsiteSink) AddSample(key []string, val float32) { + flatKey := s.flattenKey(key) + s.pushMetric(fmt.Sprintf("%s:%f|ms\n", flatKey, val)) +} + +// Flattens the key for formatting, removes spaces +func (s *StatsiteSink) flattenKey(parts []string) string { + joined := strings.Join(parts, ".") + return strings.Map(func(r rune) rune { + switch r { + case ':': + fallthrough + case ' ': + return '_' + default: + return r + } + }, joined) +} + +// Does a non-blocking push to the metrics queue +func (s *StatsiteSink) pushMetric(m string) { + select { + case s.metricQueue <- m: + default: + } +} + +// Flushes metrics +func (s *StatsiteSink) flushMetrics() { + var sock net.Conn + var err error + var wait <-chan time.Time + var buffered *bufio.Writer + ticker := time.NewTicker(flushInterval) + defer ticker.Stop() + +CONNECT: + // Attempt to connect + sock, err = net.Dial("tcp", s.addr) + if err != nil { + log.Printf("[ERR] Error connecting to statsite! Err: %s", err) + goto WAIT + } + + // Create a buffered writer + buffered = bufio.NewWriter(sock) + + for { + select { + case metric, ok := <-s.metricQueue: + // Get a metric from the queue + if !ok { + goto QUIT + } + + // Try to send to statsite + _, err := buffered.Write([]byte(metric)) + if err != nil { + log.Printf("[ERR] Error writing to statsite! Err: %s", err) + goto WAIT + } + case <-ticker.C: + if err := buffered.Flush(); err != nil { + log.Printf("[ERR] Error flushing to statsite! Err: %s", err) + goto WAIT + } + } + } + +WAIT: + // Wait for a while + wait = time.After(time.Duration(5) * time.Second) + for { + select { + // Dequeue the messages to avoid backlog + case _, ok := <-s.metricQueue: + if !ok { + goto QUIT + } + case <-wait: + goto CONNECT + } + } +QUIT: + s.metricQueue = nil +} diff --git a/Godeps/_workspace/src/github.com/armon/go-metrics/statsite_test.go b/Godeps/_workspace/src/github.com/armon/go-metrics/statsite_test.go new file mode 100644 index 0000000000000..d9c744f4164f9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/armon/go-metrics/statsite_test.go @@ -0,0 +1,101 @@ +package metrics + +import ( + "bufio" + "net" + "testing" + "time" +) + +func acceptConn(addr string) net.Conn { + ln, _ := net.Listen("tcp", addr) + conn, _ := ln.Accept() + return conn +} + +func TestStatsite_Flatten(t *testing.T) { + s := &StatsiteSink{} + flat := s.flattenKey([]string{"a", "b", "c", "d"}) + if flat != "a.b.c.d" { + t.Fatalf("Bad flat") + } +} + +func TestStatsite_PushFullQueue(t *testing.T) { + q := make(chan string, 1) + q <- "full" + + s := &StatsiteSink{metricQueue: q} + s.pushMetric("omit") + + out := <-q + if out != "full" { + t.Fatalf("bad val %v", out) + } + + select { + case v := <-q: + t.Fatalf("bad val %v", v) + default: + } +} + +func TestStatsite_Conn(t *testing.T) { + addr := "localhost:7523" + done := make(chan bool) + go func() { + conn := acceptConn(addr) + reader := bufio.NewReader(conn) + + line, err := reader.ReadString('\n') + if err != nil { + t.Fatalf("unexpected err %s", err) + } + if line != "gauge.val:1.000000|g\n" { + t.Fatalf("bad line %s", line) + } + + line, err = reader.ReadString('\n') + if err != nil { + t.Fatalf("unexpected err %s", err) + } + if line != "key.other:2.000000|kv\n" { + t.Fatalf("bad line %s", line) + } + + line, err = reader.ReadString('\n') + if err != nil { + t.Fatalf("unexpected err %s", err) + } + if line != "counter.me:3.000000|c\n" { + t.Fatalf("bad line %s", line) + } + + line, err = reader.ReadString('\n') + if err != nil { + t.Fatalf("unexpected err %s", err) + } + if line != "sample.slow_thingy:4.000000|ms\n" { + t.Fatalf("bad line %s", line) + } + + conn.Close() + done <- true + }() + s, err := NewStatsiteSink(addr) + if err != nil { + t.Fatalf("bad error") + } + + s.SetGauge([]string{"gauge", "val"}, float32(1)) + s.EmitKey([]string{"key", "other"}, float32(2)) + s.IncrCounter([]string{"counter", "me"}, float32(3)) + s.AddSample([]string{"sample", "slow thingy"}, float32(4)) + + select { + case <-done: + s.Shutdown() + case <-time.After(3 * time.Second): + t.Fatalf("timeout") + } +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/.gitignore b/Godeps/_workspace/src/github.com/boltdb/bolt/.gitignore new file mode 100644 index 0000000000000..c7bd2b7a5b84c --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/.gitignore @@ -0,0 +1,4 @@ +*.prof +*.test +*.swp +/bin/ diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/LICENSE b/Godeps/_workspace/src/github.com/boltdb/bolt/LICENSE new file mode 100644 index 0000000000000..004e77fe5d2ec --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/LICENSE @@ -0,0 +1,20 @@ +The MIT License (MIT) + +Copyright (c) 2013 Ben Johnson + +Permission is hereby granted, free of charge, to any person obtaining a copy of +this software and associated documentation files (the "Software"), to deal in +the Software without restriction, including without limitation the rights to +use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of +the Software, and to permit persons to whom the Software is furnished to do so, +subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS +FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR +COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER +IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN +CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/Makefile b/Godeps/_workspace/src/github.com/boltdb/bolt/Makefile new file mode 100644 index 0000000000000..cfbed514bbbc5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/Makefile @@ -0,0 +1,54 @@ +TEST=. +BENCH=. +COVERPROFILE=/tmp/c.out +BRANCH=`git rev-parse --abbrev-ref HEAD` +COMMIT=`git rev-parse --short HEAD` +GOLDFLAGS="-X main.branch $(BRANCH) -X main.commit $(COMMIT)" + +default: build + +bench: + go test -v -test.run=NOTHINCONTAINSTHIS -test.bench=$(BENCH) + +# http://cloc.sourceforge.net/ +cloc: + @cloc --not-match-f='Makefile|_test.go' . + +cover: fmt + go test -coverprofile=$(COVERPROFILE) -test.run=$(TEST) $(COVERFLAG) . + go tool cover -html=$(COVERPROFILE) + rm $(COVERPROFILE) + +cpuprofile: fmt + @go test -c + @./bolt.test -test.v -test.run=$(TEST) -test.cpuprofile cpu.prof + +# go get github.com/kisielk/errcheck +errcheck: + @echo "=== errcheck ===" + @errcheck github.com/boltdb/bolt + +fmt: + @go fmt ./... + +get: + @go get -d ./... + +build: get + @mkdir -p bin + @go build -ldflags=$(GOLDFLAGS) -a -o bin/bolt ./cmd/bolt + +test: fmt + @go get github.com/stretchr/testify/assert + @echo "=== TESTS ===" + @go test -v -cover -test.run=$(TEST) + @echo "" + @echo "" + @echo "=== CLI ===" + @go test -v -test.run=$(TEST) ./cmd/bolt + @echo "" + @echo "" + @echo "=== RACE DETECTOR ===" + @go test -v -race -test.run="TestSimulate_(100op|1000op)" + +.PHONY: bench cloc cover cpuprofile fmt memprofile test diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/README.md b/Godeps/_workspace/src/github.com/boltdb/bolt/README.md new file mode 100644 index 0000000000000..00fad6afb8b57 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/README.md @@ -0,0 +1,621 @@ +Bolt [![Build Status](https://drone.io/github.com/boltdb/bolt/status.png)](https://drone.io/github.com/boltdb/bolt/latest) [![Coverage Status](https://coveralls.io/repos/boltdb/bolt/badge.png?branch=master)](https://coveralls.io/r/boltdb/bolt?branch=master) [![GoDoc](https://godoc.org/github.com/boltdb/bolt?status.png)](https://godoc.org/github.com/boltdb/bolt) ![Version](http://img.shields.io/badge/version-1.0-green.png) +==== + +Bolt is a pure Go key/value store inspired by [Howard Chu's][hyc_symas] and +the [LMDB project][lmdb]. The goal of the project is to provide a simple, +fast, and reliable database for projects that don't require a full database +server such as Postgres or MySQL. + +Since Bolt is meant to be used as such a low-level piece of functionality, +simplicity is key. The API will be small and only focus on getting values +and setting values. That's it. + +[hyc_symas]: https://twitter.com/hyc_symas +[lmdb]: http://symas.com/mdb/ + + +## Project Status + +Bolt is stable and the API is fixed. Full unit test coverage and randomized +black box testing are used to ensure database consistency and thread safety. +Bolt is currently in high-load production environments serving databases as +large as 1TB. Many companies such as Shopify and Heroku use Bolt-backed +services every day. + + +## Getting Started + +### Installing + +To start using Bolt, install Go and run `go get`: + +```sh +$ go get github.com/boltdb/bolt/... +``` + +This will retrieve the library and install the `bolt` command line utility into +your `$GOBIN` path. + + +### Opening a database + +The top-level object in Bolt is a `DB`. It is represented as a single file on +your disk and represents a consistent snapshot of your data. + +To open your database, simply use the `bolt.Open()` function: + +```go +package main + +import ( + "log" + + "github.com/boltdb/bolt" +) + +func main() { + // Open the my.db data file in your current directory. + // It will be created if it doesn't exist. + db, err := bolt.Open("my.db", 0600, nil) + if err != nil { + log.Fatal(err) + } + defer db.Close() + + ... +} +``` + +Please note that Bolt obtains a file lock on the data file so multiple processes +cannot open the same database at the same time. Opening an already open Bolt +database will cause it to hang until the other process closes it. To prevent +an indefinite wait you can pass a timeout option to the `Open()` function: + +```go +db, err := bolt.Open("my.db", 0600, &bolt.Options{Timeout: 1 * time.Second}) +``` + + +### Transactions + +Bolt allows only one read-write transaction at a time but allows as many +read-only transactions as you want at a time. Each transaction has a consistent +view of the data as it existed when the transaction started. + +Individual transactions and all objects created from them (e.g. buckets, keys) +are not thread safe. To work with data in multiple goroutines you must start +a transaction for each one or use locking to ensure only one goroutine accesses +a transaction at a time. Creating transaction from the `DB` is thread safe. + +Read-only transactions and read-write transactions should not depend on one +another and generally shouldn't be opened simultaneously in the same goroutine. +This can cause a deadlock as the read-write transaction needs to periodically +re-map the data file but it cannot do so while a read-only transaction is open. + + +#### Read-write transactions + +To start a read-write transaction, you can use the `DB.Update()` function: + +```go +err := db.Update(func(tx *bolt.Tx) error { + ... + return nil +}) +``` + +Inside the closure, you have a consistent view of the database. You commit the +transaction by returning `nil` at the end. You can also rollback the transaction +at any point by returning an error. All database operations are allowed inside +a read-write transaction. + +Always check the return error as it will report any disk failures that can cause +your transaction to not complete. If you return an error within your closure +it will be passed through. + + +#### Read-only transactions + +To start a read-only transaction, you can use the `DB.View()` function: + +```go +err := db.View(func(tx *bolt.Tx) error { + ... + return nil +}) +``` + +You also get a consistent view of the database within this closure, however, +no mutating operations are allowed within a read-only transaction. You can only +retrieve buckets, retrieve values, and copy the database within a read-only +transaction. + + +#### Batch read-write transactions + +Each `DB.Update()` waits for disk to commit the writes. This overhead +can be minimized by combining multiple updates with the `DB.Batch()` +function: + +```go +err := db.Batch(func(tx *bolt.Tx) error { + ... + return nil +}) +``` + +Concurrent Batch calls are opportunistically combined into larger +transactions. Batch is only useful when there are multiple goroutines +calling it. + +The trade-off is that `Batch` can call the given +function multiple times, if parts of the transaction fail. The +function must be idempotent and side effects must take effect only +after a successful return from `DB.Batch()`. + +For example: don't display messages from inside the function, instead +set variables in the enclosing scope: + +```go +var id uint64 +err := db.Batch(func(tx *bolt.Tx) error { + // Find last key in bucket, decode as bigendian uint64, increment + // by one, encode back to []byte, and add new key. + ... + id = newValue + return nil +}) +if err != nil { + return ... +} +fmt.Println("Allocated ID %d", id) +``` + + +#### Managing transactions manually + +The `DB.View()` and `DB.Update()` functions are wrappers around the `DB.Begin()` +function. These helper functions will start the transaction, execute a function, +and then safely close your transaction if an error is returned. This is the +recommended way to use Bolt transactions. + +However, sometimes you may want to manually start and end your transactions. +You can use the `Tx.Begin()` function directly but _please_ be sure to close the +transaction. + +```go +// Start a writable transaction. +tx, err := db.Begin(true) +if err != nil { + return err +} +defer tx.Rollback() + +// Use the transaction... +_, err := tx.CreateBucket([]byte("MyBucket")) +if err != nil { + return err +} + +// Commit the transaction and check for error. +if err := tx.Commit(); err != nil { + return err +} +``` + +The first argument to `DB.Begin()` is a boolean stating if the transaction +should be writable. + + +### Using buckets + +Buckets are collections of key/value pairs within the database. All keys in a +bucket must be unique. You can create a bucket using the `DB.CreateBucket()` +function: + +```go +db.Update(func(tx *bolt.Tx) error { + b, err := tx.CreateBucket([]byte("MyBucket")) + if err != nil { + return fmt.Errorf("create bucket: %s", err) + } + return nil +}) +``` + +You can also create a bucket only if it doesn't exist by using the +`Tx.CreateBucketIfNotExists()` function. It's a common pattern to call this +function for all your top-level buckets after you open your database so you can +guarantee that they exist for future transactions. + +To delete a bucket, simply call the `Tx.DeleteBucket()` function. + + +### Using key/value pairs + +To save a key/value pair to a bucket, use the `Bucket.Put()` function: + +```go +db.Update(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("MyBucket")) + err := b.Put([]byte("answer"), []byte("42")) + return err +}) +``` + +This will set the value of the `"answer"` key to `"42"` in the `MyBucket` +bucket. To retrieve this value, we can use the `Bucket.Get()` function: + +```go +db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("MyBucket")) + v := b.Get([]byte("answer")) + fmt.Printf("The answer is: %s\n", v) + return nil +}) +``` + +The `Get()` function does not return an error because its operation is +guarenteed to work (unless there is some kind of system failure). If the key +exists then it will return its byte slice value. If it doesn't exist then it +will return `nil`. It's important to note that you can have a zero-length value +set to a key which is different than the key not existing. + +Use the `Bucket.Delete()` function to delete a key from the bucket. + +Please note that values returned from `Get()` are only valid while the +transaction is open. If you need to use a value outside of the transaction +then you must use `copy()` to copy it to another byte slice. + + +### Iterating over keys + +Bolt stores its keys in byte-sorted order within a bucket. This makes sequential +iteration over these keys extremely fast. To iterate over keys we'll use a +`Cursor`: + +```go +db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("MyBucket")) + c := b.Cursor() + + for k, v := c.First(); k != nil; k, v = c.Next() { + fmt.Printf("key=%s, value=%s\n", k, v) + } + + return nil +}) +``` + +The cursor allows you to move to a specific point in the list of keys and move +forward or backward through the keys one at a time. + +The following functions are available on the cursor: + +``` +First() Move to the first key. +Last() Move to the last key. +Seek() Move to a specific key. +Next() Move to the next key. +Prev() Move to the previous key. +``` + +When you have iterated to the end of the cursor then `Next()` will return `nil`. +You must seek to a position using `First()`, `Last()`, or `Seek()` before +calling `Next()` or `Prev()`. If you do not seek to a position then these +functions will return `nil`. + + +#### Prefix scans + +To iterate over a key prefix, you can combine `Seek()` and `bytes.HasPrefix()`: + +```go +db.View(func(tx *bolt.Tx) error { + c := tx.Bucket([]byte("MyBucket")).Cursor() + + prefix := []byte("1234") + for k, v := c.Seek(prefix); bytes.HasPrefix(k, prefix); k, v = c.Next() { + fmt.Printf("key=%s, value=%s\n", k, v) + } + + return nil +}) +``` + +#### Range scans + +Another common use case is scanning over a range such as a time range. If you +use a sortable time encoding such as RFC3339 then you can query a specific +date range like this: + +```go +db.View(func(tx *bolt.Tx) error { + // Assume our events bucket has RFC3339 encoded time keys. + c := tx.Bucket([]byte("Events")).Cursor() + + // Our time range spans the 90's decade. + min := []byte("1990-01-01T00:00:00Z") + max := []byte("2000-01-01T00:00:00Z") + + // Iterate over the 90's. + for k, v := c.Seek(min); k != nil && bytes.Compare(k, max) <= 0; k, v = c.Next() { + fmt.Printf("%s: %s\n", k, v) + } + + return nil +}) +``` + + +#### ForEach() + +You can also use the function `ForEach()` if you know you'll be iterating over +all the keys in a bucket: + +```go +db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("MyBucket")) + b.ForEach(func(k, v []byte) error { + fmt.Printf("key=%s, value=%s\n", k, v) + return nil + }) + return nil +}) +``` + + +### Nested buckets + +You can also store a bucket in a key to create nested buckets. The API is the +same as the bucket management API on the `DB` object: + +```go +func (*Bucket) CreateBucket(key []byte) (*Bucket, error) +func (*Bucket) CreateBucketIfNotExists(key []byte) (*Bucket, error) +func (*Bucket) DeleteBucket(key []byte) error +``` + + +### Database backups + +Bolt is a single file so it's easy to backup. You can use the `Tx.WriteTo()` +function to write a consistent view of the database to a writer. If you call +this from a read-only transaction, it will perform a hot backup and not block +your other database reads and writes. It will also use `O_DIRECT` when available +to prevent page cache trashing. + +One common use case is to backup over HTTP so you can use tools like `cURL` to +do database backups: + +```go +func BackupHandleFunc(w http.ResponseWriter, req *http.Request) { + err := db.View(func(tx *bolt.Tx) error { + w.Header().Set("Content-Type", "application/octet-stream") + w.Header().Set("Content-Disposition", `attachment; filename="my.db"`) + w.Header().Set("Content-Length", strconv.Itoa(int(tx.Size()))) + _, err := tx.WriteTo(w) + return err + }) + if err != nil { + http.Error(w, err.Error(), http.StatusInternalServerError) + } +} +``` + +Then you can backup using this command: + +```sh +$ curl http://localhost/backup > my.db +``` + +Or you can open your browser to `http://localhost/backup` and it will download +automatically. + +If you want to backup to another file you can use the `Tx.CopyFile()` helper +function. + + +### Statistics + +The database keeps a running count of many of the internal operations it +performs so you can better understand what's going on. By grabbing a snapshot +of these stats at two points in time we can see what operations were performed +in that time range. + +For example, we could start a goroutine to log stats every 10 seconds: + +```go +go func() { + // Grab the initial stats. + prev := db.Stats() + + for { + // Wait for 10s. + time.Sleep(10 * time.Second) + + // Grab the current stats and diff them. + stats := db.Stats() + diff := stats.Sub(&prev) + + // Encode stats to JSON and print to STDERR. + json.NewEncoder(os.Stderr).Encode(diff) + + // Save stats for the next loop. + prev = stats + } +}() +``` + +It's also useful to pipe these stats to a service such as statsd for monitoring +or to provide an HTTP endpoint that will perform a fixed-length sample. + + +### Read-Only Mode + +Sometimes it is useful to create a shared, read-only Bolt database. To this, +set the `Options.ReadOnly` flag when opening your database. Read-only mode +uses a shared lock to allow multiple processes to read from the database but +it will block any processes from opening the database in read-write mode. + +```go +db, err := bolt.Open("my.db", 0666, &bolt.Options{ReadOnly: true}) +if err != nil { + log.Fatal(err) +} +``` + + +## Resources + +For more information on getting started with Bolt, check out the following articles: + +* [Intro to BoltDB: Painless Performant Persistence](http://npf.io/2014/07/intro-to-boltdb-painless-performant-persistence/) by [Nate Finch](https://github.com/natefinch). +* [Bolt -- an embedded key/value database for Go](https://www.progville.com/go/bolt-embedded-db-golang/) by Progville + + +## Comparison with other databases + +### Postgres, MySQL, & other relational databases + +Relational databases structure data into rows and are only accessible through +the use of SQL. This approach provides flexibility in how you store and query +your data but also incurs overhead in parsing and planning SQL statements. Bolt +accesses all data by a byte slice key. This makes Bolt fast to read and write +data by key but provides no built-in support for joining values together. + +Most relational databases (with the exception of SQLite) are standalone servers +that run separately from your application. This gives your systems +flexibility to connect multiple application servers to a single database +server but also adds overhead in serializing and transporting data over the +network. Bolt runs as a library included in your application so all data access +has to go through your application's process. This brings data closer to your +application but limits multi-process access to the data. + + +### LevelDB, RocksDB + +LevelDB and its derivatives (RocksDB, HyperLevelDB) are similar to Bolt in that +they are libraries bundled into the application, however, their underlying +structure is a log-structured merge-tree (LSM tree). An LSM tree optimizes +random writes by using a write ahead log and multi-tiered, sorted files called +SSTables. Bolt uses a B+tree internally and only a single file. Both approaches +have trade offs. + +If you require a high random write throughput (>10,000 w/sec) or you need to use +spinning disks then LevelDB could be a good choice. If your application is +read-heavy or does a lot of range scans then Bolt could be a good choice. + +One other important consideration is that LevelDB does not have transactions. +It supports batch writing of key/values pairs and it supports read snapshots +but it will not give you the ability to do a compare-and-swap operation safely. +Bolt supports fully serializable ACID transactions. + + +### LMDB + +Bolt was originally a port of LMDB so it is architecturally similar. Both use +a B+tree, have ACID semantics with fully serializable transactions, and support +lock-free MVCC using a single writer and multiple readers. + +The two projects have somewhat diverged. LMDB heavily focuses on raw performance +while Bolt has focused on simplicity and ease of use. For example, LMDB allows +several unsafe actions such as direct writes for the sake of performance. Bolt +opts to disallow actions which can leave the database in a corrupted state. The +only exception to this in Bolt is `DB.NoSync`. + +There are also a few differences in API. LMDB requires a maximum mmap size when +opening an `mdb_env` whereas Bolt will handle incremental mmap resizing +automatically. LMDB overloads the getter and setter functions with multiple +flags whereas Bolt splits these specialized cases into their own functions. + + +## Caveats & Limitations + +It's important to pick the right tool for the job and Bolt is no exception. +Here are a few things to note when evaluating and using Bolt: + +* Bolt is good for read intensive workloads. Sequential write performance is + also fast but random writes can be slow. You can add a write-ahead log or + [transaction coalescer](https://github.com/boltdb/coalescer) in front of Bolt + to mitigate this issue. + +* Bolt uses a B+tree internally so there can be a lot of random page access. + SSDs provide a significant performance boost over spinning disks. + +* Try to avoid long running read transactions. Bolt uses copy-on-write so + old pages cannot be reclaimed while an old transaction is using them. + +* Byte slices returned from Bolt are only valid during a transaction. Once the + transaction has been committed or rolled back then the memory they point to + can be reused by a new page or can be unmapped from virtual memory and you'll + see an `unexpected fault address` panic when accessing it. + +* Be careful when using `Bucket.FillPercent`. Setting a high fill percent for + buckets that have random inserts will cause your database to have very poor + page utilization. + +* Use larger buckets in general. Smaller buckets causes poor page utilization + once they become larger than the page size (typically 4KB). + +* Bulk loading a lot of random writes into a new bucket can be slow as the + page will not split until the transaction is committed. Randomly inserting + more than 100,000 key/value pairs into a single new bucket in a single + transaction is not advised. + +* Bolt uses a memory-mapped file so the underlying operating system handles the + caching of the data. Typically, the OS will cache as much of the file as it + can in memory and will release memory as needed to other processes. This means + that Bolt can show very high memory usage when working with large databases. + However, this is expected and the OS will release memory as needed. Bolt can + handle databases much larger than the available physical RAM. + +* The data structures in the Bolt database are memory mapped so the data file + will be endian specific. This means that you cannot copy a Bolt file from a + little endian machine to a big endian machine and have it work. For most + users this is not a concern since most modern CPUs are little endian. + +* Because of the way pages are laid out on disk, Bolt cannot truncate data files + and return free pages back to the disk. Instead, Bolt maintains a free list + of unused pages within its data file. These free pages can be reused by later + transactions. This works well for many use cases as databases generally tend + to grow. However, it's important to note that deleting large chunks of data + will not allow you to reclaim that space on disk. + + For more information on page allocation, [see this comment][page-allocation]. + +[page-allocation]: https://github.com/boltdb/bolt/issues/308#issuecomment-74811638 + + +## Other Projects Using Bolt + +Below is a list of public, open source projects that use Bolt: + +* [Operation Go: A Routine Mission](http://gocode.io) - An online programming game for Golang using Bolt for user accounts and a leaderboard. +* [Bazil](https://bazil.org/) - A file system that lets your data reside where it is most convenient for it to reside. +* [DVID](https://github.com/janelia-flyem/dvid) - Added Bolt as optional storage engine and testing it against Basho-tuned leveldb. +* [Skybox Analytics](https://github.com/skybox/skybox) - A standalone funnel analysis tool for web analytics. +* [Scuttlebutt](https://github.com/benbjohnson/scuttlebutt) - Uses Bolt to store and process all Twitter mentions of GitHub projects. +* [Wiki](https://github.com/peterhellberg/wiki) - A tiny wiki using Goji, BoltDB and Blackfriday. +* [ChainStore](https://github.com/nulayer/chainstore) - Simple key-value interface to a variety of storage engines organized as a chain of operations. +* [MetricBase](https://github.com/msiebuhr/MetricBase) - Single-binary version of Graphite. +* [Gitchain](https://github.com/gitchain/gitchain) - Decentralized, peer-to-peer Git repositories aka "Git meets Bitcoin". +* [event-shuttle](https://github.com/sclasen/event-shuttle) - A Unix system service to collect and reliably deliver messages to Kafka. +* [ipxed](https://github.com/kelseyhightower/ipxed) - Web interface and api for ipxed. +* [BoltStore](https://github.com/yosssi/boltstore) - Session store using Bolt. +* [photosite/session](http://godoc.org/bitbucket.org/kardianos/photosite/session) - Sessions for a photo viewing site. +* [LedisDB](https://github.com/siddontang/ledisdb) - A high performance NoSQL, using Bolt as optional storage. +* [ipLocator](https://github.com/AndreasBriese/ipLocator) - A fast ip-geo-location-server using bolt with bloom filters. +* [cayley](https://github.com/google/cayley) - Cayley is an open-source graph database using Bolt as optional backend. +* [bleve](http://www.blevesearch.com/) - A pure Go search engine similar to ElasticSearch that uses Bolt as the default storage backend. +* [tentacool](https://github.com/optiflows/tentacool) - REST api server to manage system stuff (IP, DNS, Gateway...) on a linux server. +* [SkyDB](https://github.com/skydb/sky) - Behavioral analytics database. +* [Seaweed File System](https://github.com/chrislusf/weed-fs) - Highly scalable distributed key~file system with O(1) disk read. +* [InfluxDB](http://influxdb.com) - Scalable datastore for metrics, events, and real-time analytics. +* [Freehold](http://tshannon.bitbucket.org/freehold/) - An open, secure, and lightweight platform for your files and data. +* [Prometheus Annotation Server](https://github.com/oliver006/prom_annotation_server) - Annotation server for PromDash & Prometheus service monitoring system. +* [Consul](https://github.com/hashicorp/consul) - Consul is service discovery and configuration made easy. Distributed, highly available, and datacenter-aware. +* [Kala](https://github.com/ajvb/kala) - Kala is a modern job scheduler optimized to run on a single node. It is persistant, JSON over HTTP API, ISO 8601 duration notation, and dependent jobs. +* [drive](https://github.com/odeke-em/drive) - drive is an unofficial Google Drive command line client for \*NIX operating systems. + +If you are using Bolt in a project please send a pull request to add it to the list. diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/batch.go b/Godeps/_workspace/src/github.com/boltdb/bolt/batch.go new file mode 100644 index 0000000000000..84acae6bbf089 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/batch.go @@ -0,0 +1,138 @@ +package bolt + +import ( + "errors" + "fmt" + "sync" + "time" +) + +// Batch calls fn as part of a batch. It behaves similar to Update, +// except: +// +// 1. concurrent Batch calls can be combined into a single Bolt +// transaction. +// +// 2. the function passed to Batch may be called multiple times, +// regardless of whether it returns error or not. +// +// This means that Batch function side effects must be idempotent and +// take permanent effect only after a successful return is seen in +// caller. +// +// The maximum batch size and delay can be adjusted with DB.MaxBatchSize +// and DB.MaxBatchDelay, respectively. +// +// Batch is only useful when there are multiple goroutines calling it. +func (db *DB) Batch(fn func(*Tx) error) error { + errCh := make(chan error, 1) + + db.batchMu.Lock() + if (db.batch == nil) || (db.batch != nil && len(db.batch.calls) >= db.MaxBatchSize) { + // There is no existing batch, or the existing batch is full; start a new one. + db.batch = &batch{ + db: db, + } + db.batch.timer = time.AfterFunc(db.MaxBatchDelay, db.batch.trigger) + } + db.batch.calls = append(db.batch.calls, call{fn: fn, err: errCh}) + if len(db.batch.calls) >= db.MaxBatchSize { + // wake up batch, it's ready to run + go db.batch.trigger() + } + db.batchMu.Unlock() + + err := <-errCh + if err == trySolo { + err = db.Update(fn) + } + return err +} + +type call struct { + fn func(*Tx) error + err chan<- error +} + +type batch struct { + db *DB + timer *time.Timer + start sync.Once + calls []call +} + +// trigger runs the batch if it hasn't already been run. +func (b *batch) trigger() { + b.start.Do(b.run) +} + +// run performs the transactions in the batch and communicates results +// back to DB.Batch. +func (b *batch) run() { + b.db.batchMu.Lock() + b.timer.Stop() + // Make sure no new work is added to this batch, but don't break + // other batches. + if b.db.batch == b { + b.db.batch = nil + } + b.db.batchMu.Unlock() + +retry: + for len(b.calls) > 0 { + var failIdx = -1 + err := b.db.Update(func(tx *Tx) error { + for i, c := range b.calls { + if err := safelyCall(c.fn, tx); err != nil { + failIdx = i + return err + } + } + return nil + }) + + if failIdx >= 0 { + // take the failing transaction out of the batch. it's + // safe to shorten b.calls here because db.batch no longer + // points to us, and we hold the mutex anyway. + c := b.calls[failIdx] + b.calls[failIdx], b.calls = b.calls[len(b.calls)-1], b.calls[:len(b.calls)-1] + // tell the submitter re-run it solo, continue with the rest of the batch + c.err <- trySolo + continue retry + } + + // pass success, or bolt internal errors, to all callers + for _, c := range b.calls { + if c.err != nil { + c.err <- err + } + } + break retry + } +} + +// trySolo is a special sentinel error value used for signaling that a +// transaction function should be re-run. It should never be seen by +// callers. +var trySolo = errors.New("batch function returned an error and should be re-run solo") + +type panicked struct { + reason interface{} +} + +func (p panicked) Error() string { + if err, ok := p.reason.(error); ok { + return err.Error() + } + return fmt.Sprintf("panic: %v", p.reason) +} + +func safelyCall(fn func(*Tx) error, tx *Tx) (err error) { + defer func() { + if p := recover(); p != nil { + err = panicked{p} + } + }() + return fn(tx) +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/batch_benchmark_test.go b/Godeps/_workspace/src/github.com/boltdb/bolt/batch_benchmark_test.go new file mode 100644 index 0000000000000..b745a371f5d3a --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/batch_benchmark_test.go @@ -0,0 +1,170 @@ +package bolt_test + +import ( + "bytes" + "encoding/binary" + "errors" + "hash/fnv" + "sync" + "testing" + + "github.com/boltdb/bolt" +) + +func validateBatchBench(b *testing.B, db *TestDB) { + var rollback = errors.New("sentinel error to cause rollback") + validate := func(tx *bolt.Tx) error { + bucket := tx.Bucket([]byte("bench")) + h := fnv.New32a() + buf := make([]byte, 4) + for id := uint32(0); id < 1000; id++ { + binary.LittleEndian.PutUint32(buf, id) + h.Reset() + h.Write(buf[:]) + k := h.Sum(nil) + v := bucket.Get(k) + if v == nil { + b.Errorf("not found id=%d key=%x", id, k) + continue + } + if g, e := v, []byte("filler"); !bytes.Equal(g, e) { + b.Errorf("bad value for id=%d key=%x: %s != %q", id, k, g, e) + } + if err := bucket.Delete(k); err != nil { + return err + } + } + // should be empty now + c := bucket.Cursor() + for k, v := c.First(); k != nil; k, v = c.Next() { + b.Errorf("unexpected key: %x = %q", k, v) + } + return rollback + } + if err := db.Update(validate); err != nil && err != rollback { + b.Error(err) + } +} + +func BenchmarkDBBatchAutomatic(b *testing.B) { + db := NewTestDB() + defer db.Close() + db.MustCreateBucket([]byte("bench")) + + b.ResetTimer() + for i := 0; i < b.N; i++ { + start := make(chan struct{}) + var wg sync.WaitGroup + + for round := 0; round < 1000; round++ { + wg.Add(1) + + go func(id uint32) { + defer wg.Done() + <-start + + h := fnv.New32a() + buf := make([]byte, 4) + binary.LittleEndian.PutUint32(buf, id) + h.Write(buf[:]) + k := h.Sum(nil) + insert := func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("bench")) + return b.Put(k, []byte("filler")) + } + if err := db.Batch(insert); err != nil { + b.Error(err) + return + } + }(uint32(round)) + } + close(start) + wg.Wait() + } + + b.StopTimer() + validateBatchBench(b, db) +} + +func BenchmarkDBBatchSingle(b *testing.B) { + db := NewTestDB() + defer db.Close() + db.MustCreateBucket([]byte("bench")) + + b.ResetTimer() + for i := 0; i < b.N; i++ { + start := make(chan struct{}) + var wg sync.WaitGroup + + for round := 0; round < 1000; round++ { + wg.Add(1) + go func(id uint32) { + defer wg.Done() + <-start + + h := fnv.New32a() + buf := make([]byte, 4) + binary.LittleEndian.PutUint32(buf, id) + h.Write(buf[:]) + k := h.Sum(nil) + insert := func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("bench")) + return b.Put(k, []byte("filler")) + } + if err := db.Update(insert); err != nil { + b.Error(err) + return + } + }(uint32(round)) + } + close(start) + wg.Wait() + } + + b.StopTimer() + validateBatchBench(b, db) +} + +func BenchmarkDBBatchManual10x100(b *testing.B) { + db := NewTestDB() + defer db.Close() + db.MustCreateBucket([]byte("bench")) + + b.ResetTimer() + for i := 0; i < b.N; i++ { + start := make(chan struct{}) + var wg sync.WaitGroup + + for major := 0; major < 10; major++ { + wg.Add(1) + go func(id uint32) { + defer wg.Done() + <-start + + insert100 := func(tx *bolt.Tx) error { + h := fnv.New32a() + buf := make([]byte, 4) + for minor := uint32(0); minor < 100; minor++ { + binary.LittleEndian.PutUint32(buf, uint32(id*100+minor)) + h.Reset() + h.Write(buf[:]) + k := h.Sum(nil) + b := tx.Bucket([]byte("bench")) + if err := b.Put(k, []byte("filler")); err != nil { + return err + } + } + return nil + } + if err := db.Update(insert100); err != nil { + b.Fatal(err) + } + }(uint32(major)) + } + close(start) + wg.Wait() + } + + b.StopTimer() + validateBatchBench(b, db) +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/batch_example_test.go b/Godeps/_workspace/src/github.com/boltdb/bolt/batch_example_test.go new file mode 100644 index 0000000000000..74eff8af98678 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/batch_example_test.go @@ -0,0 +1,148 @@ +package bolt_test + +import ( + "encoding/binary" + "fmt" + "io/ioutil" + "log" + "math/rand" + "net/http" + "net/http/httptest" + "os" + + "github.com/boltdb/bolt" +) + +// Set this to see how the counts are actually updated. +const verbose = false + +// Counter updates a counter in Bolt for every URL path requested. +type counter struct { + db *bolt.DB +} + +func (c counter) ServeHTTP(rw http.ResponseWriter, req *http.Request) { + // Communicates the new count from a successful database + // transaction. + var result uint64 + + increment := func(tx *bolt.Tx) error { + b, err := tx.CreateBucketIfNotExists([]byte("hits")) + if err != nil { + return err + } + key := []byte(req.URL.String()) + // Decode handles key not found for us. + count := decode(b.Get(key)) + 1 + b.Put(key, encode(count)) + // All good, communicate new count. + result = count + return nil + } + if err := c.db.Batch(increment); err != nil { + http.Error(rw, err.Error(), 500) + return + } + + if verbose { + log.Printf("server: %s: %d", req.URL.String(), result) + } + + rw.Header().Set("Content-Type", "application/octet-stream") + fmt.Fprintf(rw, "%d\n", result) +} + +func client(id int, base string, paths []string) error { + // Process paths in random order. + rng := rand.New(rand.NewSource(int64(id))) + permutation := rng.Perm(len(paths)) + + for i := range paths { + path := paths[permutation[i]] + resp, err := http.Get(base + path) + if err != nil { + return err + } + defer resp.Body.Close() + buf, err := ioutil.ReadAll(resp.Body) + if err != nil { + return err + } + if verbose { + log.Printf("client: %s: %s", path, buf) + } + } + return nil +} + +func ExampleDB_Batch() { + // Open the database. + db, _ := bolt.Open(tempfile(), 0666, nil) + defer os.Remove(db.Path()) + defer db.Close() + + // Start our web server + count := counter{db} + srv := httptest.NewServer(count) + defer srv.Close() + + // Decrease the batch size to make things more interesting. + db.MaxBatchSize = 3 + + // Get every path multiple times concurrently. + const clients = 10 + paths := []string{ + "/foo", + "/bar", + "/baz", + "/quux", + "/thud", + "/xyzzy", + } + errors := make(chan error, clients) + for i := 0; i < clients; i++ { + go func(id int) { + errors <- client(id, srv.URL, paths) + }(i) + } + // Check all responses to make sure there's no error. + for i := 0; i < clients; i++ { + if err := <-errors; err != nil { + fmt.Printf("client error: %v", err) + return + } + } + + // Check the final result + db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("hits")) + c := b.Cursor() + for k, v := c.First(); k != nil; k, v = c.Next() { + fmt.Printf("hits to %s: %d\n", k, decode(v)) + } + return nil + }) + + // Output: + // hits to /bar: 10 + // hits to /baz: 10 + // hits to /foo: 10 + // hits to /quux: 10 + // hits to /thud: 10 + // hits to /xyzzy: 10 +} + +// encode marshals a counter. +func encode(n uint64) []byte { + buf := make([]byte, 8) + binary.BigEndian.PutUint64(buf, n) + return buf +} + +// decode unmarshals a counter. Nil buffers are decoded as 0. +func decode(buf []byte) uint64 { + if buf == nil { + return 0 + } + return binary.BigEndian.Uint64(buf) +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/batch_test.go b/Godeps/_workspace/src/github.com/boltdb/bolt/batch_test.go new file mode 100644 index 0000000000000..0b5075fddff3b --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/batch_test.go @@ -0,0 +1,167 @@ +package bolt_test + +import ( + "testing" + "time" + + "github.com/boltdb/bolt" +) + +// Ensure two functions can perform updates in a single batch. +func TestDB_Batch(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.MustCreateBucket([]byte("widgets")) + + // Iterate over multiple updates in separate goroutines. + n := 2 + ch := make(chan error) + for i := 0; i < n; i++ { + go func(i int) { + ch <- db.Batch(func(tx *bolt.Tx) error { + return tx.Bucket([]byte("widgets")).Put(u64tob(uint64(i)), []byte{}) + }) + }(i) + } + + // Check all responses to make sure there's no error. + for i := 0; i < n; i++ { + if err := <-ch; err != nil { + t.Fatal(err) + } + } + + // Ensure data is correct. + db.MustView(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("widgets")) + for i := 0; i < n; i++ { + if v := b.Get(u64tob(uint64(i))); v == nil { + t.Errorf("key not found: %d", i) + } + } + return nil + }) +} + +func TestDB_Batch_Panic(t *testing.T) { + db := NewTestDB() + defer db.Close() + + var sentinel int + var bork = &sentinel + var problem interface{} + var err error + + // Execute a function inside a batch that panics. + func() { + defer func() { + if p := recover(); p != nil { + problem = p + } + }() + err = db.Batch(func(tx *bolt.Tx) error { + panic(bork) + }) + }() + + // Verify there is no error. + if g, e := err, error(nil); g != e { + t.Fatalf("wrong error: %v != %v", g, e) + } + // Verify the panic was captured. + if g, e := problem, bork; g != e { + t.Fatalf("wrong error: %v != %v", g, e) + } +} + +func TestDB_BatchFull(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.MustCreateBucket([]byte("widgets")) + + const size = 3 + // buffered so we never leak goroutines + ch := make(chan error, size) + put := func(i int) { + ch <- db.Batch(func(tx *bolt.Tx) error { + return tx.Bucket([]byte("widgets")).Put(u64tob(uint64(i)), []byte{}) + }) + } + + db.MaxBatchSize = size + // high enough to never trigger here + db.MaxBatchDelay = 1 * time.Hour + + go put(1) + go put(2) + + // Give the batch a chance to exhibit bugs. + time.Sleep(10 * time.Millisecond) + + // not triggered yet + select { + case <-ch: + t.Fatalf("batch triggered too early") + default: + } + + go put(3) + + // Check all responses to make sure there's no error. + for i := 0; i < size; i++ { + if err := <-ch; err != nil { + t.Fatal(err) + } + } + + // Ensure data is correct. + db.MustView(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("widgets")) + for i := 1; i <= size; i++ { + if v := b.Get(u64tob(uint64(i))); v == nil { + t.Errorf("key not found: %d", i) + } + } + return nil + }) +} + +func TestDB_BatchTime(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.MustCreateBucket([]byte("widgets")) + + const size = 1 + // buffered so we never leak goroutines + ch := make(chan error, size) + put := func(i int) { + ch <- db.Batch(func(tx *bolt.Tx) error { + return tx.Bucket([]byte("widgets")).Put(u64tob(uint64(i)), []byte{}) + }) + } + + db.MaxBatchSize = 1000 + db.MaxBatchDelay = 0 + + go put(1) + + // Batch must trigger by time alone. + + // Check all responses to make sure there's no error. + for i := 0; i < size; i++ { + if err := <-ch; err != nil { + t.Fatal(err) + } + } + + // Ensure data is correct. + db.MustView(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("widgets")) + for i := 1; i <= size; i++ { + if v := b.Get(u64tob(uint64(i))); v == nil { + t.Errorf("key not found: %d", i) + } + } + return nil + }) +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_386.go b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_386.go new file mode 100644 index 0000000000000..e659bfb91f338 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_386.go @@ -0,0 +1,7 @@ +package bolt + +// maxMapSize represents the largest mmap size supported by Bolt. +const maxMapSize = 0x7FFFFFFF // 2GB + +// maxAllocSize is the size used when creating array pointers. +const maxAllocSize = 0xFFFFFFF diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_amd64.go b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_amd64.go new file mode 100644 index 0000000000000..cca6b7eb70704 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_amd64.go @@ -0,0 +1,7 @@ +package bolt + +// maxMapSize represents the largest mmap size supported by Bolt. +const maxMapSize = 0xFFFFFFFFFFFF // 256TB + +// maxAllocSize is the size used when creating array pointers. +const maxAllocSize = 0x7FFFFFFF diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_arm.go b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_arm.go new file mode 100644 index 0000000000000..e659bfb91f338 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_arm.go @@ -0,0 +1,7 @@ +package bolt + +// maxMapSize represents the largest mmap size supported by Bolt. +const maxMapSize = 0x7FFFFFFF // 2GB + +// maxAllocSize is the size used when creating array pointers. +const maxAllocSize = 0xFFFFFFF diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_linux.go b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_linux.go new file mode 100644 index 0000000000000..e9d1c907b63a5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_linux.go @@ -0,0 +1,12 @@ +package bolt + +import ( + "syscall" +) + +var odirect = syscall.O_DIRECT + +// fdatasync flushes written data to a file descriptor. +func fdatasync(db *DB) error { + return syscall.Fdatasync(int(db.file.Fd())) +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_openbsd.go b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_openbsd.go new file mode 100644 index 0000000000000..7c1bef1a4f404 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_openbsd.go @@ -0,0 +1,29 @@ +package bolt + +import ( + "syscall" + "unsafe" +) + +const ( + msAsync = 1 << iota // perform asynchronous writes + msSync // perform synchronous writes + msInvalidate // invalidate cached data +) + +var odirect int + +func msync(db *DB) error { + _, _, errno := syscall.Syscall(syscall.SYS_MSYNC, uintptr(unsafe.Pointer(db.data)), uintptr(db.datasz), msInvalidate) + if errno != 0 { + return errno + } + return nil +} + +func fdatasync(db *DB) error { + if db.data != nil { + return msync(db) + } + return db.file.Sync() +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_test.go b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_test.go new file mode 100644 index 0000000000000..b7bea1fc5919d --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_test.go @@ -0,0 +1,36 @@ +package bolt_test + +import ( + "fmt" + "path/filepath" + "reflect" + "runtime" + "testing" +) + +// assert fails the test if the condition is false. +func assert(tb testing.TB, condition bool, msg string, v ...interface{}) { + if !condition { + _, file, line, _ := runtime.Caller(1) + fmt.Printf("\033[31m%s:%d: "+msg+"\033[39m\n\n", append([]interface{}{filepath.Base(file), line}, v...)...) + tb.FailNow() + } +} + +// ok fails the test if an err is not nil. +func ok(tb testing.TB, err error) { + if err != nil { + _, file, line, _ := runtime.Caller(1) + fmt.Printf("\033[31m%s:%d: unexpected error: %s\033[39m\n\n", filepath.Base(file), line, err.Error()) + tb.FailNow() + } +} + +// equals fails the test if exp is not equal to act. +func equals(tb testing.TB, exp, act interface{}) { + if !reflect.DeepEqual(exp, act) { + _, file, line, _ := runtime.Caller(1) + fmt.Printf("\033[31m%s:%d:\n\n\texp: %#v\n\n\tgot: %#v\033[39m\n\n", filepath.Base(file), line, exp, act) + tb.FailNow() + } +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_unix.go b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_unix.go new file mode 100644 index 0000000000000..17ca318bf7295 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_unix.go @@ -0,0 +1,100 @@ +// +build !windows,!plan9 + +package bolt + +import ( + "fmt" + "os" + "syscall" + "time" + "unsafe" +) + +// flock acquires an advisory lock on a file descriptor. +func flock(f *os.File, exclusive bool, timeout time.Duration) error { + var t time.Time + for { + // If we're beyond our timeout then return an error. + // This can only occur after we've attempted a flock once. + if t.IsZero() { + t = time.Now() + } else if timeout > 0 && time.Since(t) > timeout { + return ErrTimeout + } + flag := syscall.LOCK_SH + if exclusive { + flag = syscall.LOCK_EX + } + + // Otherwise attempt to obtain an exclusive lock. + err := syscall.Flock(int(f.Fd()), flag|syscall.LOCK_NB) + if err == nil { + return nil + } else if err != syscall.EWOULDBLOCK { + return err + } + + // Wait for a bit and try again. + time.Sleep(50 * time.Millisecond) + } +} + +// funlock releases an advisory lock on a file descriptor. +func funlock(f *os.File) error { + return syscall.Flock(int(f.Fd()), syscall.LOCK_UN) +} + +// mmap memory maps a DB's data file. +func mmap(db *DB, sz int) error { + // Truncate and fsync to ensure file size metadata is flushed. + // https://github.com/boltdb/bolt/issues/284 + if !db.NoGrowSync && !db.readOnly { + if err := db.file.Truncate(int64(sz)); err != nil { + return fmt.Errorf("file resize error: %s", err) + } + if err := db.file.Sync(); err != nil { + return fmt.Errorf("file sync error: %s", err) + } + } + + // Map the data file to memory. + b, err := syscall.Mmap(int(db.file.Fd()), 0, sz, syscall.PROT_READ, syscall.MAP_SHARED) + if err != nil { + return err + } + + // Advise the kernel that the mmap is accessed randomly. + if err := madvise(b, syscall.MADV_RANDOM); err != nil { + return fmt.Errorf("madvise: %s", err) + } + + // Save the original byte slice and convert to a byte array pointer. + db.dataref = b + db.data = (*[maxMapSize]byte)(unsafe.Pointer(&b[0])) + db.datasz = sz + return nil +} + +// munmap unmaps a DB's data file from memory. +func munmap(db *DB) error { + // Ignore the unmap if we have no mapped data. + if db.dataref == nil { + return nil + } + + // Unmap using the original byte slice. + err := syscall.Munmap(db.dataref) + db.dataref = nil + db.data = nil + db.datasz = 0 + return err +} + +// NOTE: This function is copied from stdlib because it is not available on darwin. +func madvise(b []byte, advice int) (err error) { + _, _, e1 := syscall.Syscall(syscall.SYS_MADVISE, uintptr(unsafe.Pointer(&b[0])), uintptr(len(b)), uintptr(advice)) + if e1 != 0 { + err = e1 + } + return +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_windows.go b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_windows.go new file mode 100644 index 0000000000000..8b782be5f9eff --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/bolt_windows.go @@ -0,0 +1,76 @@ +package bolt + +import ( + "fmt" + "os" + "syscall" + "time" + "unsafe" +) + +var odirect int + +// fdatasync flushes written data to a file descriptor. +func fdatasync(db *DB) error { + return db.file.Sync() +} + +// flock acquires an advisory lock on a file descriptor. +func flock(f *os.File, _ bool, _ time.Duration) error { + return nil +} + +// funlock releases an advisory lock on a file descriptor. +func funlock(f *os.File) error { + return nil +} + +// mmap memory maps a DB's data file. +// Based on: https://github.com/edsrzf/mmap-go +func mmap(db *DB, sz int) error { + if !db.readOnly { + // Truncate the database to the size of the mmap. + if err := db.file.Truncate(int64(sz)); err != nil { + return fmt.Errorf("truncate: %s", err) + } + } + + // Open a file mapping handle. + sizelo := uint32(sz >> 32) + sizehi := uint32(sz) & 0xffffffff + h, errno := syscall.CreateFileMapping(syscall.Handle(db.file.Fd()), nil, syscall.PAGE_READONLY, sizelo, sizehi, nil) + if h == 0 { + return os.NewSyscallError("CreateFileMapping", errno) + } + + // Create the memory map. + addr, errno := syscall.MapViewOfFile(h, syscall.FILE_MAP_READ, 0, 0, uintptr(sz)) + if addr == 0 { + return os.NewSyscallError("MapViewOfFile", errno) + } + + // Close mapping handle. + if err := syscall.CloseHandle(syscall.Handle(h)); err != nil { + return os.NewSyscallError("CloseHandle", err) + } + + // Convert to a byte array. + db.data = ((*[maxMapSize]byte)(unsafe.Pointer(addr))) + db.datasz = sz + + return nil +} + +// munmap unmaps a pointer from a file. +// Based on: https://github.com/edsrzf/mmap-go +func munmap(db *DB) error { + if db.data == nil { + return nil + } + + addr := (uintptr)(unsafe.Pointer(&db.data[0])) + if err := syscall.UnmapViewOfFile(addr); err != nil { + return os.NewSyscallError("UnmapViewOfFile", err) + } + return nil +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/boltsync_unix.go b/Godeps/_workspace/src/github.com/boltdb/bolt/boltsync_unix.go new file mode 100644 index 0000000000000..8db89776fe660 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/boltsync_unix.go @@ -0,0 +1,10 @@ +// +build !windows,!plan9,!linux,!openbsd + +package bolt + +var odirect int + +// fdatasync flushes written data to a file descriptor. +func fdatasync(db *DB) error { + return db.file.Sync() +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/bucket.go b/Godeps/_workspace/src/github.com/boltdb/bolt/bucket.go new file mode 100644 index 0000000000000..6766992100f18 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/bucket.go @@ -0,0 +1,743 @@ +package bolt + +import ( + "bytes" + "fmt" + "unsafe" +) + +const ( + // MaxKeySize is the maximum length of a key, in bytes. + MaxKeySize = 32768 + + // MaxValueSize is the maximum length of a value, in bytes. + MaxValueSize = 4294967295 +) + +const ( + maxUint = ^uint(0) + minUint = 0 + maxInt = int(^uint(0) >> 1) + minInt = -maxInt - 1 +) + +const bucketHeaderSize = int(unsafe.Sizeof(bucket{})) + +const ( + minFillPercent = 0.1 + maxFillPercent = 1.0 +) + +// DefaultFillPercent is the percentage that split pages are filled. +// This value can be changed by setting Bucket.FillPercent. +const DefaultFillPercent = 0.5 + +// Bucket represents a collection of key/value pairs inside the database. +type Bucket struct { + *bucket + tx *Tx // the associated transaction + buckets map[string]*Bucket // subbucket cache + page *page // inline page reference + rootNode *node // materialized node for the root page. + nodes map[pgid]*node // node cache + + // Sets the threshold for filling nodes when they split. By default, + // the bucket will fill to 50% but it can be useful to increase this + // amount if you know that your write workloads are mostly append-only. + // + // This is non-persisted across transactions so it must be set in every Tx. + FillPercent float64 +} + +// bucket represents the on-file representation of a bucket. +// This is stored as the "value" of a bucket key. If the bucket is small enough, +// then its root page can be stored inline in the "value", after the bucket +// header. In the case of inline buckets, the "root" will be 0. +type bucket struct { + root pgid // page id of the bucket's root-level page + sequence uint64 // monotonically incrementing, used by NextSequence() +} + +// newBucket returns a new bucket associated with a transaction. +func newBucket(tx *Tx) Bucket { + var b = Bucket{tx: tx, FillPercent: DefaultFillPercent} + if tx.writable { + b.buckets = make(map[string]*Bucket) + b.nodes = make(map[pgid]*node) + } + return b +} + +// Tx returns the tx of the bucket. +func (b *Bucket) Tx() *Tx { + return b.tx +} + +// Root returns the root of the bucket. +func (b *Bucket) Root() pgid { + return b.root +} + +// Writable returns whether the bucket is writable. +func (b *Bucket) Writable() bool { + return b.tx.writable +} + +// Cursor creates a cursor associated with the bucket. +// The cursor is only valid as long as the transaction is open. +// Do not use a cursor after the transaction is closed. +func (b *Bucket) Cursor() *Cursor { + // Update transaction statistics. + b.tx.stats.CursorCount++ + + // Allocate and return a cursor. + return &Cursor{ + bucket: b, + stack: make([]elemRef, 0), + } +} + +// Bucket retrieves a nested bucket by name. +// Returns nil if the bucket does not exist. +func (b *Bucket) Bucket(name []byte) *Bucket { + if b.buckets != nil { + if child := b.buckets[string(name)]; child != nil { + return child + } + } + + // Move cursor to key. + c := b.Cursor() + k, v, flags := c.seek(name) + + // Return nil if the key doesn't exist or it is not a bucket. + if !bytes.Equal(name, k) || (flags&bucketLeafFlag) == 0 { + return nil + } + + // Otherwise create a bucket and cache it. + var child = b.openBucket(v) + if b.buckets != nil { + b.buckets[string(name)] = child + } + + return child +} + +// Helper method that re-interprets a sub-bucket value +// from a parent into a Bucket +func (b *Bucket) openBucket(value []byte) *Bucket { + var child = newBucket(b.tx) + + // If this is a writable transaction then we need to copy the bucket entry. + // Read-only transactions can point directly at the mmap entry. + if b.tx.writable { + child.bucket = &bucket{} + *child.bucket = *(*bucket)(unsafe.Pointer(&value[0])) + } else { + child.bucket = (*bucket)(unsafe.Pointer(&value[0])) + } + + // Save a reference to the inline page if the bucket is inline. + if child.root == 0 { + child.page = (*page)(unsafe.Pointer(&value[bucketHeaderSize])) + } + + return &child +} + +// CreateBucket creates a new bucket at the given key and returns the new bucket. +// Returns an error if the key already exists, if the bucket name is blank, or if the bucket name is too long. +func (b *Bucket) CreateBucket(key []byte) (*Bucket, error) { + if b.tx.db == nil { + return nil, ErrTxClosed + } else if !b.tx.writable { + return nil, ErrTxNotWritable + } else if len(key) == 0 { + return nil, ErrBucketNameRequired + } + + // Move cursor to correct position. + c := b.Cursor() + k, _, flags := c.seek(key) + + // Return an error if there is an existing key. + if bytes.Equal(key, k) { + if (flags & bucketLeafFlag) != 0 { + return nil, ErrBucketExists + } else { + return nil, ErrIncompatibleValue + } + } + + // Create empty, inline bucket. + var bucket = Bucket{ + bucket: &bucket{}, + rootNode: &node{isLeaf: true}, + FillPercent: DefaultFillPercent, + } + var value = bucket.write() + + // Insert into node. + key = cloneBytes(key) + c.node().put(key, key, value, 0, bucketLeafFlag) + + // Since subbuckets are not allowed on inline buckets, we need to + // dereference the inline page, if it exists. This will cause the bucket + // to be treated as a regular, non-inline bucket for the rest of the tx. + b.page = nil + + return b.Bucket(key), nil +} + +// CreateBucketIfNotExists creates a new bucket if it doesn't already exist and returns a reference to it. +// Returns an error if the bucket name is blank, or if the bucket name is too long. +func (b *Bucket) CreateBucketIfNotExists(key []byte) (*Bucket, error) { + child, err := b.CreateBucket(key) + if err == ErrBucketExists { + return b.Bucket(key), nil + } else if err != nil { + return nil, err + } + return child, nil +} + +// DeleteBucket deletes a bucket at the given key. +// Returns an error if the bucket does not exists, or if the key represents a non-bucket value. +func (b *Bucket) DeleteBucket(key []byte) error { + if b.tx.db == nil { + return ErrTxClosed + } else if !b.Writable() { + return ErrTxNotWritable + } + + // Move cursor to correct position. + c := b.Cursor() + k, _, flags := c.seek(key) + + // Return an error if bucket doesn't exist or is not a bucket. + if !bytes.Equal(key, k) { + return ErrBucketNotFound + } else if (flags & bucketLeafFlag) == 0 { + return ErrIncompatibleValue + } + + // Recursively delete all child buckets. + child := b.Bucket(key) + err := child.ForEach(func(k, v []byte) error { + if v == nil { + if err := child.DeleteBucket(k); err != nil { + return fmt.Errorf("delete bucket: %s", err) + } + } + return nil + }) + if err != nil { + return err + } + + // Remove cached copy. + delete(b.buckets, string(key)) + + // Release all bucket pages to freelist. + child.nodes = nil + child.rootNode = nil + child.free() + + // Delete the node if we have a matching key. + c.node().del(key) + + return nil +} + +// Get retrieves the value for a key in the bucket. +// Returns a nil value if the key does not exist or if the key is a nested bucket. +// The returned value is only valid for the life of the transaction. +func (b *Bucket) Get(key []byte) []byte { + k, v, flags := b.Cursor().seek(key) + + // Return nil if this is a bucket. + if (flags & bucketLeafFlag) != 0 { + return nil + } + + // If our target node isn't the same key as what's passed in then return nil. + if !bytes.Equal(key, k) { + return nil + } + return v +} + +// Put sets the value for a key in the bucket. +// If the key exist then its previous value will be overwritten. +// Returns an error if the bucket was created from a read-only transaction, if the key is blank, if the key is too large, or if the value is too large. +func (b *Bucket) Put(key []byte, value []byte) error { + if b.tx.db == nil { + return ErrTxClosed + } else if !b.Writable() { + return ErrTxNotWritable + } else if len(key) == 0 { + return ErrKeyRequired + } else if len(key) > MaxKeySize { + return ErrKeyTooLarge + } else if int64(len(value)) > MaxValueSize { + return ErrValueTooLarge + } + + // Move cursor to correct position. + c := b.Cursor() + k, _, flags := c.seek(key) + + // Return an error if there is an existing key with a bucket value. + if bytes.Equal(key, k) && (flags&bucketLeafFlag) != 0 { + return ErrIncompatibleValue + } + + // Insert into node. + key = cloneBytes(key) + c.node().put(key, key, value, 0, 0) + + return nil +} + +// Delete removes a key from the bucket. +// If the key does not exist then nothing is done and a nil error is returned. +// Returns an error if the bucket was created from a read-only transaction. +func (b *Bucket) Delete(key []byte) error { + if b.tx.db == nil { + return ErrTxClosed + } else if !b.Writable() { + return ErrTxNotWritable + } + + // Move cursor to correct position. + c := b.Cursor() + _, _, flags := c.seek(key) + + // Return an error if there is already existing bucket value. + if (flags & bucketLeafFlag) != 0 { + return ErrIncompatibleValue + } + + // Delete the node if we have a matching key. + c.node().del(key) + + return nil +} + +// NextSequence returns an autoincrementing integer for the bucket. +func (b *Bucket) NextSequence() (uint64, error) { + if b.tx.db == nil { + return 0, ErrTxClosed + } else if !b.Writable() { + return 0, ErrTxNotWritable + } + + // Materialize the root node if it hasn't been already so that the + // bucket will be saved during commit. + if b.rootNode == nil { + _ = b.node(b.root, nil) + } + + // Increment and return the sequence. + b.bucket.sequence++ + return b.bucket.sequence, nil +} + +// ForEach executes a function for each key/value pair in a bucket. +// If the provided function returns an error then the iteration is stopped and +// the error is returned to the caller. +func (b *Bucket) ForEach(fn func(k, v []byte) error) error { + if b.tx.db == nil { + return ErrTxClosed + } + c := b.Cursor() + for k, v := c.First(); k != nil; k, v = c.Next() { + if err := fn(k, v); err != nil { + return err + } + } + return nil +} + +// Stat returns stats on a bucket. +func (b *Bucket) Stats() BucketStats { + var s, subStats BucketStats + pageSize := b.tx.db.pageSize + s.BucketN += 1 + if b.root == 0 { + s.InlineBucketN += 1 + } + b.forEachPage(func(p *page, depth int) { + if (p.flags & leafPageFlag) != 0 { + s.KeyN += int(p.count) + + // used totals the used bytes for the page + used := pageHeaderSize + + if p.count != 0 { + // If page has any elements, add all element headers. + used += leafPageElementSize * int(p.count-1) + + // Add all element key, value sizes. + // The computation takes advantage of the fact that the position + // of the last element's key/value equals to the total of the sizes + // of all previous elements' keys and values. + // It also includes the last element's header. + lastElement := p.leafPageElement(p.count - 1) + used += int(lastElement.pos + lastElement.ksize + lastElement.vsize) + } + + if b.root == 0 { + // For inlined bucket just update the inline stats + s.InlineBucketInuse += used + } else { + // For non-inlined bucket update all the leaf stats + s.LeafPageN++ + s.LeafInuse += used + s.LeafOverflowN += int(p.overflow) + + // Collect stats from sub-buckets. + // Do that by iterating over all element headers + // looking for the ones with the bucketLeafFlag. + for i := uint16(0); i < p.count; i++ { + e := p.leafPageElement(i) + if (e.flags & bucketLeafFlag) != 0 { + // For any bucket element, open the element value + // and recursively call Stats on the contained bucket. + subStats.Add(b.openBucket(e.value()).Stats()) + } + } + } + } else if (p.flags & branchPageFlag) != 0 { + s.BranchPageN++ + lastElement := p.branchPageElement(p.count - 1) + + // used totals the used bytes for the page + // Add header and all element headers. + used := pageHeaderSize + (branchPageElementSize * int(p.count-1)) + + // Add size of all keys and values. + // Again, use the fact that last element's position equals to + // the total of key, value sizes of all previous elements. + used += int(lastElement.pos + lastElement.ksize) + s.BranchInuse += used + s.BranchOverflowN += int(p.overflow) + } + + // Keep track of maximum page depth. + if depth+1 > s.Depth { + s.Depth = (depth + 1) + } + }) + + // Alloc stats can be computed from page counts and pageSize. + s.BranchAlloc = (s.BranchPageN + s.BranchOverflowN) * pageSize + s.LeafAlloc = (s.LeafPageN + s.LeafOverflowN) * pageSize + + // Add the max depth of sub-buckets to get total nested depth. + s.Depth += subStats.Depth + // Add the stats for all sub-buckets + s.Add(subStats) + return s +} + +// forEachPage iterates over every page in a bucket, including inline pages. +func (b *Bucket) forEachPage(fn func(*page, int)) { + // If we have an inline page then just use that. + if b.page != nil { + fn(b.page, 0) + return + } + + // Otherwise traverse the page hierarchy. + b.tx.forEachPage(b.root, 0, fn) +} + +// forEachPageNode iterates over every page (or node) in a bucket. +// This also includes inline pages. +func (b *Bucket) forEachPageNode(fn func(*page, *node, int)) { + // If we have an inline page or root node then just use that. + if b.page != nil { + fn(b.page, nil, 0) + return + } + b._forEachPageNode(b.root, 0, fn) +} + +func (b *Bucket) _forEachPageNode(pgid pgid, depth int, fn func(*page, *node, int)) { + var p, n = b.pageNode(pgid) + + // Execute function. + fn(p, n, depth) + + // Recursively loop over children. + if p != nil { + if (p.flags & branchPageFlag) != 0 { + for i := 0; i < int(p.count); i++ { + elem := p.branchPageElement(uint16(i)) + b._forEachPageNode(elem.pgid, depth+1, fn) + } + } + } else { + if !n.isLeaf { + for _, inode := range n.inodes { + b._forEachPageNode(inode.pgid, depth+1, fn) + } + } + } +} + +// spill writes all the nodes for this bucket to dirty pages. +func (b *Bucket) spill() error { + // Spill all child buckets first. + for name, child := range b.buckets { + // If the child bucket is small enough and it has no child buckets then + // write it inline into the parent bucket's page. Otherwise spill it + // like a normal bucket and make the parent value a pointer to the page. + var value []byte + if child.inlineable() { + child.free() + value = child.write() + } else { + if err := child.spill(); err != nil { + return err + } + + // Update the child bucket header in this bucket. + value = make([]byte, unsafe.Sizeof(bucket{})) + var bucket = (*bucket)(unsafe.Pointer(&value[0])) + *bucket = *child.bucket + } + + // Skip writing the bucket if there are no materialized nodes. + if child.rootNode == nil { + continue + } + + // Update parent node. + var c = b.Cursor() + k, _, flags := c.seek([]byte(name)) + if !bytes.Equal([]byte(name), k) { + panic(fmt.Sprintf("misplaced bucket header: %x -> %x", []byte(name), k)) + } + if flags&bucketLeafFlag == 0 { + panic(fmt.Sprintf("unexpected bucket header flag: %x", flags)) + } + c.node().put([]byte(name), []byte(name), value, 0, bucketLeafFlag) + } + + // Ignore if there's not a materialized root node. + if b.rootNode == nil { + return nil + } + + // Spill nodes. + if err := b.rootNode.spill(); err != nil { + return err + } + b.rootNode = b.rootNode.root() + + // Update the root node for this bucket. + if b.rootNode.pgid >= b.tx.meta.pgid { + panic(fmt.Sprintf("pgid (%d) above high water mark (%d)", b.rootNode.pgid, b.tx.meta.pgid)) + } + b.root = b.rootNode.pgid + + return nil +} + +// inlineable returns true if a bucket is small enough to be written inline +// and if it contains no subbuckets. Otherwise returns false. +func (b *Bucket) inlineable() bool { + var n = b.rootNode + + // Bucket must only contain a single leaf node. + if n == nil || !n.isLeaf { + return false + } + + // Bucket is not inlineable if it contains subbuckets or if it goes beyond + // our threshold for inline bucket size. + var size = pageHeaderSize + for _, inode := range n.inodes { + size += leafPageElementSize + len(inode.key) + len(inode.value) + + if inode.flags&bucketLeafFlag != 0 { + return false + } else if size > b.maxInlineBucketSize() { + return false + } + } + + return true +} + +// Returns the maximum total size of a bucket to make it a candidate for inlining. +func (b *Bucket) maxInlineBucketSize() int { + return b.tx.db.pageSize / 4 +} + +// write allocates and writes a bucket to a byte slice. +func (b *Bucket) write() []byte { + // Allocate the appropriate size. + var n = b.rootNode + var value = make([]byte, bucketHeaderSize+n.size()) + + // Write a bucket header. + var bucket = (*bucket)(unsafe.Pointer(&value[0])) + *bucket = *b.bucket + + // Convert byte slice to a fake page and write the root node. + var p = (*page)(unsafe.Pointer(&value[bucketHeaderSize])) + n.write(p) + + return value +} + +// rebalance attempts to balance all nodes. +func (b *Bucket) rebalance() { + for _, n := range b.nodes { + n.rebalance() + } + for _, child := range b.buckets { + child.rebalance() + } +} + +// node creates a node from a page and associates it with a given parent. +func (b *Bucket) node(pgid pgid, parent *node) *node { + _assert(b.nodes != nil, "nodes map expected") + + // Retrieve node if it's already been created. + if n := b.nodes[pgid]; n != nil { + return n + } + + // Otherwise create a node and cache it. + n := &node{bucket: b, parent: parent} + if parent == nil { + b.rootNode = n + } else { + parent.children = append(parent.children, n) + } + + // Use the inline page if this is an inline bucket. + var p = b.page + if p == nil { + p = b.tx.page(pgid) + } + + // Read the page into the node and cache it. + n.read(p) + b.nodes[pgid] = n + + // Update statistics. + b.tx.stats.NodeCount++ + + return n +} + +// free recursively frees all pages in the bucket. +func (b *Bucket) free() { + if b.root == 0 { + return + } + + var tx = b.tx + b.forEachPageNode(func(p *page, n *node, _ int) { + if p != nil { + tx.db.freelist.free(tx.meta.txid, p) + } else { + n.free() + } + }) + b.root = 0 +} + +// dereference removes all references to the old mmap. +func (b *Bucket) dereference() { + if b.rootNode != nil { + b.rootNode.root().dereference() + } + + for _, child := range b.buckets { + child.dereference() + } +} + +// pageNode returns the in-memory node, if it exists. +// Otherwise returns the underlying page. +func (b *Bucket) pageNode(id pgid) (*page, *node) { + // Inline buckets have a fake page embedded in their value so treat them + // differently. We'll return the rootNode (if available) or the fake page. + if b.root == 0 { + if id != 0 { + panic(fmt.Sprintf("inline bucket non-zero page access(2): %d != 0", id)) + } + if b.rootNode != nil { + return nil, b.rootNode + } + return b.page, nil + } + + // Check the node cache for non-inline buckets. + if b.nodes != nil { + if n := b.nodes[id]; n != nil { + return nil, n + } + } + + // Finally lookup the page from the transaction if no node is materialized. + return b.tx.page(id), nil +} + +// BucketStats records statistics about resources used by a bucket. +type BucketStats struct { + // Page count statistics. + BranchPageN int // number of logical branch pages + BranchOverflowN int // number of physical branch overflow pages + LeafPageN int // number of logical leaf pages + LeafOverflowN int // number of physical leaf overflow pages + + // Tree statistics. + KeyN int // number of keys/value pairs + Depth int // number of levels in B+tree + + // Page size utilization. + BranchAlloc int // bytes allocated for physical branch pages + BranchInuse int // bytes actually used for branch data + LeafAlloc int // bytes allocated for physical leaf pages + LeafInuse int // bytes actually used for leaf data + + // Bucket statistics + BucketN int // total number of buckets including the top bucket + InlineBucketN int // total number on inlined buckets + InlineBucketInuse int // bytes used for inlined buckets (also accounted for in LeafInuse) +} + +func (s *BucketStats) Add(other BucketStats) { + s.BranchPageN += other.BranchPageN + s.BranchOverflowN += other.BranchOverflowN + s.LeafPageN += other.LeafPageN + s.LeafOverflowN += other.LeafOverflowN + s.KeyN += other.KeyN + if s.Depth < other.Depth { + s.Depth = other.Depth + } + s.BranchAlloc += other.BranchAlloc + s.BranchInuse += other.BranchInuse + s.LeafAlloc += other.LeafAlloc + s.LeafInuse += other.LeafInuse + + s.BucketN += other.BucketN + s.InlineBucketN += other.InlineBucketN + s.InlineBucketInuse += other.InlineBucketInuse +} + +// cloneBytes returns a copy of a given slice. +func cloneBytes(v []byte) []byte { + var clone = make([]byte, len(v)) + copy(clone, v) + return clone +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/bucket_test.go b/Godeps/_workspace/src/github.com/boltdb/bolt/bucket_test.go new file mode 100644 index 0000000000000..62b8c587831dc --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/bucket_test.go @@ -0,0 +1,1169 @@ +package bolt_test + +import ( + "bytes" + "encoding/binary" + "errors" + "fmt" + "math/rand" + "os" + "strconv" + "strings" + "testing" + "testing/quick" + + "github.com/boltdb/bolt" +) + +// Ensure that a bucket that gets a non-existent key returns nil. +func TestBucket_Get_NonExistent(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + value := tx.Bucket([]byte("widgets")).Get([]byte("foo")) + assert(t, value == nil, "") + return nil + }) +} + +// Ensure that a bucket can read a value that is not flushed yet. +func TestBucket_Get_FromNode(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + b := tx.Bucket([]byte("widgets")) + b.Put([]byte("foo"), []byte("bar")) + value := b.Get([]byte("foo")) + equals(t, []byte("bar"), value) + return nil + }) +} + +// Ensure that a bucket retrieved via Get() returns a nil. +func TestBucket_Get_IncompatibleValue(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + _, err := tx.Bucket([]byte("widgets")).CreateBucket([]byte("foo")) + ok(t, err) + assert(t, tx.Bucket([]byte("widgets")).Get([]byte("foo")) == nil, "") + return nil + }) +} + +// Ensure that a bucket can write a key/value. +func TestBucket_Put(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + err := tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar")) + ok(t, err) + value := tx.Bucket([]byte("widgets")).Get([]byte("foo")) + equals(t, value, []byte("bar")) + return nil + }) +} + +// Ensure that a bucket can rewrite a key in the same transaction. +func TestBucket_Put_Repeat(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + b := tx.Bucket([]byte("widgets")) + ok(t, b.Put([]byte("foo"), []byte("bar"))) + ok(t, b.Put([]byte("foo"), []byte("baz"))) + value := tx.Bucket([]byte("widgets")).Get([]byte("foo")) + equals(t, value, []byte("baz")) + return nil + }) +} + +// Ensure that a bucket can write a bunch of large values. +func TestBucket_Put_Large(t *testing.T) { + db := NewTestDB() + defer db.Close() + + count, factor := 100, 200 + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + b := tx.Bucket([]byte("widgets")) + for i := 1; i < count; i++ { + ok(t, b.Put([]byte(strings.Repeat("0", i*factor)), []byte(strings.Repeat("X", (count-i)*factor)))) + } + return nil + }) + db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("widgets")) + for i := 1; i < count; i++ { + value := b.Get([]byte(strings.Repeat("0", i*factor))) + equals(t, []byte(strings.Repeat("X", (count-i)*factor)), value) + } + return nil + }) +} + +// Ensure that a database can perform multiple large appends safely. +func TestDB_Put_VeryLarge(t *testing.T) { + if testing.Short() { + t.Skip("skipping test in short mode.") + } + + n, batchN := 400000, 200000 + ksize, vsize := 8, 500 + + db := NewTestDB() + defer db.Close() + + for i := 0; i < n; i += batchN { + err := db.Update(func(tx *bolt.Tx) error { + b, _ := tx.CreateBucketIfNotExists([]byte("widgets")) + for j := 0; j < batchN; j++ { + k, v := make([]byte, ksize), make([]byte, vsize) + binary.BigEndian.PutUint32(k, uint32(i+j)) + ok(t, b.Put(k, v)) + } + return nil + }) + ok(t, err) + } +} + +// Ensure that a setting a value on a key with a bucket value returns an error. +func TestBucket_Put_IncompatibleValue(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + _, err := tx.Bucket([]byte("widgets")).CreateBucket([]byte("foo")) + ok(t, err) + equals(t, bolt.ErrIncompatibleValue, tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar"))) + return nil + }) +} + +// Ensure that a setting a value while the transaction is closed returns an error. +func TestBucket_Put_Closed(t *testing.T) { + db := NewTestDB() + defer db.Close() + tx, _ := db.Begin(true) + tx.CreateBucket([]byte("widgets")) + b := tx.Bucket([]byte("widgets")) + tx.Rollback() + equals(t, bolt.ErrTxClosed, b.Put([]byte("foo"), []byte("bar"))) +} + +// Ensure that setting a value on a read-only bucket returns an error. +func TestBucket_Put_ReadOnly(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte("widgets")) + ok(t, err) + return nil + }) + db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("widgets")) + err := b.Put([]byte("foo"), []byte("bar")) + equals(t, err, bolt.ErrTxNotWritable) + return nil + }) +} + +// Ensure that a bucket can delete an existing key. +func TestBucket_Delete(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar")) + err := tx.Bucket([]byte("widgets")).Delete([]byte("foo")) + ok(t, err) + value := tx.Bucket([]byte("widgets")).Get([]byte("foo")) + assert(t, value == nil, "") + return nil + }) +} + +// Ensure that deleting a large set of keys will work correctly. +func TestBucket_Delete_Large(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + var b, _ = tx.CreateBucket([]byte("widgets")) + for i := 0; i < 100; i++ { + ok(t, b.Put([]byte(strconv.Itoa(i)), []byte(strings.Repeat("*", 1024)))) + } + return nil + }) + db.Update(func(tx *bolt.Tx) error { + var b = tx.Bucket([]byte("widgets")) + for i := 0; i < 100; i++ { + ok(t, b.Delete([]byte(strconv.Itoa(i)))) + } + return nil + }) + db.View(func(tx *bolt.Tx) error { + var b = tx.Bucket([]byte("widgets")) + for i := 0; i < 100; i++ { + assert(t, b.Get([]byte(strconv.Itoa(i))) == nil, "") + } + return nil + }) +} + +// Deleting a very large list of keys will cause the freelist to use overflow. +func TestBucket_Delete_FreelistOverflow(t *testing.T) { + if testing.Short() { + t.Skip("skipping test in short mode.") + } + + db := NewTestDB() + defer db.Close() + k := make([]byte, 16) + for i := uint64(0); i < 10000; i++ { + err := db.Update(func(tx *bolt.Tx) error { + b, err := tx.CreateBucketIfNotExists([]byte("0")) + if err != nil { + t.Fatalf("bucket error: %s", err) + } + + for j := uint64(0); j < 1000; j++ { + binary.BigEndian.PutUint64(k[:8], i) + binary.BigEndian.PutUint64(k[8:], j) + if err := b.Put(k, nil); err != nil { + t.Fatalf("put error: %s", err) + } + } + + return nil + }) + + if err != nil { + t.Fatalf("update error: %s", err) + } + } + + // Delete all of them in one large transaction + err := db.Update(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("0")) + c := b.Cursor() + for k, _ := c.First(); k != nil; k, _ = c.Next() { + b.Delete(k) + } + return nil + }) + + // Check that a freelist overflow occurred. + ok(t, err) +} + +// Ensure that accessing and updating nested buckets is ok across transactions. +func TestBucket_Nested(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + // Create a widgets bucket. + b, err := tx.CreateBucket([]byte("widgets")) + ok(t, err) + + // Create a widgets/foo bucket. + _, err = b.CreateBucket([]byte("foo")) + ok(t, err) + + // Create a widgets/bar key. + ok(t, b.Put([]byte("bar"), []byte("0000"))) + + return nil + }) + db.MustCheck() + + // Update widgets/bar. + db.Update(func(tx *bolt.Tx) error { + var b = tx.Bucket([]byte("widgets")) + ok(t, b.Put([]byte("bar"), []byte("xxxx"))) + return nil + }) + db.MustCheck() + + // Cause a split. + db.Update(func(tx *bolt.Tx) error { + var b = tx.Bucket([]byte("widgets")) + for i := 0; i < 10000; i++ { + ok(t, b.Put([]byte(strconv.Itoa(i)), []byte(strconv.Itoa(i)))) + } + return nil + }) + db.MustCheck() + + // Insert into widgets/foo/baz. + db.Update(func(tx *bolt.Tx) error { + var b = tx.Bucket([]byte("widgets")) + ok(t, b.Bucket([]byte("foo")).Put([]byte("baz"), []byte("yyyy"))) + return nil + }) + db.MustCheck() + + // Verify. + db.View(func(tx *bolt.Tx) error { + var b = tx.Bucket([]byte("widgets")) + equals(t, []byte("yyyy"), b.Bucket([]byte("foo")).Get([]byte("baz"))) + equals(t, []byte("xxxx"), b.Get([]byte("bar"))) + for i := 0; i < 10000; i++ { + equals(t, []byte(strconv.Itoa(i)), b.Get([]byte(strconv.Itoa(i)))) + } + return nil + }) +} + +// Ensure that deleting a bucket using Delete() returns an error. +func TestBucket_Delete_Bucket(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + b := tx.Bucket([]byte("widgets")) + _, err := b.CreateBucket([]byte("foo")) + ok(t, err) + equals(t, bolt.ErrIncompatibleValue, b.Delete([]byte("foo"))) + return nil + }) +} + +// Ensure that deleting a key on a read-only bucket returns an error. +func TestBucket_Delete_ReadOnly(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + return nil + }) + db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("widgets")) + err := b.Delete([]byte("foo")) + equals(t, err, bolt.ErrTxNotWritable) + return nil + }) +} + +// Ensure that a deleting value while the transaction is closed returns an error. +func TestBucket_Delete_Closed(t *testing.T) { + db := NewTestDB() + defer db.Close() + tx, _ := db.Begin(true) + tx.CreateBucket([]byte("widgets")) + b := tx.Bucket([]byte("widgets")) + tx.Rollback() + equals(t, bolt.ErrTxClosed, b.Delete([]byte("foo"))) +} + +// Ensure that deleting a bucket causes nested buckets to be deleted. +func TestBucket_DeleteBucket_Nested(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + _, err := tx.Bucket([]byte("widgets")).CreateBucket([]byte("foo")) + ok(t, err) + _, err = tx.Bucket([]byte("widgets")).Bucket([]byte("foo")).CreateBucket([]byte("bar")) + ok(t, err) + ok(t, tx.Bucket([]byte("widgets")).Bucket([]byte("foo")).Bucket([]byte("bar")).Put([]byte("baz"), []byte("bat"))) + ok(t, tx.Bucket([]byte("widgets")).DeleteBucket([]byte("foo"))) + return nil + }) +} + +// Ensure that deleting a bucket causes nested buckets to be deleted after they have been committed. +func TestBucket_DeleteBucket_Nested2(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + _, err := tx.Bucket([]byte("widgets")).CreateBucket([]byte("foo")) + ok(t, err) + _, err = tx.Bucket([]byte("widgets")).Bucket([]byte("foo")).CreateBucket([]byte("bar")) + ok(t, err) + ok(t, tx.Bucket([]byte("widgets")).Bucket([]byte("foo")).Bucket([]byte("bar")).Put([]byte("baz"), []byte("bat"))) + return nil + }) + db.Update(func(tx *bolt.Tx) error { + assert(t, tx.Bucket([]byte("widgets")) != nil, "") + assert(t, tx.Bucket([]byte("widgets")).Bucket([]byte("foo")) != nil, "") + assert(t, tx.Bucket([]byte("widgets")).Bucket([]byte("foo")).Bucket([]byte("bar")) != nil, "") + equals(t, []byte("bat"), tx.Bucket([]byte("widgets")).Bucket([]byte("foo")).Bucket([]byte("bar")).Get([]byte("baz"))) + ok(t, tx.DeleteBucket([]byte("widgets"))) + return nil + }) + db.View(func(tx *bolt.Tx) error { + assert(t, tx.Bucket([]byte("widgets")) == nil, "") + return nil + }) +} + +// Ensure that deleting a child bucket with multiple pages causes all pages to get collected. +func TestBucket_DeleteBucket_Large(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte("widgets")) + ok(t, err) + _, err = tx.Bucket([]byte("widgets")).CreateBucket([]byte("foo")) + ok(t, err) + b := tx.Bucket([]byte("widgets")).Bucket([]byte("foo")) + for i := 0; i < 1000; i++ { + ok(t, b.Put([]byte(fmt.Sprintf("%d", i)), []byte(fmt.Sprintf("%0100d", i)))) + } + return nil + }) + db.Update(func(tx *bolt.Tx) error { + ok(t, tx.DeleteBucket([]byte("widgets"))) + return nil + }) + + // NOTE: Consistency check in TestDB.Close() will panic if pages not freed properly. +} + +// Ensure that a simple value retrieved via Bucket() returns a nil. +func TestBucket_Bucket_IncompatibleValue(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + ok(t, tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar"))) + assert(t, tx.Bucket([]byte("widgets")).Bucket([]byte("foo")) == nil, "") + return nil + }) +} + +// Ensure that creating a bucket on an existing non-bucket key returns an error. +func TestBucket_CreateBucket_IncompatibleValue(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte("widgets")) + ok(t, err) + ok(t, tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar"))) + _, err = tx.Bucket([]byte("widgets")).CreateBucket([]byte("foo")) + equals(t, bolt.ErrIncompatibleValue, err) + return nil + }) +} + +// Ensure that deleting a bucket on an existing non-bucket key returns an error. +func TestBucket_DeleteBucket_IncompatibleValue(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte("widgets")) + ok(t, err) + ok(t, tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar"))) + equals(t, bolt.ErrIncompatibleValue, tx.Bucket([]byte("widgets")).DeleteBucket([]byte("foo"))) + return nil + }) +} + +// Ensure that a bucket can return an autoincrementing sequence. +func TestBucket_NextSequence(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.CreateBucket([]byte("woojits")) + + // Make sure sequence increments. + seq, err := tx.Bucket([]byte("widgets")).NextSequence() + ok(t, err) + equals(t, seq, uint64(1)) + seq, err = tx.Bucket([]byte("widgets")).NextSequence() + ok(t, err) + equals(t, seq, uint64(2)) + + // Buckets should be separate. + seq, err = tx.Bucket([]byte("woojits")).NextSequence() + ok(t, err) + equals(t, seq, uint64(1)) + return nil + }) +} + +// Ensure that a bucket will persist an autoincrementing sequence even if its +// the only thing updated on the bucket. +// https://github.com/boltdb/bolt/issues/296 +func TestBucket_NextSequence_Persist(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + _, _ = tx.CreateBucket([]byte("widgets")) + return nil + }) + + db.Update(func(tx *bolt.Tx) error { + _, _ = tx.Bucket([]byte("widgets")).NextSequence() + return nil + }) + + db.Update(func(tx *bolt.Tx) error { + seq, err := tx.Bucket([]byte("widgets")).NextSequence() + if err != nil { + t.Fatalf("unexpected error: %s", err) + } else if seq != 2 { + t.Fatalf("unexpected sequence: %d", seq) + } + return nil + }) +} + +// Ensure that retrieving the next sequence on a read-only bucket returns an error. +func TestBucket_NextSequence_ReadOnly(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + return nil + }) + db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("widgets")) + i, err := b.NextSequence() + equals(t, i, uint64(0)) + equals(t, err, bolt.ErrTxNotWritable) + return nil + }) +} + +// Ensure that retrieving the next sequence for a bucket on a closed database return an error. +func TestBucket_NextSequence_Closed(t *testing.T) { + db := NewTestDB() + defer db.Close() + tx, _ := db.Begin(true) + tx.CreateBucket([]byte("widgets")) + b := tx.Bucket([]byte("widgets")) + tx.Rollback() + _, err := b.NextSequence() + equals(t, bolt.ErrTxClosed, err) +} + +// Ensure a user can loop over all key/value pairs in a bucket. +func TestBucket_ForEach(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("0000")) + tx.Bucket([]byte("widgets")).Put([]byte("baz"), []byte("0001")) + tx.Bucket([]byte("widgets")).Put([]byte("bar"), []byte("0002")) + + var index int + err := tx.Bucket([]byte("widgets")).ForEach(func(k, v []byte) error { + switch index { + case 0: + equals(t, k, []byte("bar")) + equals(t, v, []byte("0002")) + case 1: + equals(t, k, []byte("baz")) + equals(t, v, []byte("0001")) + case 2: + equals(t, k, []byte("foo")) + equals(t, v, []byte("0000")) + } + index++ + return nil + }) + ok(t, err) + equals(t, index, 3) + return nil + }) +} + +// Ensure a database can stop iteration early. +func TestBucket_ForEach_ShortCircuit(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.Bucket([]byte("widgets")).Put([]byte("bar"), []byte("0000")) + tx.Bucket([]byte("widgets")).Put([]byte("baz"), []byte("0000")) + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("0000")) + + var index int + err := tx.Bucket([]byte("widgets")).ForEach(func(k, v []byte) error { + index++ + if bytes.Equal(k, []byte("baz")) { + return errors.New("marker") + } + return nil + }) + equals(t, errors.New("marker"), err) + equals(t, 2, index) + return nil + }) +} + +// Ensure that looping over a bucket on a closed database returns an error. +func TestBucket_ForEach_Closed(t *testing.T) { + db := NewTestDB() + defer db.Close() + tx, _ := db.Begin(true) + tx.CreateBucket([]byte("widgets")) + b := tx.Bucket([]byte("widgets")) + tx.Rollback() + err := b.ForEach(func(k, v []byte) error { return nil }) + equals(t, bolt.ErrTxClosed, err) +} + +// Ensure that an error is returned when inserting with an empty key. +func TestBucket_Put_EmptyKey(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + err := tx.Bucket([]byte("widgets")).Put([]byte(""), []byte("bar")) + equals(t, err, bolt.ErrKeyRequired) + err = tx.Bucket([]byte("widgets")).Put(nil, []byte("bar")) + equals(t, err, bolt.ErrKeyRequired) + return nil + }) +} + +// Ensure that an error is returned when inserting with a key that's too large. +func TestBucket_Put_KeyTooLarge(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + err := tx.Bucket([]byte("widgets")).Put(make([]byte, 32769), []byte("bar")) + equals(t, err, bolt.ErrKeyTooLarge) + return nil + }) +} + +// Ensure that an error is returned when inserting a value that's too large. +func TestBucket_Put_ValueTooLarge(t *testing.T) { + if os.Getenv("DRONE") == "true" { + t.Skip("not enough RAM for test") + } + + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + err := tx.Bucket([]byte("widgets")).Put([]byte("foo"), make([]byte, bolt.MaxValueSize+1)) + equals(t, err, bolt.ErrValueTooLarge) + return nil + }) +} + +// Ensure a bucket can calculate stats. +func TestBucket_Stats(t *testing.T) { + db := NewTestDB() + defer db.Close() + + // Add bucket with fewer keys but one big value. + big_key := []byte("really-big-value") + for i := 0; i < 500; i++ { + db.Update(func(tx *bolt.Tx) error { + b, _ := tx.CreateBucketIfNotExists([]byte("woojits")) + return b.Put([]byte(fmt.Sprintf("%03d", i)), []byte(strconv.Itoa(i))) + }) + } + db.Update(func(tx *bolt.Tx) error { + b, _ := tx.CreateBucketIfNotExists([]byte("woojits")) + return b.Put(big_key, []byte(strings.Repeat("*", 10000))) + }) + + db.MustCheck() + db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("woojits")) + stats := b.Stats() + equals(t, 1, stats.BranchPageN) + equals(t, 0, stats.BranchOverflowN) + equals(t, 7, stats.LeafPageN) + equals(t, 2, stats.LeafOverflowN) + equals(t, 501, stats.KeyN) + equals(t, 2, stats.Depth) + + branchInuse := 16 // branch page header + branchInuse += 7 * 16 // branch elements + branchInuse += 7 * 3 // branch keys (6 3-byte keys) + equals(t, branchInuse, stats.BranchInuse) + + leafInuse := 7 * 16 // leaf page header + leafInuse += 501 * 16 // leaf elements + leafInuse += 500*3 + len(big_key) // leaf keys + leafInuse += 1*10 + 2*90 + 3*400 + 10000 // leaf values + equals(t, leafInuse, stats.LeafInuse) + + if os.Getpagesize() == 4096 { + // Incompatible page size + equals(t, 4096, stats.BranchAlloc) + equals(t, 36864, stats.LeafAlloc) + } + + equals(t, 1, stats.BucketN) + equals(t, 0, stats.InlineBucketN) + equals(t, 0, stats.InlineBucketInuse) + return nil + }) +} + +// Ensure a bucket with random insertion utilizes fill percentage correctly. +func TestBucket_Stats_RandomFill(t *testing.T) { + if testing.Short() { + t.Skip("skipping test in short mode.") + } else if os.Getpagesize() != 4096 { + t.Skip("invalid page size for test") + } + + db := NewTestDB() + defer db.Close() + + // Add a set of values in random order. It will be the same random + // order so we can maintain consistency between test runs. + var count int + r := rand.New(rand.NewSource(42)) + for _, i := range r.Perm(1000) { + db.Update(func(tx *bolt.Tx) error { + b, _ := tx.CreateBucketIfNotExists([]byte("woojits")) + b.FillPercent = 0.9 + for _, j := range r.Perm(100) { + index := (j * 10000) + i + b.Put([]byte(fmt.Sprintf("%d000000000000000", index)), []byte("0000000000")) + count++ + } + return nil + }) + } + db.MustCheck() + + db.View(func(tx *bolt.Tx) error { + s := tx.Bucket([]byte("woojits")).Stats() + equals(t, 100000, s.KeyN) + + equals(t, 98, s.BranchPageN) + equals(t, 0, s.BranchOverflowN) + equals(t, 130984, s.BranchInuse) + equals(t, 401408, s.BranchAlloc) + + equals(t, 3412, s.LeafPageN) + equals(t, 0, s.LeafOverflowN) + equals(t, 4742482, s.LeafInuse) + equals(t, 13975552, s.LeafAlloc) + return nil + }) +} + +// Ensure a bucket can calculate stats. +func TestBucket_Stats_Small(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + // Add a bucket that fits on a single root leaf. + b, err := tx.CreateBucket([]byte("whozawhats")) + ok(t, err) + b.Put([]byte("foo"), []byte("bar")) + + return nil + }) + db.MustCheck() + db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("whozawhats")) + stats := b.Stats() + equals(t, 0, stats.BranchPageN) + equals(t, 0, stats.BranchOverflowN) + equals(t, 0, stats.LeafPageN) + equals(t, 0, stats.LeafOverflowN) + equals(t, 1, stats.KeyN) + equals(t, 1, stats.Depth) + equals(t, 0, stats.BranchInuse) + equals(t, 0, stats.LeafInuse) + if os.Getpagesize() == 4096 { + // Incompatible page size + equals(t, 0, stats.BranchAlloc) + equals(t, 0, stats.LeafAlloc) + } + equals(t, 1, stats.BucketN) + equals(t, 1, stats.InlineBucketN) + equals(t, 16+16+6, stats.InlineBucketInuse) + return nil + }) +} + +func TestBucket_Stats_EmptyBucket(t *testing.T) { + db := NewTestDB() + defer db.Close() + + db.Update(func(tx *bolt.Tx) error { + // Add a bucket that fits on a single root leaf. + _, err := tx.CreateBucket([]byte("whozawhats")) + ok(t, err) + return nil + }) + db.MustCheck() + db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("whozawhats")) + stats := b.Stats() + equals(t, 0, stats.BranchPageN) + equals(t, 0, stats.BranchOverflowN) + equals(t, 0, stats.LeafPageN) + equals(t, 0, stats.LeafOverflowN) + equals(t, 0, stats.KeyN) + equals(t, 1, stats.Depth) + equals(t, 0, stats.BranchInuse) + equals(t, 0, stats.LeafInuse) + if os.Getpagesize() == 4096 { + // Incompatible page size + equals(t, 0, stats.BranchAlloc) + equals(t, 0, stats.LeafAlloc) + } + equals(t, 1, stats.BucketN) + equals(t, 1, stats.InlineBucketN) + equals(t, 16, stats.InlineBucketInuse) + return nil + }) +} + +// Ensure a bucket can calculate stats. +func TestBucket_Stats_Nested(t *testing.T) { + db := NewTestDB() + defer db.Close() + + db.Update(func(tx *bolt.Tx) error { + b, err := tx.CreateBucket([]byte("foo")) + ok(t, err) + for i := 0; i < 100; i++ { + b.Put([]byte(fmt.Sprintf("%02d", i)), []byte(fmt.Sprintf("%02d", i))) + } + bar, err := b.CreateBucket([]byte("bar")) + ok(t, err) + for i := 0; i < 10; i++ { + bar.Put([]byte(strconv.Itoa(i)), []byte(strconv.Itoa(i))) + } + baz, err := bar.CreateBucket([]byte("baz")) + ok(t, err) + for i := 0; i < 10; i++ { + baz.Put([]byte(strconv.Itoa(i)), []byte(strconv.Itoa(i))) + } + return nil + }) + + db.MustCheck() + + db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("foo")) + stats := b.Stats() + equals(t, 0, stats.BranchPageN) + equals(t, 0, stats.BranchOverflowN) + equals(t, 2, stats.LeafPageN) + equals(t, 0, stats.LeafOverflowN) + equals(t, 122, stats.KeyN) + equals(t, 3, stats.Depth) + equals(t, 0, stats.BranchInuse) + + foo := 16 // foo (pghdr) + foo += 101 * 16 // foo leaf elements + foo += 100*2 + 100*2 // foo leaf key/values + foo += 3 + 16 // foo -> bar key/value + + bar := 16 // bar (pghdr) + bar += 11 * 16 // bar leaf elements + bar += 10 + 10 // bar leaf key/values + bar += 3 + 16 // bar -> baz key/value + + baz := 16 // baz (inline) (pghdr) + baz += 10 * 16 // baz leaf elements + baz += 10 + 10 // baz leaf key/values + + equals(t, foo+bar+baz, stats.LeafInuse) + if os.Getpagesize() == 4096 { + // Incompatible page size + equals(t, 0, stats.BranchAlloc) + equals(t, 8192, stats.LeafAlloc) + } + equals(t, 3, stats.BucketN) + equals(t, 1, stats.InlineBucketN) + equals(t, baz, stats.InlineBucketInuse) + return nil + }) +} + +// Ensure a large bucket can calculate stats. +func TestBucket_Stats_Large(t *testing.T) { + if testing.Short() { + t.Skip("skipping test in short mode.") + } + + db := NewTestDB() + defer db.Close() + + var index int + for i := 0; i < 100; i++ { + db.Update(func(tx *bolt.Tx) error { + // Add bucket with lots of keys. + b, _ := tx.CreateBucketIfNotExists([]byte("widgets")) + for i := 0; i < 1000; i++ { + b.Put([]byte(strconv.Itoa(index)), []byte(strconv.Itoa(index))) + index++ + } + return nil + }) + } + db.MustCheck() + + db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("widgets")) + stats := b.Stats() + equals(t, 13, stats.BranchPageN) + equals(t, 0, stats.BranchOverflowN) + equals(t, 1196, stats.LeafPageN) + equals(t, 0, stats.LeafOverflowN) + equals(t, 100000, stats.KeyN) + equals(t, 3, stats.Depth) + equals(t, 25257, stats.BranchInuse) + equals(t, 2596916, stats.LeafInuse) + if os.Getpagesize() == 4096 { + // Incompatible page size + equals(t, 53248, stats.BranchAlloc) + equals(t, 4898816, stats.LeafAlloc) + } + equals(t, 1, stats.BucketN) + equals(t, 0, stats.InlineBucketN) + equals(t, 0, stats.InlineBucketInuse) + return nil + }) +} + +// Ensure that a bucket can write random keys and values across multiple transactions. +func TestBucket_Put_Single(t *testing.T) { + if testing.Short() { + t.Skip("skipping test in short mode.") + } + + index := 0 + f := func(items testdata) bool { + db := NewTestDB() + defer db.Close() + + m := make(map[string][]byte) + + db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte("widgets")) + return err + }) + for _, item := range items { + db.Update(func(tx *bolt.Tx) error { + if err := tx.Bucket([]byte("widgets")).Put(item.Key, item.Value); err != nil { + panic("put error: " + err.Error()) + } + m[string(item.Key)] = item.Value + return nil + }) + + // Verify all key/values so far. + db.View(func(tx *bolt.Tx) error { + i := 0 + for k, v := range m { + value := tx.Bucket([]byte("widgets")).Get([]byte(k)) + if !bytes.Equal(value, v) { + t.Logf("value mismatch [run %d] (%d of %d):\nkey: %x\ngot: %x\nexp: %x", index, i, len(m), []byte(k), value, v) + db.CopyTempFile() + t.FailNow() + } + i++ + } + return nil + }) + } + + index++ + return true + } + if err := quick.Check(f, qconfig()); err != nil { + t.Error(err) + } +} + +// Ensure that a transaction can insert multiple key/value pairs at once. +func TestBucket_Put_Multiple(t *testing.T) { + if testing.Short() { + t.Skip("skipping test in short mode.") + } + + f := func(items testdata) bool { + db := NewTestDB() + defer db.Close() + // Bulk insert all values. + db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte("widgets")) + return err + }) + err := db.Update(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("widgets")) + for _, item := range items { + ok(t, b.Put(item.Key, item.Value)) + } + return nil + }) + ok(t, err) + + // Verify all items exist. + db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("widgets")) + for _, item := range items { + value := b.Get(item.Key) + if !bytes.Equal(item.Value, value) { + db.CopyTempFile() + t.Fatalf("exp=%x; got=%x", item.Value, value) + } + } + return nil + }) + return true + } + if err := quick.Check(f, qconfig()); err != nil { + t.Error(err) + } +} + +// Ensure that a transaction can delete all key/value pairs and return to a single leaf page. +func TestBucket_Delete_Quick(t *testing.T) { + if testing.Short() { + t.Skip("skipping test in short mode.") + } + + f := func(items testdata) bool { + db := NewTestDB() + defer db.Close() + // Bulk insert all values. + db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte("widgets")) + return err + }) + err := db.Update(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("widgets")) + for _, item := range items { + ok(t, b.Put(item.Key, item.Value)) + } + return nil + }) + ok(t, err) + + // Remove items one at a time and check consistency. + for _, item := range items { + err := db.Update(func(tx *bolt.Tx) error { + return tx.Bucket([]byte("widgets")).Delete(item.Key) + }) + ok(t, err) + } + + // Anything before our deletion index should be nil. + db.View(func(tx *bolt.Tx) error { + tx.Bucket([]byte("widgets")).ForEach(func(k, v []byte) error { + t.Fatalf("bucket should be empty; found: %06x", trunc(k, 3)) + return nil + }) + return nil + }) + return true + } + if err := quick.Check(f, qconfig()); err != nil { + t.Error(err) + } +} + +func ExampleBucket_Put() { + // Open the database. + db, _ := bolt.Open(tempfile(), 0666, nil) + defer os.Remove(db.Path()) + defer db.Close() + + // Start a write transaction. + db.Update(func(tx *bolt.Tx) error { + // Create a bucket. + tx.CreateBucket([]byte("widgets")) + + // Set the value "bar" for the key "foo". + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar")) + return nil + }) + + // Read value back in a different read-only transaction. + db.View(func(tx *bolt.Tx) error { + value := tx.Bucket([]byte("widgets")).Get([]byte("foo")) + fmt.Printf("The value of 'foo' is: %s\n", value) + return nil + }) + + // Output: + // The value of 'foo' is: bar +} + +func ExampleBucket_Delete() { + // Open the database. + db, _ := bolt.Open(tempfile(), 0666, nil) + defer os.Remove(db.Path()) + defer db.Close() + + // Start a write transaction. + db.Update(func(tx *bolt.Tx) error { + // Create a bucket. + tx.CreateBucket([]byte("widgets")) + b := tx.Bucket([]byte("widgets")) + + // Set the value "bar" for the key "foo". + b.Put([]byte("foo"), []byte("bar")) + + // Retrieve the key back from the database and verify it. + value := b.Get([]byte("foo")) + fmt.Printf("The value of 'foo' was: %s\n", value) + return nil + }) + + // Delete the key in a different write transaction. + db.Update(func(tx *bolt.Tx) error { + return tx.Bucket([]byte("widgets")).Delete([]byte("foo")) + }) + + // Retrieve the key again. + db.View(func(tx *bolt.Tx) error { + value := tx.Bucket([]byte("widgets")).Get([]byte("foo")) + if value == nil { + fmt.Printf("The value of 'foo' is now: nil\n") + } + return nil + }) + + // Output: + // The value of 'foo' was: bar + // The value of 'foo' is now: nil +} + +func ExampleBucket_ForEach() { + // Open the database. + db, _ := bolt.Open(tempfile(), 0666, nil) + defer os.Remove(db.Path()) + defer db.Close() + + // Insert data into a bucket. + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("animals")) + b := tx.Bucket([]byte("animals")) + b.Put([]byte("dog"), []byte("fun")) + b.Put([]byte("cat"), []byte("lame")) + b.Put([]byte("liger"), []byte("awesome")) + + // Iterate over items in sorted key order. + b.ForEach(func(k, v []byte) error { + fmt.Printf("A %s is %s.\n", k, v) + return nil + }) + return nil + }) + + // Output: + // A cat is lame. + // A dog is fun. + // A liger is awesome. +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/cmd/bolt/main.go b/Godeps/_workspace/src/github.com/boltdb/bolt/cmd/bolt/main.go new file mode 100644 index 0000000000000..c41ebe404d959 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/cmd/bolt/main.go @@ -0,0 +1,1529 @@ +package main + +import ( + "bytes" + "encoding/binary" + "errors" + "flag" + "fmt" + "io" + "io/ioutil" + "math/rand" + "os" + "runtime" + "runtime/pprof" + "strconv" + "strings" + "time" + "unicode" + "unicode/utf8" + "unsafe" + + "github.com/boltdb/bolt" +) + +var ( + // ErrUsage is returned when a usage message was printed and the process + // should simply exit with an error. + ErrUsage = errors.New("usage") + + // ErrUnknownCommand is returned when a CLI command is not specified. + ErrUnknownCommand = errors.New("unknown command") + + // ErrPathRequired is returned when the path to a Bolt database is not specified. + ErrPathRequired = errors.New("path required") + + // ErrFileNotFound is returned when a Bolt database does not exist. + ErrFileNotFound = errors.New("file not found") + + // ErrInvalidValue is returned when a benchmark reads an unexpected value. + ErrInvalidValue = errors.New("invalid value") + + // ErrCorrupt is returned when a checking a data file finds errors. + ErrCorrupt = errors.New("invalid value") + + // ErrNonDivisibleBatchSize is returned when the batch size can't be evenly + // divided by the iteration count. + ErrNonDivisibleBatchSize = errors.New("number of iterations must be divisible by the batch size") + + // ErrPageIDRequired is returned when a required page id is not specified. + ErrPageIDRequired = errors.New("page id required") + + // ErrPageNotFound is returned when specifying a page above the high water mark. + ErrPageNotFound = errors.New("page not found") + + // ErrPageFreed is returned when reading a page that has already been freed. + ErrPageFreed = errors.New("page freed") +) + +// PageHeaderSize represents the size of the bolt.page header. +const PageHeaderSize = 16 + +func main() { + m := NewMain() + if err := m.Run(os.Args[1:]...); err == ErrUsage { + os.Exit(2) + } else if err != nil { + fmt.Println(err.Error()) + os.Exit(1) + } +} + +// Main represents the main program execution. +type Main struct { + Stdin io.Reader + Stdout io.Writer + Stderr io.Writer +} + +// NewMain returns a new instance of Main connect to the standard input/output. +func NewMain() *Main { + return &Main{ + Stdin: os.Stdin, + Stdout: os.Stdout, + Stderr: os.Stderr, + } +} + +// Run executes the program. +func (m *Main) Run(args ...string) error { + // Require a command at the beginning. + if len(args) == 0 || strings.HasPrefix(args[0], "-") { + fmt.Fprintln(m.Stderr, m.Usage()) + return ErrUsage + } + + // Execute command. + switch args[0] { + case "help": + fmt.Fprintln(m.Stderr, m.Usage()) + return ErrUsage + case "bench": + return newBenchCommand(m).Run(args[1:]...) + case "check": + return newCheckCommand(m).Run(args[1:]...) + case "dump": + return newDumpCommand(m).Run(args[1:]...) + case "info": + return newInfoCommand(m).Run(args[1:]...) + case "page": + return newPageCommand(m).Run(args[1:]...) + case "pages": + return newPagesCommand(m).Run(args[1:]...) + case "stats": + return newStatsCommand(m).Run(args[1:]...) + default: + return ErrUnknownCommand + } +} + +// Usage returns the help message. +func (m *Main) Usage() string { + return strings.TrimLeft(` +Bolt is a tool for inspecting bolt databases. + +Usage: + + bolt command [arguments] + +The commands are: + + bench run synthetic benchmark against bolt + check verifies integrity of bolt database + info print basic info + help print this screen + pages print list of pages with their types + stats iterate over all pages and generate usage stats + +Use "bolt [command] -h" for more information about a command. +`, "\n") +} + +// CheckCommand represents the "check" command execution. +type CheckCommand struct { + Stdin io.Reader + Stdout io.Writer + Stderr io.Writer +} + +// NewCheckCommand returns a CheckCommand. +func newCheckCommand(m *Main) *CheckCommand { + return &CheckCommand{ + Stdin: m.Stdin, + Stdout: m.Stdout, + Stderr: m.Stderr, + } +} + +// Run executes the command. +func (cmd *CheckCommand) Run(args ...string) error { + // Parse flags. + fs := flag.NewFlagSet("", flag.ContinueOnError) + help := fs.Bool("h", false, "") + if err := fs.Parse(args); err != nil { + return err + } else if *help { + fmt.Fprintln(cmd.Stderr, cmd.Usage()) + return ErrUsage + } + + // Require database path. + path := fs.Arg(0) + if path == "" { + return ErrPathRequired + } else if _, err := os.Stat(path); os.IsNotExist(err) { + return ErrFileNotFound + } + + // Open database. + db, err := bolt.Open(path, 0666, nil) + if err != nil { + return err + } + defer db.Close() + + // Perform consistency check. + return db.View(func(tx *bolt.Tx) error { + var count int + ch := tx.Check() + loop: + for { + select { + case err, ok := <-ch: + if !ok { + break loop + } + fmt.Fprintln(cmd.Stdout, err) + count++ + } + } + + // Print summary of errors. + if count > 0 { + fmt.Fprintf(cmd.Stdout, "%d errors found\n", count) + return ErrCorrupt + } + + // Notify user that database is valid. + fmt.Fprintln(cmd.Stdout, "OK") + return nil + }) +} + +// Usage returns the help message. +func (cmd *CheckCommand) Usage() string { + return strings.TrimLeft(` +usage: bolt check PATH + +Check opens a database at PATH and runs an exhaustive check to verify that +all pages are accessible or are marked as freed. It also verifies that no +pages are double referenced. + +Verification errors will stream out as they are found and the process will +return after all pages have been checked. +`, "\n") +} + +// InfoCommand represents the "info" command execution. +type InfoCommand struct { + Stdin io.Reader + Stdout io.Writer + Stderr io.Writer +} + +// NewInfoCommand returns a InfoCommand. +func newInfoCommand(m *Main) *InfoCommand { + return &InfoCommand{ + Stdin: m.Stdin, + Stdout: m.Stdout, + Stderr: m.Stderr, + } +} + +// Run executes the command. +func (cmd *InfoCommand) Run(args ...string) error { + // Parse flags. + fs := flag.NewFlagSet("", flag.ContinueOnError) + help := fs.Bool("h", false, "") + if err := fs.Parse(args); err != nil { + return err + } else if *help { + fmt.Fprintln(cmd.Stderr, cmd.Usage()) + return ErrUsage + } + + // Require database path. + path := fs.Arg(0) + if path == "" { + return ErrPathRequired + } else if _, err := os.Stat(path); os.IsNotExist(err) { + return ErrFileNotFound + } + + // Open the database. + db, err := bolt.Open(path, 0666, nil) + if err != nil { + return err + } + defer db.Close() + + // Print basic database info. + info := db.Info() + fmt.Fprintf(cmd.Stdout, "Page Size: %d\n", info.PageSize) + + return nil +} + +// Usage returns the help message. +func (cmd *InfoCommand) Usage() string { + return strings.TrimLeft(` +usage: bolt info PATH + +Info prints basic information about the Bolt database at PATH. +`, "\n") +} + +// DumpCommand represents the "dump" command execution. +type DumpCommand struct { + Stdin io.Reader + Stdout io.Writer + Stderr io.Writer +} + +// newDumpCommand returns a DumpCommand. +func newDumpCommand(m *Main) *DumpCommand { + return &DumpCommand{ + Stdin: m.Stdin, + Stdout: m.Stdout, + Stderr: m.Stderr, + } +} + +// Run executes the command. +func (cmd *DumpCommand) Run(args ...string) error { + // Parse flags. + fs := flag.NewFlagSet("", flag.ContinueOnError) + help := fs.Bool("h", false, "") + if err := fs.Parse(args); err != nil { + return err + } else if *help { + fmt.Fprintln(cmd.Stderr, cmd.Usage()) + return ErrUsage + } + + // Require database path and page id. + path := fs.Arg(0) + if path == "" { + return ErrPathRequired + } else if _, err := os.Stat(path); os.IsNotExist(err) { + return ErrFileNotFound + } + + // Read page ids. + pageIDs, err := atois(fs.Args()[1:]) + if err != nil { + return err + } else if len(pageIDs) == 0 { + return ErrPageIDRequired + } + + // Open database to retrieve page size. + pageSize, err := ReadPageSize(path) + if err != nil { + return err + } + + // Open database file handler. + f, err := os.Open(path) + if err != nil { + return err + } + defer func() { _ = f.Close() }() + + // Print each page listed. + for i, pageID := range pageIDs { + // Print a separator. + if i > 0 { + fmt.Fprintln(cmd.Stdout, "===============================================") + } + + // Print page to stdout. + if err := cmd.PrintPage(cmd.Stdout, f, pageID, pageSize); err != nil { + return err + } + } + + return nil +} + +// PrintPage prints a given page as hexidecimal. +func (cmd *DumpCommand) PrintPage(w io.Writer, r io.ReaderAt, pageID int, pageSize int) error { + const bytesPerLineN = 16 + + // Read page into buffer. + buf := make([]byte, pageSize) + addr := pageID * pageSize + if n, err := r.ReadAt(buf, int64(addr)); err != nil { + return err + } else if n != pageSize { + return io.ErrUnexpectedEOF + } + + // Write out to writer in 16-byte lines. + var prev []byte + var skipped bool + for offset := 0; offset < pageSize; offset += bytesPerLineN { + // Retrieve current 16-byte line. + line := buf[offset : offset+bytesPerLineN] + isLastLine := (offset == (pageSize - bytesPerLineN)) + + // If it's the same as the previous line then print a skip. + if bytes.Equal(line, prev) && !isLastLine { + if !skipped { + fmt.Fprintf(w, "%07x *\n", addr+offset) + skipped = true + } + } else { + // Print line as hexadecimal in 2-byte groups. + fmt.Fprintf(w, "%07x %04x %04x %04x %04x %04x %04x %04x %04x\n", addr+offset, + line[0:2], line[2:4], line[4:6], line[6:8], + line[8:10], line[10:12], line[12:14], line[14:16], + ) + + skipped = false + } + + // Save the previous line. + prev = line + } + fmt.Fprint(w, "\n") + + return nil +} + +// Usage returns the help message. +func (cmd *DumpCommand) Usage() string { + return strings.TrimLeft(` +usage: bolt dump -page PAGEID PATH + +Dump prints a hexidecimal dump of a single page. +`, "\n") +} + +// PageCommand represents the "page" command execution. +type PageCommand struct { + Stdin io.Reader + Stdout io.Writer + Stderr io.Writer +} + +// newPageCommand returns a PageCommand. +func newPageCommand(m *Main) *PageCommand { + return &PageCommand{ + Stdin: m.Stdin, + Stdout: m.Stdout, + Stderr: m.Stderr, + } +} + +// Run executes the command. +func (cmd *PageCommand) Run(args ...string) error { + // Parse flags. + fs := flag.NewFlagSet("", flag.ContinueOnError) + help := fs.Bool("h", false, "") + if err := fs.Parse(args); err != nil { + return err + } else if *help { + fmt.Fprintln(cmd.Stderr, cmd.Usage()) + return ErrUsage + } + + // Require database path and page id. + path := fs.Arg(0) + if path == "" { + return ErrPathRequired + } else if _, err := os.Stat(path); os.IsNotExist(err) { + return ErrFileNotFound + } + + // Read page ids. + pageIDs, err := atois(fs.Args()[1:]) + if err != nil { + return err + } else if len(pageIDs) == 0 { + return ErrPageIDRequired + } + + // Open database file handler. + f, err := os.Open(path) + if err != nil { + return err + } + defer func() { _ = f.Close() }() + + // Print each page listed. + for i, pageID := range pageIDs { + // Print a separator. + if i > 0 { + fmt.Fprintln(cmd.Stdout, "===============================================") + } + + // Retrieve page info and page size. + p, buf, err := ReadPage(path, pageID) + if err != nil { + return err + } + + // Print basic page info. + fmt.Fprintf(cmd.Stdout, "Page ID: %d\n", p.id) + fmt.Fprintf(cmd.Stdout, "Page Type: %s\n", p.Type()) + fmt.Fprintf(cmd.Stdout, "Total Size: %d bytes\n", len(buf)) + + // Print type-specific data. + switch p.Type() { + case "meta": + err = cmd.PrintMeta(cmd.Stdout, buf) + case "leaf": + err = cmd.PrintLeaf(cmd.Stdout, buf) + case "branch": + err = cmd.PrintBranch(cmd.Stdout, buf) + case "freelist": + err = cmd.PrintFreelist(cmd.Stdout, buf) + } + if err != nil { + return err + } + } + + return nil +} + +// PrintMeta prints the data from the meta page. +func (cmd *PageCommand) PrintMeta(w io.Writer, buf []byte) error { + m := (*meta)(unsafe.Pointer(&buf[PageHeaderSize])) + fmt.Fprintf(w, "Version: %d\n", m.version) + fmt.Fprintf(w, "Page Size: %d bytes\n", m.pageSize) + fmt.Fprintf(w, "Flags: %08x\n", m.flags) + fmt.Fprintf(w, "Root: \n", m.root.root) + fmt.Fprintf(w, "Freelist: \n", m.freelist) + fmt.Fprintf(w, "HWM: \n", m.pgid) + fmt.Fprintf(w, "Txn ID: %d\n", m.txid) + fmt.Fprintf(w, "Checksum: %016x\n", m.checksum) + fmt.Fprintf(w, "\n") + return nil +} + +// PrintLeaf prints the data for a leaf page. +func (cmd *PageCommand) PrintLeaf(w io.Writer, buf []byte) error { + p := (*page)(unsafe.Pointer(&buf[0])) + + // Print number of items. + fmt.Fprintf(w, "Item Count: %d\n", p.count) + fmt.Fprintf(w, "\n") + + // Print each key/value. + for i := uint16(0); i < p.count; i++ { + e := p.leafPageElement(i) + + // Format key as string. + var k string + if isPrintable(string(e.key())) { + k = fmt.Sprintf("%q", string(e.key())) + } else { + k = fmt.Sprintf("%x", string(e.key())) + } + + // Format value as string. + var v string + if (e.flags & uint32(bucketLeafFlag)) != 0 { + b := (*bucket)(unsafe.Pointer(&e.value()[0])) + v = fmt.Sprintf("", b.root, b.sequence) + } else if isPrintable(string(e.value())) { + k = fmt.Sprintf("%q", string(e.value())) + } else { + k = fmt.Sprintf("%x", string(e.value())) + } + + fmt.Fprintf(w, "%s: %s\n", k, v) + } + fmt.Fprintf(w, "\n") + return nil +} + +// PrintBranch prints the data for a leaf page. +func (cmd *PageCommand) PrintBranch(w io.Writer, buf []byte) error { + p := (*page)(unsafe.Pointer(&buf[0])) + + // Print number of items. + fmt.Fprintf(w, "Item Count: %d\n", p.count) + fmt.Fprintf(w, "\n") + + // Print each key/value. + for i := uint16(0); i < p.count; i++ { + e := p.branchPageElement(i) + + // Format key as string. + var k string + if isPrintable(string(e.key())) { + k = fmt.Sprintf("%q", string(e.key())) + } else { + k = fmt.Sprintf("%x", string(e.key())) + } + + fmt.Fprintf(w, "%s: \n", k, e.pgid) + } + fmt.Fprintf(w, "\n") + return nil +} + +// PrintFreelist prints the data for a freelist page. +func (cmd *PageCommand) PrintFreelist(w io.Writer, buf []byte) error { + p := (*page)(unsafe.Pointer(&buf[0])) + + // Print number of items. + fmt.Fprintf(w, "Item Count: %d\n", p.count) + fmt.Fprintf(w, "\n") + + // Print each page in the freelist. + ids := (*[maxAllocSize]pgid)(unsafe.Pointer(&p.ptr)) + for i := uint16(0); i < p.count; i++ { + fmt.Fprintf(w, "%d\n", ids[i]) + } + fmt.Fprintf(w, "\n") + return nil +} + +// PrintPage prints a given page as hexidecimal. +func (cmd *PageCommand) PrintPage(w io.Writer, r io.ReaderAt, pageID int, pageSize int) error { + const bytesPerLineN = 16 + + // Read page into buffer. + buf := make([]byte, pageSize) + addr := pageID * pageSize + if n, err := r.ReadAt(buf, int64(addr)); err != nil { + return err + } else if n != pageSize { + return io.ErrUnexpectedEOF + } + + // Write out to writer in 16-byte lines. + var prev []byte + var skipped bool + for offset := 0; offset < pageSize; offset += bytesPerLineN { + // Retrieve current 16-byte line. + line := buf[offset : offset+bytesPerLineN] + isLastLine := (offset == (pageSize - bytesPerLineN)) + + // If it's the same as the previous line then print a skip. + if bytes.Equal(line, prev) && !isLastLine { + if !skipped { + fmt.Fprintf(w, "%07x *\n", addr+offset) + skipped = true + } + } else { + // Print line as hexadecimal in 2-byte groups. + fmt.Fprintf(w, "%07x %04x %04x %04x %04x %04x %04x %04x %04x\n", addr+offset, + line[0:2], line[2:4], line[4:6], line[6:8], + line[8:10], line[10:12], line[12:14], line[14:16], + ) + + skipped = false + } + + // Save the previous line. + prev = line + } + fmt.Fprint(w, "\n") + + return nil +} + +// Usage returns the help message. +func (cmd *PageCommand) Usage() string { + return strings.TrimLeft(` +usage: bolt page -page PATH pageid [pageid...] + +Page prints one or more pages in human readable format. +`, "\n") +} + +// PagesCommand represents the "pages" command execution. +type PagesCommand struct { + Stdin io.Reader + Stdout io.Writer + Stderr io.Writer +} + +// NewPagesCommand returns a PagesCommand. +func newPagesCommand(m *Main) *PagesCommand { + return &PagesCommand{ + Stdin: m.Stdin, + Stdout: m.Stdout, + Stderr: m.Stderr, + } +} + +// Run executes the command. +func (cmd *PagesCommand) Run(args ...string) error { + // Parse flags. + fs := flag.NewFlagSet("", flag.ContinueOnError) + help := fs.Bool("h", false, "") + if err := fs.Parse(args); err != nil { + return err + } else if *help { + fmt.Fprintln(cmd.Stderr, cmd.Usage()) + return ErrUsage + } + + // Require database path. + path := fs.Arg(0) + if path == "" { + return ErrPathRequired + } else if _, err := os.Stat(path); os.IsNotExist(err) { + return ErrFileNotFound + } + + // Open database. + db, err := bolt.Open(path, 0666, nil) + if err != nil { + return err + } + defer func() { _ = db.Close() }() + + // Write header. + fmt.Fprintln(cmd.Stdout, "ID TYPE ITEMS OVRFLW") + fmt.Fprintln(cmd.Stdout, "======== ========== ====== ======") + + return db.Update(func(tx *bolt.Tx) error { + var id int + for { + p, err := tx.Page(id) + if err != nil { + return &PageError{ID: id, Err: err} + } else if p == nil { + break + } + + // Only display count and overflow if this is a non-free page. + var count, overflow string + if p.Type != "free" { + count = strconv.Itoa(p.Count) + if p.OverflowCount > 0 { + overflow = strconv.Itoa(p.OverflowCount) + } + } + + // Print table row. + fmt.Fprintf(cmd.Stdout, "%-8d %-10s %-6s %-6s\n", p.ID, p.Type, count, overflow) + + // Move to the next non-overflow page. + id += 1 + if p.Type != "free" { + id += p.OverflowCount + } + } + return nil + }) +} + +// Usage returns the help message. +func (cmd *PagesCommand) Usage() string { + return strings.TrimLeft(` +usage: bolt pages PATH + +Pages prints a table of pages with their type (meta, leaf, branch, freelist). +Leaf and branch pages will show a key count in the "items" column while the +freelist will show the number of free pages in the "items" column. + +The "overflow" column shows the number of blocks that the page spills over +into. Normally there is no overflow but large keys and values can cause +a single page to take up multiple blocks. +`, "\n") +} + +// StatsCommand represents the "stats" command execution. +type StatsCommand struct { + Stdin io.Reader + Stdout io.Writer + Stderr io.Writer +} + +// NewStatsCommand returns a StatsCommand. +func newStatsCommand(m *Main) *StatsCommand { + return &StatsCommand{ + Stdin: m.Stdin, + Stdout: m.Stdout, + Stderr: m.Stderr, + } +} + +// Run executes the command. +func (cmd *StatsCommand) Run(args ...string) error { + // Parse flags. + fs := flag.NewFlagSet("", flag.ContinueOnError) + help := fs.Bool("h", false, "") + if err := fs.Parse(args); err != nil { + return err + } else if *help { + fmt.Fprintln(cmd.Stderr, cmd.Usage()) + return ErrUsage + } + + // Require database path. + path, prefix := fs.Arg(0), fs.Arg(1) + if path == "" { + return ErrPathRequired + } else if _, err := os.Stat(path); os.IsNotExist(err) { + return ErrFileNotFound + } + + // Open database. + db, err := bolt.Open(path, 0666, nil) + if err != nil { + return err + } + defer db.Close() + + return db.View(func(tx *bolt.Tx) error { + var s bolt.BucketStats + var count int + if err := tx.ForEach(func(name []byte, b *bolt.Bucket) error { + if bytes.HasPrefix(name, []byte(prefix)) { + s.Add(b.Stats()) + count += 1 + } + return nil + }); err != nil { + return err + } + + fmt.Fprintf(cmd.Stdout, "Aggregate statistics for %d buckets\n\n", count) + + fmt.Fprintln(cmd.Stdout, "Page count statistics") + fmt.Fprintf(cmd.Stdout, "\tNumber of logical branch pages: %d\n", s.BranchPageN) + fmt.Fprintf(cmd.Stdout, "\tNumber of physical branch overflow pages: %d\n", s.BranchOverflowN) + fmt.Fprintf(cmd.Stdout, "\tNumber of logical leaf pages: %d\n", s.LeafPageN) + fmt.Fprintf(cmd.Stdout, "\tNumber of physical leaf overflow pages: %d\n", s.LeafOverflowN) + + fmt.Fprintln(cmd.Stdout, "Tree statistics") + fmt.Fprintf(cmd.Stdout, "\tNumber of keys/value pairs: %d\n", s.KeyN) + fmt.Fprintf(cmd.Stdout, "\tNumber of levels in B+tree: %d\n", s.Depth) + + fmt.Fprintln(cmd.Stdout, "Page size utilization") + fmt.Fprintf(cmd.Stdout, "\tBytes allocated for physical branch pages: %d\n", s.BranchAlloc) + var percentage int + if s.BranchAlloc != 0 { + percentage = int(float32(s.BranchInuse) * 100.0 / float32(s.BranchAlloc)) + } + fmt.Fprintf(cmd.Stdout, "\tBytes actually used for branch data: %d (%d%%)\n", s.BranchInuse, percentage) + fmt.Fprintf(cmd.Stdout, "\tBytes allocated for physical leaf pages: %d\n", s.LeafAlloc) + percentage = 0 + if s.LeafAlloc != 0 { + percentage = int(float32(s.LeafInuse) * 100.0 / float32(s.LeafAlloc)) + } + fmt.Fprintf(cmd.Stdout, "\tBytes actually used for leaf data: %d (%d%%)\n", s.LeafInuse, percentage) + + fmt.Fprintln(cmd.Stdout, "Bucket statistics") + fmt.Fprintf(cmd.Stdout, "\tTotal number of buckets: %d\n", s.BucketN) + percentage = int(float32(s.InlineBucketN) * 100.0 / float32(s.BucketN)) + fmt.Fprintf(cmd.Stdout, "\tTotal number on inlined buckets: %d (%d%%)\n", s.InlineBucketN, percentage) + percentage = 0 + if s.LeafInuse != 0 { + percentage = int(float32(s.InlineBucketInuse) * 100.0 / float32(s.LeafInuse)) + } + fmt.Fprintf(cmd.Stdout, "\tBytes used for inlined buckets: %d (%d%%)\n", s.InlineBucketInuse, percentage) + + return nil + }) +} + +// Usage returns the help message. +func (cmd *StatsCommand) Usage() string { + return strings.TrimLeft(` +usage: bolt stats PATH + +Stats performs an extensive search of the database to track every page +reference. It starts at the current meta page and recursively iterates +through every accessible bucket. + +The following errors can be reported: + + already freed + The page is referenced more than once in the freelist. + + unreachable unfreed + The page is not referenced by a bucket or in the freelist. + + reachable freed + The page is referenced by a bucket but is also in the freelist. + + out of bounds + A page is referenced that is above the high water mark. + + multiple references + A page is referenced by more than one other page. + + invalid type + The page type is not "meta", "leaf", "branch", or "freelist". + +No errors should occur in your database. However, if for some reason you +experience corruption, please submit a ticket to the Bolt project page: + + https://github.com/boltdb/bolt/issues +`, "\n") +} + +var benchBucketName = []byte("bench") + +// BenchCommand represents the "bench" command execution. +type BenchCommand struct { + Stdin io.Reader + Stdout io.Writer + Stderr io.Writer +} + +// NewBenchCommand returns a BenchCommand using the +func newBenchCommand(m *Main) *BenchCommand { + return &BenchCommand{ + Stdin: m.Stdin, + Stdout: m.Stdout, + Stderr: m.Stderr, + } +} + +// Run executes the "bench" command. +func (cmd *BenchCommand) Run(args ...string) error { + // Parse CLI arguments. + options, err := cmd.ParseFlags(args) + if err != nil { + return err + } + + // Remove path if "-work" is not set. Otherwise keep path. + if options.Work { + fmt.Fprintf(cmd.Stdout, "work: %s\n", options.Path) + } else { + defer os.Remove(options.Path) + } + + // Create database. + db, err := bolt.Open(options.Path, 0666, nil) + if err != nil { + return err + } + db.NoSync = options.NoSync + defer db.Close() + + // Write to the database. + var results BenchResults + if err := cmd.runWrites(db, options, &results); err != nil { + return fmt.Errorf("write: %v", err) + } + + // Read from the database. + if err := cmd.runReads(db, options, &results); err != nil { + return fmt.Errorf("bench: read: %s", err) + } + + // Print results. + fmt.Fprintf(os.Stderr, "# Write\t%v\t(%v/op)\t(%v op/sec)\n", results.WriteDuration, results.WriteOpDuration(), results.WriteOpsPerSecond()) + fmt.Fprintf(os.Stderr, "# Read\t%v\t(%v/op)\t(%v op/sec)\n", results.ReadDuration, results.ReadOpDuration(), results.ReadOpsPerSecond()) + fmt.Fprintln(os.Stderr, "") + return nil +} + +// ParseFlags parses the command line flags. +func (cmd *BenchCommand) ParseFlags(args []string) (*BenchOptions, error) { + var options BenchOptions + + // Parse flagset. + fs := flag.NewFlagSet("", flag.ContinueOnError) + fs.StringVar(&options.ProfileMode, "profile-mode", "rw", "") + fs.StringVar(&options.WriteMode, "write-mode", "seq", "") + fs.StringVar(&options.ReadMode, "read-mode", "seq", "") + fs.IntVar(&options.Iterations, "count", 1000, "") + fs.IntVar(&options.BatchSize, "batch-size", 0, "") + fs.IntVar(&options.KeySize, "key-size", 8, "") + fs.IntVar(&options.ValueSize, "value-size", 32, "") + fs.StringVar(&options.CPUProfile, "cpuprofile", "", "") + fs.StringVar(&options.MemProfile, "memprofile", "", "") + fs.StringVar(&options.BlockProfile, "blockprofile", "", "") + fs.Float64Var(&options.FillPercent, "fill-percent", bolt.DefaultFillPercent, "") + fs.BoolVar(&options.NoSync, "no-sync", false, "") + fs.BoolVar(&options.Work, "work", false, "") + fs.StringVar(&options.Path, "path", "", "") + fs.SetOutput(cmd.Stderr) + if err := fs.Parse(args); err != nil { + return nil, err + } + + // Set batch size to iteration size if not set. + // Require that batch size can be evenly divided by the iteration count. + if options.BatchSize == 0 { + options.BatchSize = options.Iterations + } else if options.Iterations%options.BatchSize != 0 { + return nil, ErrNonDivisibleBatchSize + } + + // Generate temp path if one is not passed in. + if options.Path == "" { + f, err := ioutil.TempFile("", "bolt-bench-") + if err != nil { + return nil, fmt.Errorf("temp file: %s", err) + } + f.Close() + os.Remove(f.Name()) + options.Path = f.Name() + } + + return &options, nil +} + +// Writes to the database. +func (cmd *BenchCommand) runWrites(db *bolt.DB, options *BenchOptions, results *BenchResults) error { + // Start profiling for writes. + if options.ProfileMode == "rw" || options.ProfileMode == "w" { + cmd.startProfiling(options) + } + + t := time.Now() + + var err error + switch options.WriteMode { + case "seq": + err = cmd.runWritesSequential(db, options, results) + case "rnd": + err = cmd.runWritesRandom(db, options, results) + case "seq-nest": + err = cmd.runWritesSequentialNested(db, options, results) + case "rnd-nest": + err = cmd.runWritesRandomNested(db, options, results) + default: + return fmt.Errorf("invalid write mode: %s", options.WriteMode) + } + + // Save time to write. + results.WriteDuration = time.Since(t) + + // Stop profiling for writes only. + if options.ProfileMode == "w" { + cmd.stopProfiling() + } + + return err +} + +func (cmd *BenchCommand) runWritesSequential(db *bolt.DB, options *BenchOptions, results *BenchResults) error { + var i = uint32(0) + return cmd.runWritesWithSource(db, options, results, func() uint32 { i++; return i }) +} + +func (cmd *BenchCommand) runWritesRandom(db *bolt.DB, options *BenchOptions, results *BenchResults) error { + r := rand.New(rand.NewSource(time.Now().UnixNano())) + return cmd.runWritesWithSource(db, options, results, func() uint32 { return r.Uint32() }) +} + +func (cmd *BenchCommand) runWritesSequentialNested(db *bolt.DB, options *BenchOptions, results *BenchResults) error { + var i = uint32(0) + return cmd.runWritesWithSource(db, options, results, func() uint32 { i++; return i }) +} + +func (cmd *BenchCommand) runWritesRandomNested(db *bolt.DB, options *BenchOptions, results *BenchResults) error { + r := rand.New(rand.NewSource(time.Now().UnixNano())) + return cmd.runWritesWithSource(db, options, results, func() uint32 { return r.Uint32() }) +} + +func (cmd *BenchCommand) runWritesWithSource(db *bolt.DB, options *BenchOptions, results *BenchResults, keySource func() uint32) error { + results.WriteOps = options.Iterations + + for i := 0; i < options.Iterations; i += options.BatchSize { + if err := db.Update(func(tx *bolt.Tx) error { + b, _ := tx.CreateBucketIfNotExists(benchBucketName) + b.FillPercent = options.FillPercent + + for j := 0; j < options.BatchSize; j++ { + key := make([]byte, options.KeySize) + value := make([]byte, options.ValueSize) + + // Write key as uint32. + binary.BigEndian.PutUint32(key, keySource()) + + // Insert key/value. + if err := b.Put(key, value); err != nil { + return err + } + } + + return nil + }); err != nil { + return err + } + } + return nil +} + +func (cmd *BenchCommand) runWritesNestedWithSource(db *bolt.DB, options *BenchOptions, results *BenchResults, keySource func() uint32) error { + results.WriteOps = options.Iterations + + for i := 0; i < options.Iterations; i += options.BatchSize { + if err := db.Update(func(tx *bolt.Tx) error { + top, err := tx.CreateBucketIfNotExists(benchBucketName) + if err != nil { + return err + } + top.FillPercent = options.FillPercent + + // Create bucket key. + name := make([]byte, options.KeySize) + binary.BigEndian.PutUint32(name, keySource()) + + // Create bucket. + b, err := top.CreateBucketIfNotExists(name) + if err != nil { + return err + } + b.FillPercent = options.FillPercent + + for j := 0; j < options.BatchSize; j++ { + var key = make([]byte, options.KeySize) + var value = make([]byte, options.ValueSize) + + // Generate key as uint32. + binary.BigEndian.PutUint32(key, keySource()) + + // Insert value into subbucket. + if err := b.Put(key, value); err != nil { + return err + } + } + + return nil + }); err != nil { + return err + } + } + return nil +} + +// Reads from the database. +func (cmd *BenchCommand) runReads(db *bolt.DB, options *BenchOptions, results *BenchResults) error { + // Start profiling for reads. + if options.ProfileMode == "r" { + cmd.startProfiling(options) + } + + t := time.Now() + + var err error + switch options.ReadMode { + case "seq": + switch options.WriteMode { + case "seq-nest", "rnd-nest": + err = cmd.runReadsSequentialNested(db, options, results) + default: + err = cmd.runReadsSequential(db, options, results) + } + default: + return fmt.Errorf("invalid read mode: %s", options.ReadMode) + } + + // Save read time. + results.ReadDuration = time.Since(t) + + // Stop profiling for reads. + if options.ProfileMode == "rw" || options.ProfileMode == "r" { + cmd.stopProfiling() + } + + return err +} + +func (cmd *BenchCommand) runReadsSequential(db *bolt.DB, options *BenchOptions, results *BenchResults) error { + return db.View(func(tx *bolt.Tx) error { + t := time.Now() + + for { + var count int + + c := tx.Bucket(benchBucketName).Cursor() + for k, v := c.First(); k != nil; k, v = c.Next() { + if v == nil { + return errors.New("invalid value") + } + count++ + } + + if options.WriteMode == "seq" && count != options.Iterations { + return fmt.Errorf("read seq: iter mismatch: expected %d, got %d", options.Iterations, count) + } + + results.ReadOps += count + + // Make sure we do this for at least a second. + if time.Since(t) >= time.Second { + break + } + } + + return nil + }) +} + +func (cmd *BenchCommand) runReadsSequentialNested(db *bolt.DB, options *BenchOptions, results *BenchResults) error { + return db.View(func(tx *bolt.Tx) error { + t := time.Now() + + for { + var count int + var top = tx.Bucket(benchBucketName) + if err := top.ForEach(func(name, _ []byte) error { + c := top.Bucket(name).Cursor() + for k, v := c.First(); k != nil; k, v = c.Next() { + if v == nil { + return ErrInvalidValue + } + count++ + } + return nil + }); err != nil { + return err + } + + if options.WriteMode == "seq-nest" && count != options.Iterations { + return fmt.Errorf("read seq-nest: iter mismatch: expected %d, got %d", options.Iterations, count) + } + + results.ReadOps += count + + // Make sure we do this for at least a second. + if time.Since(t) >= time.Second { + break + } + } + + return nil + }) +} + +// File handlers for the various profiles. +var cpuprofile, memprofile, blockprofile *os.File + +// Starts all profiles set on the options. +func (cmd *BenchCommand) startProfiling(options *BenchOptions) { + var err error + + // Start CPU profiling. + if options.CPUProfile != "" { + cpuprofile, err = os.Create(options.CPUProfile) + if err != nil { + fmt.Fprintf(cmd.Stderr, "bench: could not create cpu profile %q: %v\n", options.CPUProfile, err) + os.Exit(1) + } + pprof.StartCPUProfile(cpuprofile) + } + + // Start memory profiling. + if options.MemProfile != "" { + memprofile, err = os.Create(options.MemProfile) + if err != nil { + fmt.Fprintf(cmd.Stderr, "bench: could not create memory profile %q: %v\n", options.MemProfile, err) + os.Exit(1) + } + runtime.MemProfileRate = 4096 + } + + // Start fatal profiling. + if options.BlockProfile != "" { + blockprofile, err = os.Create(options.BlockProfile) + if err != nil { + fmt.Fprintf(cmd.Stderr, "bench: could not create block profile %q: %v\n", options.BlockProfile, err) + os.Exit(1) + } + runtime.SetBlockProfileRate(1) + } +} + +// Stops all profiles. +func (cmd *BenchCommand) stopProfiling() { + if cpuprofile != nil { + pprof.StopCPUProfile() + cpuprofile.Close() + cpuprofile = nil + } + + if memprofile != nil { + pprof.Lookup("heap").WriteTo(memprofile, 0) + memprofile.Close() + memprofile = nil + } + + if blockprofile != nil { + pprof.Lookup("block").WriteTo(blockprofile, 0) + blockprofile.Close() + blockprofile = nil + runtime.SetBlockProfileRate(0) + } +} + +// BenchOptions represents the set of options that can be passed to "bolt bench". +type BenchOptions struct { + ProfileMode string + WriteMode string + ReadMode string + Iterations int + BatchSize int + KeySize int + ValueSize int + CPUProfile string + MemProfile string + BlockProfile string + StatsInterval time.Duration + FillPercent float64 + NoSync bool + Work bool + Path string +} + +// BenchResults represents the performance results of the benchmark. +type BenchResults struct { + WriteOps int + WriteDuration time.Duration + ReadOps int + ReadDuration time.Duration +} + +// Returns the duration for a single write operation. +func (r *BenchResults) WriteOpDuration() time.Duration { + if r.WriteOps == 0 { + return 0 + } + return r.WriteDuration / time.Duration(r.WriteOps) +} + +// Returns average number of write operations that can be performed per second. +func (r *BenchResults) WriteOpsPerSecond() int { + var op = r.WriteOpDuration() + if op == 0 { + return 0 + } + return int(time.Second) / int(op) +} + +// Returns the duration for a single read operation. +func (r *BenchResults) ReadOpDuration() time.Duration { + if r.ReadOps == 0 { + return 0 + } + return r.ReadDuration / time.Duration(r.ReadOps) +} + +// Returns average number of read operations that can be performed per second. +func (r *BenchResults) ReadOpsPerSecond() int { + var op = r.ReadOpDuration() + if op == 0 { + return 0 + } + return int(time.Second) / int(op) +} + +type PageError struct { + ID int + Err error +} + +func (e *PageError) Error() string { + return fmt.Sprintf("page error: id=%d, err=%s", e.ID, e.Err) +} + +// isPrintable returns true if the string is valid unicode and contains only printable runes. +func isPrintable(s string) bool { + if !utf8.ValidString(s) { + return false + } + for _, ch := range s { + if !unicode.IsPrint(ch) { + return false + } + } + return true +} + +// ReadPage reads page info & full page data from a path. +// This is not transactionally safe. +func ReadPage(path string, pageID int) (*page, []byte, error) { + // Find page size. + pageSize, err := ReadPageSize(path) + if err != nil { + return nil, nil, fmt.Errorf("read page size: %s", err) + } + + // Open database file. + f, err := os.Open(path) + if err != nil { + return nil, nil, err + } + defer f.Close() + + // Read one block into buffer. + buf := make([]byte, pageSize) + if n, err := f.ReadAt(buf, int64(pageID*pageSize)); err != nil { + return nil, nil, err + } else if n != len(buf) { + return nil, nil, io.ErrUnexpectedEOF + } + + // Determine total number of blocks. + p := (*page)(unsafe.Pointer(&buf[0])) + overflowN := p.overflow + + // Re-read entire page (with overflow) into buffer. + buf = make([]byte, (int(overflowN)+1)*pageSize) + if n, err := f.ReadAt(buf, int64(pageID*pageSize)); err != nil { + return nil, nil, err + } else if n != len(buf) { + return nil, nil, io.ErrUnexpectedEOF + } + p = (*page)(unsafe.Pointer(&buf[0])) + + return p, buf, nil +} + +// ReadPageSize reads page size a path. +// This is not transactionally safe. +func ReadPageSize(path string) (int, error) { + // Open database file. + f, err := os.Open(path) + if err != nil { + return 0, err + } + defer f.Close() + + // Read 4KB chunk. + buf := make([]byte, 4096) + if _, err := io.ReadFull(f, buf); err != nil { + return 0, err + } + + // Read page size from metadata. + m := (*meta)(unsafe.Pointer(&buf[PageHeaderSize])) + return int(m.pageSize), nil +} + +// atois parses a slice of strings into integers. +func atois(strs []string) ([]int, error) { + var a []int + for _, str := range strs { + i, err := strconv.Atoi(str) + if err != nil { + return nil, err + } + a = append(a, i) + } + return a, nil +} + +// DO NOT EDIT. Copied from the "bolt" package. +const maxAllocSize = 0xFFFFFFF + +// DO NOT EDIT. Copied from the "bolt" package. +const ( + branchPageFlag = 0x01 + leafPageFlag = 0x02 + metaPageFlag = 0x04 + freelistPageFlag = 0x10 +) + +// DO NOT EDIT. Copied from the "bolt" package. +const bucketLeafFlag = 0x01 + +// DO NOT EDIT. Copied from the "bolt" package. +type pgid uint64 + +// DO NOT EDIT. Copied from the "bolt" package. +type txid uint64 + +// DO NOT EDIT. Copied from the "bolt" package. +type meta struct { + magic uint32 + version uint32 + pageSize uint32 + flags uint32 + root bucket + freelist pgid + pgid pgid + txid txid + checksum uint64 +} + +// DO NOT EDIT. Copied from the "bolt" package. +type bucket struct { + root pgid + sequence uint64 +} + +// DO NOT EDIT. Copied from the "bolt" package. +type page struct { + id pgid + flags uint16 + count uint16 + overflow uint32 + ptr uintptr +} + +// DO NOT EDIT. Copied from the "bolt" package. +func (p *page) Type() string { + if (p.flags & branchPageFlag) != 0 { + return "branch" + } else if (p.flags & leafPageFlag) != 0 { + return "leaf" + } else if (p.flags & metaPageFlag) != 0 { + return "meta" + } else if (p.flags & freelistPageFlag) != 0 { + return "freelist" + } + return fmt.Sprintf("unknown<%02x>", p.flags) +} + +// DO NOT EDIT. Copied from the "bolt" package. +func (p *page) leafPageElement(index uint16) *leafPageElement { + n := &((*[0x7FFFFFF]leafPageElement)(unsafe.Pointer(&p.ptr)))[index] + return n +} + +// DO NOT EDIT. Copied from the "bolt" package. +func (p *page) branchPageElement(index uint16) *branchPageElement { + return &((*[0x7FFFFFF]branchPageElement)(unsafe.Pointer(&p.ptr)))[index] +} + +// DO NOT EDIT. Copied from the "bolt" package. +type branchPageElement struct { + pos uint32 + ksize uint32 + pgid pgid +} + +// DO NOT EDIT. Copied from the "bolt" package. +func (n *branchPageElement) key() []byte { + buf := (*[maxAllocSize]byte)(unsafe.Pointer(n)) + return buf[n.pos : n.pos+n.ksize] +} + +// DO NOT EDIT. Copied from the "bolt" package. +type leafPageElement struct { + flags uint32 + pos uint32 + ksize uint32 + vsize uint32 +} + +// DO NOT EDIT. Copied from the "bolt" package. +func (n *leafPageElement) key() []byte { + buf := (*[maxAllocSize]byte)(unsafe.Pointer(n)) + return buf[n.pos : n.pos+n.ksize] +} + +// DO NOT EDIT. Copied from the "bolt" package. +func (n *leafPageElement) value() []byte { + buf := (*[maxAllocSize]byte)(unsafe.Pointer(n)) + return buf[n.pos+n.ksize : n.pos+n.ksize+n.vsize] +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/cmd/bolt/main_test.go b/Godeps/_workspace/src/github.com/boltdb/bolt/cmd/bolt/main_test.go new file mode 100644 index 0000000000000..b9e8c671f3632 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/cmd/bolt/main_test.go @@ -0,0 +1,145 @@ +package main_test + +import ( + "bytes" + "io/ioutil" + "os" + "strconv" + "testing" + + "github.com/boltdb/bolt" + "github.com/boltdb/bolt/cmd/bolt" +) + +// Ensure the "info" command can print information about a database. +func TestInfoCommand_Run(t *testing.T) { + db := MustOpen(0666, nil) + db.DB.Close() + defer db.Close() + + // Run the info command. + m := NewMain() + if err := m.Run("info", db.Path); err != nil { + t.Fatal(err) + } +} + +// Ensure the "stats" command can execute correctly. +func TestStatsCommand_Run(t *testing.T) { + // Ignore + if os.Getpagesize() != 4096 { + t.Skip("system does not use 4KB page size") + } + + db := MustOpen(0666, nil) + defer db.Close() + + if err := db.Update(func(tx *bolt.Tx) error { + // Create "foo" bucket. + b, err := tx.CreateBucket([]byte("foo")) + if err != nil { + return err + } + for i := 0; i < 10; i++ { + if err := b.Put([]byte(strconv.Itoa(i)), []byte(strconv.Itoa(i))); err != nil { + return err + } + } + + // Create "bar" bucket. + b, err = tx.CreateBucket([]byte("bar")) + if err != nil { + return err + } + for i := 0; i < 100; i++ { + if err := b.Put([]byte(strconv.Itoa(i)), []byte(strconv.Itoa(i))); err != nil { + return err + } + } + + // Create "baz" bucket. + b, err = tx.CreateBucket([]byte("baz")) + if err != nil { + return err + } + if err := b.Put([]byte("key"), []byte("value")); err != nil { + return err + } + + return nil + }); err != nil { + t.Fatal(err) + } + db.DB.Close() + + // Generate expected result. + exp := "Aggregate statistics for 3 buckets\n\n" + + "Page count statistics\n" + + "\tNumber of logical branch pages: 0\n" + + "\tNumber of physical branch overflow pages: 0\n" + + "\tNumber of logical leaf pages: 1\n" + + "\tNumber of physical leaf overflow pages: 0\n" + + "Tree statistics\n" + + "\tNumber of keys/value pairs: 111\n" + + "\tNumber of levels in B+tree: 1\n" + + "Page size utilization\n" + + "\tBytes allocated for physical branch pages: 0\n" + + "\tBytes actually used for branch data: 0 (0%)\n" + + "\tBytes allocated for physical leaf pages: 4096\n" + + "\tBytes actually used for leaf data: 1996 (48%)\n" + + "Bucket statistics\n" + + "\tTotal number of buckets: 3\n" + + "\tTotal number on inlined buckets: 2 (66%)\n" + + "\tBytes used for inlined buckets: 236 (11%)\n" + + // Run the command. + m := NewMain() + if err := m.Run("stats", db.Path); err != nil { + t.Fatal(err) + } else if m.Stdout.String() != exp { + t.Fatalf("unexpected stdout:\n\n%s", m.Stdout.String()) + } +} + +// Main represents a test wrapper for main.Main that records output. +type Main struct { + *main.Main + Stdin bytes.Buffer + Stdout bytes.Buffer + Stderr bytes.Buffer +} + +// NewMain returns a new instance of Main. +func NewMain() *Main { + m := &Main{Main: main.NewMain()} + m.Main.Stdin = &m.Stdin + m.Main.Stdout = &m.Stdout + m.Main.Stderr = &m.Stderr + return m +} + +// MustOpen creates a Bolt database in a temporary location. +func MustOpen(mode os.FileMode, options *bolt.Options) *DB { + // Create temporary path. + f, _ := ioutil.TempFile("", "bolt-") + f.Close() + os.Remove(f.Name()) + + db, err := bolt.Open(f.Name(), mode, options) + if err != nil { + panic(err.Error()) + } + return &DB{DB: db, Path: f.Name()} +} + +// DB is a test wrapper for bolt.DB. +type DB struct { + *bolt.DB + Path string +} + +// Close closes and removes the database. +func (db *DB) Close() error { + defer os.Remove(db.Path) + return db.DB.Close() +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/cursor.go b/Godeps/_workspace/src/github.com/boltdb/bolt/cursor.go new file mode 100644 index 0000000000000..006c54889e8ab --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/cursor.go @@ -0,0 +1,384 @@ +package bolt + +import ( + "bytes" + "fmt" + "sort" +) + +// Cursor represents an iterator that can traverse over all key/value pairs in a bucket in sorted order. +// Cursors see nested buckets with value == nil. +// Cursors can be obtained from a transaction and are valid as long as the transaction is open. +// +// Keys and values returned from the cursor are only valid for the life of the transaction. +// +// Changing data while traversing with a cursor may cause it to be invalidated +// and return unexpected keys and/or values. You must reposition your cursor +// after mutating data. +type Cursor struct { + bucket *Bucket + stack []elemRef +} + +// Bucket returns the bucket that this cursor was created from. +func (c *Cursor) Bucket() *Bucket { + return c.bucket +} + +// First moves the cursor to the first item in the bucket and returns its key and value. +// If the bucket is empty then a nil key and value are returned. +// The returned key and value are only valid for the life of the transaction. +func (c *Cursor) First() (key []byte, value []byte) { + _assert(c.bucket.tx.db != nil, "tx closed") + c.stack = c.stack[:0] + p, n := c.bucket.pageNode(c.bucket.root) + c.stack = append(c.stack, elemRef{page: p, node: n, index: 0}) + c.first() + k, v, flags := c.keyValue() + if (flags & uint32(bucketLeafFlag)) != 0 { + return k, nil + } + return k, v + +} + +// Last moves the cursor to the last item in the bucket and returns its key and value. +// If the bucket is empty then a nil key and value are returned. +// The returned key and value are only valid for the life of the transaction. +func (c *Cursor) Last() (key []byte, value []byte) { + _assert(c.bucket.tx.db != nil, "tx closed") + c.stack = c.stack[:0] + p, n := c.bucket.pageNode(c.bucket.root) + ref := elemRef{page: p, node: n} + ref.index = ref.count() - 1 + c.stack = append(c.stack, ref) + c.last() + k, v, flags := c.keyValue() + if (flags & uint32(bucketLeafFlag)) != 0 { + return k, nil + } + return k, v +} + +// Next moves the cursor to the next item in the bucket and returns its key and value. +// If the cursor is at the end of the bucket then a nil key and value are returned. +// The returned key and value are only valid for the life of the transaction. +func (c *Cursor) Next() (key []byte, value []byte) { + _assert(c.bucket.tx.db != nil, "tx closed") + k, v, flags := c.next() + if (flags & uint32(bucketLeafFlag)) != 0 { + return k, nil + } + return k, v +} + +// Prev moves the cursor to the previous item in the bucket and returns its key and value. +// If the cursor is at the beginning of the bucket then a nil key and value are returned. +// The returned key and value are only valid for the life of the transaction. +func (c *Cursor) Prev() (key []byte, value []byte) { + _assert(c.bucket.tx.db != nil, "tx closed") + + // Attempt to move back one element until we're successful. + // Move up the stack as we hit the beginning of each page in our stack. + for i := len(c.stack) - 1; i >= 0; i-- { + elem := &c.stack[i] + if elem.index > 0 { + elem.index-- + break + } + c.stack = c.stack[:i] + } + + // If we've hit the end then return nil. + if len(c.stack) == 0 { + return nil, nil + } + + // Move down the stack to find the last element of the last leaf under this branch. + c.last() + k, v, flags := c.keyValue() + if (flags & uint32(bucketLeafFlag)) != 0 { + return k, nil + } + return k, v +} + +// Seek moves the cursor to a given key and returns it. +// If the key does not exist then the next key is used. If no keys +// follow, a nil key is returned. +// The returned key and value are only valid for the life of the transaction. +func (c *Cursor) Seek(seek []byte) (key []byte, value []byte) { + k, v, flags := c.seek(seek) + + // If we ended up after the last element of a page then move to the next one. + if ref := &c.stack[len(c.stack)-1]; ref.index >= ref.count() { + k, v, flags = c.next() + } + + if k == nil { + return nil, nil + } else if (flags & uint32(bucketLeafFlag)) != 0 { + return k, nil + } + return k, v +} + +// Delete removes the current key/value under the cursor from the bucket. +// Delete fails if current key/value is a bucket or if the transaction is not writable. +func (c *Cursor) Delete() error { + if c.bucket.tx.db == nil { + return ErrTxClosed + } else if !c.bucket.Writable() { + return ErrTxNotWritable + } + + key, _, flags := c.keyValue() + // Return an error if current value is a bucket. + if (flags & bucketLeafFlag) != 0 { + return ErrIncompatibleValue + } + c.node().del(key) + + return nil +} + +// seek moves the cursor to a given key and returns it. +// If the key does not exist then the next key is used. +func (c *Cursor) seek(seek []byte) (key []byte, value []byte, flags uint32) { + _assert(c.bucket.tx.db != nil, "tx closed") + + // Start from root page/node and traverse to correct page. + c.stack = c.stack[:0] + c.search(seek, c.bucket.root) + ref := &c.stack[len(c.stack)-1] + + // If the cursor is pointing to the end of page/node then return nil. + if ref.index >= ref.count() { + return nil, nil, 0 + } + + // If this is a bucket then return a nil value. + return c.keyValue() +} + +// first moves the cursor to the first leaf element under the last page in the stack. +func (c *Cursor) first() { + for { + // Exit when we hit a leaf page. + var ref = &c.stack[len(c.stack)-1] + if ref.isLeaf() { + break + } + + // Keep adding pages pointing to the first element to the stack. + var pgid pgid + if ref.node != nil { + pgid = ref.node.inodes[ref.index].pgid + } else { + pgid = ref.page.branchPageElement(uint16(ref.index)).pgid + } + p, n := c.bucket.pageNode(pgid) + c.stack = append(c.stack, elemRef{page: p, node: n, index: 0}) + } +} + +// last moves the cursor to the last leaf element under the last page in the stack. +func (c *Cursor) last() { + for { + // Exit when we hit a leaf page. + ref := &c.stack[len(c.stack)-1] + if ref.isLeaf() { + break + } + + // Keep adding pages pointing to the last element in the stack. + var pgid pgid + if ref.node != nil { + pgid = ref.node.inodes[ref.index].pgid + } else { + pgid = ref.page.branchPageElement(uint16(ref.index)).pgid + } + p, n := c.bucket.pageNode(pgid) + + var nextRef = elemRef{page: p, node: n} + nextRef.index = nextRef.count() - 1 + c.stack = append(c.stack, nextRef) + } +} + +// next moves to the next leaf element and returns the key and value. +// If the cursor is at the last leaf element then it stays there and returns nil. +func (c *Cursor) next() (key []byte, value []byte, flags uint32) { + // Attempt to move over one element until we're successful. + // Move up the stack as we hit the end of each page in our stack. + var i int + for i = len(c.stack) - 1; i >= 0; i-- { + elem := &c.stack[i] + if elem.index < elem.count()-1 { + elem.index++ + break + } + } + + // If we've hit the root page then stop and return. This will leave the + // cursor on the last element of the last page. + if i == -1 { + return nil, nil, 0 + } + + // Otherwise start from where we left off in the stack and find the + // first element of the first leaf page. + c.stack = c.stack[:i+1] + c.first() + return c.keyValue() +} + +// search recursively performs a binary search against a given page/node until it finds a given key. +func (c *Cursor) search(key []byte, pgid pgid) { + p, n := c.bucket.pageNode(pgid) + if p != nil && (p.flags&(branchPageFlag|leafPageFlag)) == 0 { + panic(fmt.Sprintf("invalid page type: %d: %x", p.id, p.flags)) + } + e := elemRef{page: p, node: n} + c.stack = append(c.stack, e) + + // If we're on a leaf page/node then find the specific node. + if e.isLeaf() { + c.nsearch(key) + return + } + + if n != nil { + c.searchNode(key, n) + return + } + c.searchPage(key, p) +} + +func (c *Cursor) searchNode(key []byte, n *node) { + var exact bool + index := sort.Search(len(n.inodes), func(i int) bool { + // TODO(benbjohnson): Optimize this range search. It's a bit hacky right now. + // sort.Search() finds the lowest index where f() != -1 but we need the highest index. + ret := bytes.Compare(n.inodes[i].key, key) + if ret == 0 { + exact = true + } + return ret != -1 + }) + if !exact && index > 0 { + index-- + } + c.stack[len(c.stack)-1].index = index + + // Recursively search to the next page. + c.search(key, n.inodes[index].pgid) +} + +func (c *Cursor) searchPage(key []byte, p *page) { + // Binary search for the correct range. + inodes := p.branchPageElements() + + var exact bool + index := sort.Search(int(p.count), func(i int) bool { + // TODO(benbjohnson): Optimize this range search. It's a bit hacky right now. + // sort.Search() finds the lowest index where f() != -1 but we need the highest index. + ret := bytes.Compare(inodes[i].key(), key) + if ret == 0 { + exact = true + } + return ret != -1 + }) + if !exact && index > 0 { + index-- + } + c.stack[len(c.stack)-1].index = index + + // Recursively search to the next page. + c.search(key, inodes[index].pgid) +} + +// nsearch searches the leaf node on the top of the stack for a key. +func (c *Cursor) nsearch(key []byte) { + e := &c.stack[len(c.stack)-1] + p, n := e.page, e.node + + // If we have a node then search its inodes. + if n != nil { + index := sort.Search(len(n.inodes), func(i int) bool { + return bytes.Compare(n.inodes[i].key, key) != -1 + }) + e.index = index + return + } + + // If we have a page then search its leaf elements. + inodes := p.leafPageElements() + index := sort.Search(int(p.count), func(i int) bool { + return bytes.Compare(inodes[i].key(), key) != -1 + }) + e.index = index +} + +// keyValue returns the key and value of the current leaf element. +func (c *Cursor) keyValue() ([]byte, []byte, uint32) { + ref := &c.stack[len(c.stack)-1] + if ref.count() == 0 || ref.index >= ref.count() { + return nil, nil, 0 + } + + // Retrieve value from node. + if ref.node != nil { + inode := &ref.node.inodes[ref.index] + return inode.key, inode.value, inode.flags + } + + // Or retrieve value from page. + elem := ref.page.leafPageElement(uint16(ref.index)) + return elem.key(), elem.value(), elem.flags +} + +// node returns the node that the cursor is currently positioned on. +func (c *Cursor) node() *node { + _assert(len(c.stack) > 0, "accessing a node with a zero-length cursor stack") + + // If the top of the stack is a leaf node then just return it. + if ref := &c.stack[len(c.stack)-1]; ref.node != nil && ref.isLeaf() { + return ref.node + } + + // Start from root and traverse down the hierarchy. + var n = c.stack[0].node + if n == nil { + n = c.bucket.node(c.stack[0].page.id, nil) + } + for _, ref := range c.stack[:len(c.stack)-1] { + _assert(!n.isLeaf, "expected branch node") + n = n.childAt(int(ref.index)) + } + _assert(n.isLeaf, "expected leaf node") + return n +} + +// elemRef represents a reference to an element on a given page/node. +type elemRef struct { + page *page + node *node + index int +} + +// isLeaf returns whether the ref is pointing at a leaf page/node. +func (r *elemRef) isLeaf() bool { + if r.node != nil { + return r.node.isLeaf + } + return (r.page.flags & leafPageFlag) != 0 +} + +// count returns the number of inodes or page elements. +func (r *elemRef) count() int { + if r.node != nil { + return len(r.node.inodes) + } + return int(r.page.count) +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/cursor_test.go b/Godeps/_workspace/src/github.com/boltdb/bolt/cursor_test.go new file mode 100644 index 0000000000000..b12e1f9153efb --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/cursor_test.go @@ -0,0 +1,511 @@ +package bolt_test + +import ( + "bytes" + "encoding/binary" + "fmt" + "os" + "sort" + "testing" + "testing/quick" + + "github.com/boltdb/bolt" +) + +// Ensure that a cursor can return a reference to the bucket that created it. +func TestCursor_Bucket(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + b, _ := tx.CreateBucket([]byte("widgets")) + c := b.Cursor() + equals(t, b, c.Bucket()) + return nil + }) +} + +// Ensure that a Tx cursor can seek to the appropriate keys. +func TestCursor_Seek(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + b, err := tx.CreateBucket([]byte("widgets")) + ok(t, err) + ok(t, b.Put([]byte("foo"), []byte("0001"))) + ok(t, b.Put([]byte("bar"), []byte("0002"))) + ok(t, b.Put([]byte("baz"), []byte("0003"))) + _, err = b.CreateBucket([]byte("bkt")) + ok(t, err) + return nil + }) + db.View(func(tx *bolt.Tx) error { + c := tx.Bucket([]byte("widgets")).Cursor() + + // Exact match should go to the key. + k, v := c.Seek([]byte("bar")) + equals(t, []byte("bar"), k) + equals(t, []byte("0002"), v) + + // Inexact match should go to the next key. + k, v = c.Seek([]byte("bas")) + equals(t, []byte("baz"), k) + equals(t, []byte("0003"), v) + + // Low key should go to the first key. + k, v = c.Seek([]byte("")) + equals(t, []byte("bar"), k) + equals(t, []byte("0002"), v) + + // High key should return no key. + k, v = c.Seek([]byte("zzz")) + assert(t, k == nil, "") + assert(t, v == nil, "") + + // Buckets should return their key but no value. + k, v = c.Seek([]byte("bkt")) + equals(t, []byte("bkt"), k) + assert(t, v == nil, "") + + return nil + }) +} + +func TestCursor_Delete(t *testing.T) { + db := NewTestDB() + defer db.Close() + + var count = 1000 + + // Insert every other key between 0 and $count. + db.Update(func(tx *bolt.Tx) error { + b, _ := tx.CreateBucket([]byte("widgets")) + for i := 0; i < count; i += 1 { + k := make([]byte, 8) + binary.BigEndian.PutUint64(k, uint64(i)) + b.Put(k, make([]byte, 100)) + } + b.CreateBucket([]byte("sub")) + return nil + }) + + db.Update(func(tx *bolt.Tx) error { + c := tx.Bucket([]byte("widgets")).Cursor() + bound := make([]byte, 8) + binary.BigEndian.PutUint64(bound, uint64(count/2)) + for key, _ := c.First(); bytes.Compare(key, bound) < 0; key, _ = c.Next() { + if err := c.Delete(); err != nil { + return err + } + } + c.Seek([]byte("sub")) + err := c.Delete() + equals(t, err, bolt.ErrIncompatibleValue) + return nil + }) + + db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("widgets")) + equals(t, b.Stats().KeyN, count/2+1) + return nil + }) +} + +// Ensure that a Tx cursor can seek to the appropriate keys when there are a +// large number of keys. This test also checks that seek will always move +// forward to the next key. +// +// Related: https://github.com/boltdb/bolt/pull/187 +func TestCursor_Seek_Large(t *testing.T) { + db := NewTestDB() + defer db.Close() + + var count = 10000 + + // Insert every other key between 0 and $count. + db.Update(func(tx *bolt.Tx) error { + b, _ := tx.CreateBucket([]byte("widgets")) + for i := 0; i < count; i += 100 { + for j := i; j < i+100; j += 2 { + k := make([]byte, 8) + binary.BigEndian.PutUint64(k, uint64(j)) + b.Put(k, make([]byte, 100)) + } + } + return nil + }) + + db.View(func(tx *bolt.Tx) error { + c := tx.Bucket([]byte("widgets")).Cursor() + for i := 0; i < count; i++ { + seek := make([]byte, 8) + binary.BigEndian.PutUint64(seek, uint64(i)) + + k, _ := c.Seek(seek) + + // The last seek is beyond the end of the the range so + // it should return nil. + if i == count-1 { + assert(t, k == nil, "") + continue + } + + // Otherwise we should seek to the exact key or the next key. + num := binary.BigEndian.Uint64(k) + if i%2 == 0 { + equals(t, uint64(i), num) + } else { + equals(t, uint64(i+1), num) + } + } + + return nil + }) +} + +// Ensure that a cursor can iterate over an empty bucket without error. +func TestCursor_EmptyBucket(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte("widgets")) + return err + }) + db.View(func(tx *bolt.Tx) error { + c := tx.Bucket([]byte("widgets")).Cursor() + k, v := c.First() + assert(t, k == nil, "") + assert(t, v == nil, "") + return nil + }) +} + +// Ensure that a Tx cursor can reverse iterate over an empty bucket without error. +func TestCursor_EmptyBucketReverse(t *testing.T) { + db := NewTestDB() + defer db.Close() + + db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte("widgets")) + return err + }) + db.View(func(tx *bolt.Tx) error { + c := tx.Bucket([]byte("widgets")).Cursor() + k, v := c.Last() + assert(t, k == nil, "") + assert(t, v == nil, "") + return nil + }) +} + +// Ensure that a Tx cursor can iterate over a single root with a couple elements. +func TestCursor_Iterate_Leaf(t *testing.T) { + db := NewTestDB() + defer db.Close() + + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.Bucket([]byte("widgets")).Put([]byte("baz"), []byte{}) + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte{0}) + tx.Bucket([]byte("widgets")).Put([]byte("bar"), []byte{1}) + return nil + }) + tx, _ := db.Begin(false) + c := tx.Bucket([]byte("widgets")).Cursor() + + k, v := c.First() + equals(t, string(k), "bar") + equals(t, v, []byte{1}) + + k, v = c.Next() + equals(t, string(k), "baz") + equals(t, v, []byte{}) + + k, v = c.Next() + equals(t, string(k), "foo") + equals(t, v, []byte{0}) + + k, v = c.Next() + assert(t, k == nil, "") + assert(t, v == nil, "") + + k, v = c.Next() + assert(t, k == nil, "") + assert(t, v == nil, "") + + tx.Rollback() +} + +// Ensure that a Tx cursor can iterate in reverse over a single root with a couple elements. +func TestCursor_LeafRootReverse(t *testing.T) { + db := NewTestDB() + defer db.Close() + + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.Bucket([]byte("widgets")).Put([]byte("baz"), []byte{}) + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte{0}) + tx.Bucket([]byte("widgets")).Put([]byte("bar"), []byte{1}) + return nil + }) + tx, _ := db.Begin(false) + c := tx.Bucket([]byte("widgets")).Cursor() + + k, v := c.Last() + equals(t, string(k), "foo") + equals(t, v, []byte{0}) + + k, v = c.Prev() + equals(t, string(k), "baz") + equals(t, v, []byte{}) + + k, v = c.Prev() + equals(t, string(k), "bar") + equals(t, v, []byte{1}) + + k, v = c.Prev() + assert(t, k == nil, "") + assert(t, v == nil, "") + + k, v = c.Prev() + assert(t, k == nil, "") + assert(t, v == nil, "") + + tx.Rollback() +} + +// Ensure that a Tx cursor can restart from the beginning. +func TestCursor_Restart(t *testing.T) { + db := NewTestDB() + defer db.Close() + + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.Bucket([]byte("widgets")).Put([]byte("bar"), []byte{}) + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte{}) + return nil + }) + + tx, _ := db.Begin(false) + c := tx.Bucket([]byte("widgets")).Cursor() + + k, _ := c.First() + equals(t, string(k), "bar") + + k, _ = c.Next() + equals(t, string(k), "foo") + + k, _ = c.First() + equals(t, string(k), "bar") + + k, _ = c.Next() + equals(t, string(k), "foo") + + tx.Rollback() +} + +// Ensure that a Tx can iterate over all elements in a bucket. +func TestCursor_QuickCheck(t *testing.T) { + f := func(items testdata) bool { + db := NewTestDB() + defer db.Close() + + // Bulk insert all values. + tx, _ := db.Begin(true) + tx.CreateBucket([]byte("widgets")) + b := tx.Bucket([]byte("widgets")) + for _, item := range items { + ok(t, b.Put(item.Key, item.Value)) + } + ok(t, tx.Commit()) + + // Sort test data. + sort.Sort(items) + + // Iterate over all items and check consistency. + var index = 0 + tx, _ = db.Begin(false) + c := tx.Bucket([]byte("widgets")).Cursor() + for k, v := c.First(); k != nil && index < len(items); k, v = c.Next() { + equals(t, k, items[index].Key) + equals(t, v, items[index].Value) + index++ + } + equals(t, len(items), index) + tx.Rollback() + + return true + } + if err := quick.Check(f, qconfig()); err != nil { + t.Error(err) + } +} + +// Ensure that a transaction can iterate over all elements in a bucket in reverse. +func TestCursor_QuickCheck_Reverse(t *testing.T) { + f := func(items testdata) bool { + db := NewTestDB() + defer db.Close() + + // Bulk insert all values. + tx, _ := db.Begin(true) + tx.CreateBucket([]byte("widgets")) + b := tx.Bucket([]byte("widgets")) + for _, item := range items { + ok(t, b.Put(item.Key, item.Value)) + } + ok(t, tx.Commit()) + + // Sort test data. + sort.Sort(revtestdata(items)) + + // Iterate over all items and check consistency. + var index = 0 + tx, _ = db.Begin(false) + c := tx.Bucket([]byte("widgets")).Cursor() + for k, v := c.Last(); k != nil && index < len(items); k, v = c.Prev() { + equals(t, k, items[index].Key) + equals(t, v, items[index].Value) + index++ + } + equals(t, len(items), index) + tx.Rollback() + + return true + } + if err := quick.Check(f, qconfig()); err != nil { + t.Error(err) + } +} + +// Ensure that a Tx cursor can iterate over subbuckets. +func TestCursor_QuickCheck_BucketsOnly(t *testing.T) { + db := NewTestDB() + defer db.Close() + + db.Update(func(tx *bolt.Tx) error { + b, err := tx.CreateBucket([]byte("widgets")) + ok(t, err) + _, err = b.CreateBucket([]byte("foo")) + ok(t, err) + _, err = b.CreateBucket([]byte("bar")) + ok(t, err) + _, err = b.CreateBucket([]byte("baz")) + ok(t, err) + return nil + }) + db.View(func(tx *bolt.Tx) error { + var names []string + c := tx.Bucket([]byte("widgets")).Cursor() + for k, v := c.First(); k != nil; k, v = c.Next() { + names = append(names, string(k)) + assert(t, v == nil, "") + } + equals(t, names, []string{"bar", "baz", "foo"}) + return nil + }) +} + +// Ensure that a Tx cursor can reverse iterate over subbuckets. +func TestCursor_QuickCheck_BucketsOnly_Reverse(t *testing.T) { + db := NewTestDB() + defer db.Close() + + db.Update(func(tx *bolt.Tx) error { + b, err := tx.CreateBucket([]byte("widgets")) + ok(t, err) + _, err = b.CreateBucket([]byte("foo")) + ok(t, err) + _, err = b.CreateBucket([]byte("bar")) + ok(t, err) + _, err = b.CreateBucket([]byte("baz")) + ok(t, err) + return nil + }) + db.View(func(tx *bolt.Tx) error { + var names []string + c := tx.Bucket([]byte("widgets")).Cursor() + for k, v := c.Last(); k != nil; k, v = c.Prev() { + names = append(names, string(k)) + assert(t, v == nil, "") + } + equals(t, names, []string{"foo", "baz", "bar"}) + return nil + }) +} + +func ExampleCursor() { + // Open the database. + db, _ := bolt.Open(tempfile(), 0666, nil) + defer os.Remove(db.Path()) + defer db.Close() + + // Start a read-write transaction. + db.Update(func(tx *bolt.Tx) error { + // Create a new bucket. + tx.CreateBucket([]byte("animals")) + + // Insert data into a bucket. + b := tx.Bucket([]byte("animals")) + b.Put([]byte("dog"), []byte("fun")) + b.Put([]byte("cat"), []byte("lame")) + b.Put([]byte("liger"), []byte("awesome")) + + // Create a cursor for iteration. + c := b.Cursor() + + // Iterate over items in sorted key order. This starts from the + // first key/value pair and updates the k/v variables to the + // next key/value on each iteration. + // + // The loop finishes at the end of the cursor when a nil key is returned. + for k, v := c.First(); k != nil; k, v = c.Next() { + fmt.Printf("A %s is %s.\n", k, v) + } + + return nil + }) + + // Output: + // A cat is lame. + // A dog is fun. + // A liger is awesome. +} + +func ExampleCursor_reverse() { + // Open the database. + db, _ := bolt.Open(tempfile(), 0666, nil) + defer os.Remove(db.Path()) + defer db.Close() + + // Start a read-write transaction. + db.Update(func(tx *bolt.Tx) error { + // Create a new bucket. + tx.CreateBucket([]byte("animals")) + + // Insert data into a bucket. + b := tx.Bucket([]byte("animals")) + b.Put([]byte("dog"), []byte("fun")) + b.Put([]byte("cat"), []byte("lame")) + b.Put([]byte("liger"), []byte("awesome")) + + // Create a cursor for iteration. + c := b.Cursor() + + // Iterate over items in reverse sorted key order. This starts + // from the last key/value pair and updates the k/v variables to + // the previous key/value on each iteration. + // + // The loop finishes at the beginning of the cursor when a nil key + // is returned. + for k, v := c.Last(); k != nil; k, v = c.Prev() { + fmt.Printf("A %s is %s.\n", k, v) + } + + return nil + }) + + // Output: + // A liger is awesome. + // A dog is fun. + // A cat is lame. +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/db.go b/Godeps/_workspace/src/github.com/boltdb/bolt/db.go new file mode 100644 index 0000000000000..d39c4aa9ccef2 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/db.go @@ -0,0 +1,792 @@ +package bolt + +import ( + "fmt" + "hash/fnv" + "os" + "runtime" + "runtime/debug" + "strings" + "sync" + "time" + "unsafe" +) + +// The largest step that can be taken when remapping the mmap. +const maxMmapStep = 1 << 30 // 1GB + +// The data file format version. +const version = 2 + +// Represents a marker value to indicate that a file is a Bolt DB. +const magic uint32 = 0xED0CDAED + +// IgnoreNoSync specifies whether the NoSync field of a DB is ignored when +// syncing changes to a file. This is required as some operating systems, +// such as OpenBSD, do not have a unified buffer cache (UBC) and writes +// must be synchronzied using the msync(2) syscall. +const IgnoreNoSync = runtime.GOOS == "openbsd" + +// Default values if not set in a DB instance. +const ( + DefaultMaxBatchSize int = 1000 + DefaultMaxBatchDelay = 10 * time.Millisecond +) + +// DB represents a collection of buckets persisted to a file on disk. +// All data access is performed through transactions which can be obtained through the DB. +// All the functions on DB will return a ErrDatabaseNotOpen if accessed before Open() is called. +type DB struct { + // When enabled, the database will perform a Check() after every commit. + // A panic is issued if the database is in an inconsistent state. This + // flag has a large performance impact so it should only be used for + // debugging purposes. + StrictMode bool + + // Setting the NoSync flag will cause the database to skip fsync() + // calls after each commit. This can be useful when bulk loading data + // into a database and you can restart the bulk load in the event of + // a system failure or database corruption. Do not set this flag for + // normal use. + // + // If the package global IgnoreNoSync constant is true, this value is + // ignored. See the comment on that constant for more details. + // + // THIS IS UNSAFE. PLEASE USE WITH CAUTION. + NoSync bool + + // When true, skips the truncate call when growing the database. + // Setting this to true is only safe on non-ext3/ext4 systems. + // Skipping truncation avoids preallocation of hard drive space and + // bypasses a truncate() and fsync() syscall on remapping. + // + // https://github.com/boltdb/bolt/issues/284 + NoGrowSync bool + + // MaxBatchSize is the maximum size of a batch. Default value is + // copied from DefaultMaxBatchSize in Open. + // + // If <=0, disables batching. + // + // Do not change concurrently with calls to Batch. + MaxBatchSize int + + // MaxBatchDelay is the maximum delay before a batch starts. + // Default value is copied from DefaultMaxBatchDelay in Open. + // + // If <=0, effectively disables batching. + // + // Do not change concurrently with calls to Batch. + MaxBatchDelay time.Duration + + path string + file *os.File + dataref []byte // mmap'ed readonly, write throws SEGV + data *[maxMapSize]byte + datasz int + meta0 *meta + meta1 *meta + pageSize int + opened bool + rwtx *Tx + txs []*Tx + freelist *freelist + stats Stats + + batchMu sync.Mutex + batch *batch + + rwlock sync.Mutex // Allows only one writer at a time. + metalock sync.Mutex // Protects meta page access. + mmaplock sync.RWMutex // Protects mmap access during remapping. + statlock sync.RWMutex // Protects stats access. + + ops struct { + writeAt func(b []byte, off int64) (n int, err error) + } + + // Read only mode. + // When true, Update() and Begin(true) return ErrDatabaseReadOnly immediately. + readOnly bool +} + +// Path returns the path to currently open database file. +func (db *DB) Path() string { + return db.path +} + +// GoString returns the Go string representation of the database. +func (db *DB) GoString() string { + return fmt.Sprintf("bolt.DB{path:%q}", db.path) +} + +// String returns the string representation of the database. +func (db *DB) String() string { + return fmt.Sprintf("DB<%q>", db.path) +} + +// Open creates and opens a database at the given path. +// If the file does not exist then it will be created automatically. +// Passing in nil options will cause Bolt to open the database with the default options. +func Open(path string, mode os.FileMode, options *Options) (*DB, error) { + var db = &DB{opened: true} + + // Set default options if no options are provided. + if options == nil { + options = DefaultOptions + } + db.NoGrowSync = options.NoGrowSync + + // Set default values for later DB operations. + db.MaxBatchSize = DefaultMaxBatchSize + db.MaxBatchDelay = DefaultMaxBatchDelay + + flag := os.O_RDWR + if options.ReadOnly { + flag = os.O_RDONLY + db.readOnly = true + } + + // Open data file and separate sync handler for metadata writes. + db.path = path + var err error + if db.file, err = os.OpenFile(db.path, flag|os.O_CREATE, mode); err != nil { + _ = db.close() + return nil, err + } + + // Lock file so that other processes using Bolt in read-write mode cannot + // use the database at the same time. This would cause corruption since + // the two processes would write meta pages and free pages separately. + // The database file is locked exclusively (only one process can grab the lock) + // if !options.ReadOnly. + // The database file is locked using the shared lock (more than one process may + // hold a lock at the same time) otherwise (options.ReadOnly is set). + if err := flock(db.file, !db.readOnly, options.Timeout); err != nil { + _ = db.close() + return nil, err + } + + // Default values for test hooks + db.ops.writeAt = db.file.WriteAt + + // Initialize the database if it doesn't exist. + if info, err := db.file.Stat(); err != nil { + return nil, fmt.Errorf("stat error: %s", err) + } else if info.Size() == 0 { + // Initialize new files with meta pages. + if err := db.init(); err != nil { + return nil, err + } + } else { + // Read the first meta page to determine the page size. + var buf [0x1000]byte + if _, err := db.file.ReadAt(buf[:], 0); err == nil { + m := db.pageInBuffer(buf[:], 0).meta() + if err := m.validate(); err != nil { + return nil, fmt.Errorf("meta0 error: %s", err) + } + db.pageSize = int(m.pageSize) + } + } + + // Memory map the data file. + if err := db.mmap(0); err != nil { + _ = db.close() + return nil, err + } + + // Read in the freelist. + db.freelist = newFreelist() + db.freelist.read(db.page(db.meta().freelist)) + + // Mark the database as opened and return. + return db, nil +} + +// mmap opens the underlying memory-mapped file and initializes the meta references. +// minsz is the minimum size that the new mmap can be. +func (db *DB) mmap(minsz int) error { + db.mmaplock.Lock() + defer db.mmaplock.Unlock() + + info, err := db.file.Stat() + if err != nil { + return fmt.Errorf("mmap stat error: %s", err) + } else if int(info.Size()) < db.pageSize*2 { + return fmt.Errorf("file size too small") + } + + // Ensure the size is at least the minimum size. + var size = int(info.Size()) + if size < minsz { + size = minsz + } + size, err = db.mmapSize(size) + if err != nil { + return err + } + + // Dereference all mmap references before unmapping. + if db.rwtx != nil { + db.rwtx.root.dereference() + } + + // Unmap existing data before continuing. + if err := db.munmap(); err != nil { + return err + } + + // Memory-map the data file as a byte slice. + if err := mmap(db, size); err != nil { + return err + } + + // Save references to the meta pages. + db.meta0 = db.page(0).meta() + db.meta1 = db.page(1).meta() + + // Validate the meta pages. + if err := db.meta0.validate(); err != nil { + return fmt.Errorf("meta0 error: %s", err) + } + if err := db.meta1.validate(); err != nil { + return fmt.Errorf("meta1 error: %s", err) + } + + return nil +} + +// munmap unmaps the data file from memory. +func (db *DB) munmap() error { + if err := munmap(db); err != nil { + return fmt.Errorf("unmap error: " + err.Error()) + } + return nil +} + +// mmapSize determines the appropriate size for the mmap given the current size +// of the database. The minimum size is 1MB and doubles until it reaches 1GB. +// Returns an error if the new mmap size is greater than the max allowed. +func (db *DB) mmapSize(size int) (int, error) { + // Double the size from 32KB until 1GB. + for i := uint(15); i <= 30; i++ { + if size <= 1< maxMapSize { + return 0, fmt.Errorf("mmap too large") + } + + // If larger than 1GB then grow by 1GB at a time. + sz := int64(size) + if remainder := sz % int64(maxMmapStep); remainder > 0 { + sz += int64(maxMmapStep) - remainder + } + + // Ensure that the mmap size is a multiple of the page size. + // This should always be true since we're incrementing in MBs. + pageSize := int64(db.pageSize) + if (sz % pageSize) != 0 { + sz = ((sz / pageSize) + 1) * pageSize + } + + // If we've exceeded the max size then only grow up to the max size. + if sz > maxMapSize { + sz = maxMapSize + } + + return int(sz), nil +} + +// init creates a new database file and initializes its meta pages. +func (db *DB) init() error { + // Set the page size to the OS page size. + db.pageSize = os.Getpagesize() + + // Create two meta pages on a buffer. + buf := make([]byte, db.pageSize*4) + for i := 0; i < 2; i++ { + p := db.pageInBuffer(buf[:], pgid(i)) + p.id = pgid(i) + p.flags = metaPageFlag + + // Initialize the meta page. + m := p.meta() + m.magic = magic + m.version = version + m.pageSize = uint32(db.pageSize) + m.freelist = 2 + m.root = bucket{root: 3} + m.pgid = 4 + m.txid = txid(i) + } + + // Write an empty freelist at page 3. + p := db.pageInBuffer(buf[:], pgid(2)) + p.id = pgid(2) + p.flags = freelistPageFlag + p.count = 0 + + // Write an empty leaf page at page 4. + p = db.pageInBuffer(buf[:], pgid(3)) + p.id = pgid(3) + p.flags = leafPageFlag + p.count = 0 + + // Write the buffer to our data file. + if _, err := db.ops.writeAt(buf, 0); err != nil { + return err + } + if err := fdatasync(db); err != nil { + return err + } + + return nil +} + +// Close releases all database resources. +// All transactions must be closed before closing the database. +func (db *DB) Close() error { + db.rwlock.Lock() + defer db.rwlock.Unlock() + + db.metalock.Lock() + defer db.metalock.Unlock() + + db.mmaplock.RLock() + defer db.mmaplock.RUnlock() + + return db.close() +} + +func (db *DB) close() error { + db.opened = false + + db.freelist = nil + db.path = "" + + // Clear ops. + db.ops.writeAt = nil + + // Close the mmap. + if err := db.munmap(); err != nil { + return err + } + + // Close file handles. + if db.file != nil { + // No need to unlock read-only file. + if !db.readOnly { + // Unlock the file. + _ = funlock(db.file) + } + + // Close the file descriptor. + if err := db.file.Close(); err != nil { + return fmt.Errorf("db file close: %s", err) + } + db.file = nil + } + + return nil +} + +// Begin starts a new transaction. +// Multiple read-only transactions can be used concurrently but only one +// write transaction can be used at a time. Starting multiple write transactions +// will cause the calls to block and be serialized until the current write +// transaction finishes. +// +// Transactions should not be depedent on one another. Opening a read +// transaction and a write transaction in the same goroutine can cause the +// writer to deadlock because the database periodically needs to re-mmap itself +// as it grows and it cannot do that while a read transaction is open. +// +// IMPORTANT: You must close read-only transactions after you are finished or +// else the database will not reclaim old pages. +func (db *DB) Begin(writable bool) (*Tx, error) { + if writable { + return db.beginRWTx() + } + return db.beginTx() +} + +func (db *DB) beginTx() (*Tx, error) { + // Lock the meta pages while we initialize the transaction. We obtain + // the meta lock before the mmap lock because that's the order that the + // write transaction will obtain them. + db.metalock.Lock() + + // Obtain a read-only lock on the mmap. When the mmap is remapped it will + // obtain a write lock so all transactions must finish before it can be + // remapped. + db.mmaplock.RLock() + + // Exit if the database is not open yet. + if !db.opened { + db.mmaplock.RUnlock() + db.metalock.Unlock() + return nil, ErrDatabaseNotOpen + } + + // Create a transaction associated with the database. + t := &Tx{} + t.init(db) + + // Keep track of transaction until it closes. + db.txs = append(db.txs, t) + n := len(db.txs) + + // Unlock the meta pages. + db.metalock.Unlock() + + // Update the transaction stats. + db.statlock.Lock() + db.stats.TxN++ + db.stats.OpenTxN = n + db.statlock.Unlock() + + return t, nil +} + +func (db *DB) beginRWTx() (*Tx, error) { + // If the database was opened with Options.ReadOnly, return an error. + if db.readOnly { + return nil, ErrDatabaseReadOnly + } + + // Obtain writer lock. This is released by the transaction when it closes. + // This enforces only one writer transaction at a time. + db.rwlock.Lock() + + // Once we have the writer lock then we can lock the meta pages so that + // we can set up the transaction. + db.metalock.Lock() + defer db.metalock.Unlock() + + // Exit if the database is not open yet. + if !db.opened { + db.rwlock.Unlock() + return nil, ErrDatabaseNotOpen + } + + // Create a transaction associated with the database. + t := &Tx{writable: true} + t.init(db) + db.rwtx = t + + // Free any pages associated with closed read-only transactions. + var minid txid = 0xFFFFFFFFFFFFFFFF + for _, t := range db.txs { + if t.meta.txid < minid { + minid = t.meta.txid + } + } + if minid > 0 { + db.freelist.release(minid - 1) + } + + return t, nil +} + +// removeTx removes a transaction from the database. +func (db *DB) removeTx(tx *Tx) { + // Release the read lock on the mmap. + db.mmaplock.RUnlock() + + // Use the meta lock to restrict access to the DB object. + db.metalock.Lock() + + // Remove the transaction. + for i, t := range db.txs { + if t == tx { + db.txs = append(db.txs[:i], db.txs[i+1:]...) + break + } + } + n := len(db.txs) + + // Unlock the meta pages. + db.metalock.Unlock() + + // Merge statistics. + db.statlock.Lock() + db.stats.OpenTxN = n + db.stats.TxStats.add(&tx.stats) + db.statlock.Unlock() +} + +// Update executes a function within the context of a read-write managed transaction. +// If no error is returned from the function then the transaction is committed. +// If an error is returned then the entire transaction is rolled back. +// Any error that is returned from the function or returned from the commit is +// returned from the Update() method. +// +// Attempting to manually commit or rollback within the function will cause a panic. +func (db *DB) Update(fn func(*Tx) error) error { + t, err := db.Begin(true) + if err != nil { + return err + } + + // Make sure the transaction rolls back in the event of a panic. + defer func() { + if t.db != nil { + t.rollback() + } + }() + + // Mark as a managed tx so that the inner function cannot manually commit. + t.managed = true + + // If an error is returned from the function then rollback and return error. + err = fn(t) + t.managed = false + if err != nil { + _ = t.Rollback() + return err + } + + return t.Commit() +} + +// View executes a function within the context of a managed read-only transaction. +// Any error that is returned from the function is returned from the View() method. +// +// Attempting to manually rollback within the function will cause a panic. +func (db *DB) View(fn func(*Tx) error) error { + t, err := db.Begin(false) + if err != nil { + return err + } + + // Make sure the transaction rolls back in the event of a panic. + defer func() { + if t.db != nil { + t.rollback() + } + }() + + // Mark as a managed tx so that the inner function cannot manually rollback. + t.managed = true + + // If an error is returned from the function then pass it through. + err = fn(t) + t.managed = false + if err != nil { + _ = t.Rollback() + return err + } + + if err := t.Rollback(); err != nil { + return err + } + + return nil +} + +// Sync executes fdatasync() against the database file handle. +// +// This is not necessary under normal operation, however, if you use NoSync +// then it allows you to force the database file to sync against the disk. +func (db *DB) Sync() error { return fdatasync(db) } + +// Stats retrieves ongoing performance stats for the database. +// This is only updated when a transaction closes. +func (db *DB) Stats() Stats { + db.statlock.RLock() + defer db.statlock.RUnlock() + return db.stats +} + +// This is for internal access to the raw data bytes from the C cursor, use +// carefully, or not at all. +func (db *DB) Info() *Info { + return &Info{uintptr(unsafe.Pointer(&db.data[0])), db.pageSize} +} + +// page retrieves a page reference from the mmap based on the current page size. +func (db *DB) page(id pgid) *page { + pos := id * pgid(db.pageSize) + return (*page)(unsafe.Pointer(&db.data[pos])) +} + +// pageInBuffer retrieves a page reference from a given byte array based on the current page size. +func (db *DB) pageInBuffer(b []byte, id pgid) *page { + return (*page)(unsafe.Pointer(&b[id*pgid(db.pageSize)])) +} + +// meta retrieves the current meta page reference. +func (db *DB) meta() *meta { + if db.meta0.txid > db.meta1.txid { + return db.meta0 + } + return db.meta1 +} + +// allocate returns a contiguous block of memory starting at a given page. +func (db *DB) allocate(count int) (*page, error) { + // Allocate a temporary buffer for the page. + buf := make([]byte, count*db.pageSize) + p := (*page)(unsafe.Pointer(&buf[0])) + p.overflow = uint32(count - 1) + + // Use pages from the freelist if they are available. + if p.id = db.freelist.allocate(count); p.id != 0 { + return p, nil + } + + // Resize mmap() if we're at the end. + p.id = db.rwtx.meta.pgid + var minsz = int((p.id+pgid(count))+1) * db.pageSize + if minsz >= db.datasz { + if err := db.mmap(minsz); err != nil { + return nil, fmt.Errorf("mmap allocate error: %s", err) + } + } + + // Move the page id high water mark. + db.rwtx.meta.pgid += pgid(count) + + return p, nil +} + +func (db *DB) IsReadOnly() bool { + return db.readOnly +} + +// Options represents the options that can be set when opening a database. +type Options struct { + // Timeout is the amount of time to wait to obtain a file lock. + // When set to zero it will wait indefinitely. This option is only + // available on Darwin and Linux. + Timeout time.Duration + + // Sets the DB.NoGrowSync flag before memory mapping the file. + NoGrowSync bool + + // Open database in read-only mode. Uses flock(..., LOCK_SH |LOCK_NB) to + // grab a shared lock (UNIX). + ReadOnly bool +} + +// DefaultOptions represent the options used if nil options are passed into Open(). +// No timeout is used which will cause Bolt to wait indefinitely for a lock. +var DefaultOptions = &Options{ + Timeout: 0, + NoGrowSync: false, +} + +// Stats represents statistics about the database. +type Stats struct { + // Freelist stats + FreePageN int // total number of free pages on the freelist + PendingPageN int // total number of pending pages on the freelist + FreeAlloc int // total bytes allocated in free pages + FreelistInuse int // total bytes used by the freelist + + // Transaction stats + TxN int // total number of started read transactions + OpenTxN int // number of currently open read transactions + + TxStats TxStats // global, ongoing stats. +} + +// Sub calculates and returns the difference between two sets of database stats. +// This is useful when obtaining stats at two different points and time and +// you need the performance counters that occurred within that time span. +func (s *Stats) Sub(other *Stats) Stats { + if other == nil { + return *s + } + var diff Stats + diff.FreePageN = s.FreePageN + diff.PendingPageN = s.PendingPageN + diff.FreeAlloc = s.FreeAlloc + diff.FreelistInuse = s.FreelistInuse + diff.TxN = other.TxN - s.TxN + diff.TxStats = s.TxStats.Sub(&other.TxStats) + return diff +} + +func (s *Stats) add(other *Stats) { + s.TxStats.add(&other.TxStats) +} + +type Info struct { + Data uintptr + PageSize int +} + +type meta struct { + magic uint32 + version uint32 + pageSize uint32 + flags uint32 + root bucket + freelist pgid + pgid pgid + txid txid + checksum uint64 +} + +// validate checks the marker bytes and version of the meta page to ensure it matches this binary. +func (m *meta) validate() error { + if m.checksum != 0 && m.checksum != m.sum64() { + return ErrChecksum + } else if m.magic != magic { + return ErrInvalid + } else if m.version != version { + return ErrVersionMismatch + } + return nil +} + +// copy copies one meta object to another. +func (m *meta) copy(dest *meta) { + *dest = *m +} + +// write writes the meta onto a page. +func (m *meta) write(p *page) { + if m.root.root >= m.pgid { + panic(fmt.Sprintf("root bucket pgid (%d) above high water mark (%d)", m.root.root, m.pgid)) + } else if m.freelist >= m.pgid { + panic(fmt.Sprintf("freelist pgid (%d) above high water mark (%d)", m.freelist, m.pgid)) + } + + // Page id is either going to be 0 or 1 which we can determine by the transaction ID. + p.id = pgid(m.txid % 2) + p.flags |= metaPageFlag + + // Calculate the checksum. + m.checksum = m.sum64() + + m.copy(p.meta()) +} + +// generates the checksum for the meta. +func (m *meta) sum64() uint64 { + var h = fnv.New64a() + _, _ = h.Write((*[unsafe.Offsetof(meta{}.checksum)]byte)(unsafe.Pointer(m))[:]) + return h.Sum64() +} + +// _assert will panic with a given formatted message if the given condition is false. +func _assert(condition bool, msg string, v ...interface{}) { + if !condition { + panic(fmt.Sprintf("assertion failed: "+msg, v...)) + } +} + +func warn(v ...interface{}) { fmt.Fprintln(os.Stderr, v...) } +func warnf(msg string, v ...interface{}) { fmt.Fprintf(os.Stderr, msg+"\n", v...) } + +func printstack() { + stack := strings.Join(strings.Split(string(debug.Stack()), "\n")[2:], "\n") + fmt.Fprintln(os.Stderr, stack) +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/db_test.go b/Godeps/_workspace/src/github.com/boltdb/bolt/db_test.go new file mode 100644 index 0000000000000..dddf22b46fcd9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/db_test.go @@ -0,0 +1,903 @@ +package bolt_test + +import ( + "encoding/binary" + "errors" + "flag" + "fmt" + "io/ioutil" + "os" + "regexp" + "runtime" + "sort" + "strings" + "testing" + "time" + + "github.com/boltdb/bolt" +) + +var statsFlag = flag.Bool("stats", false, "show performance stats") + +// Ensure that opening a database with a bad path returns an error. +func TestOpen_BadPath(t *testing.T) { + db, err := bolt.Open("", 0666, nil) + assert(t, err != nil, "err: %s", err) + assert(t, db == nil, "") +} + +// Ensure that a database can be opened without error. +func TestOpen(t *testing.T) { + path := tempfile() + defer os.Remove(path) + db, err := bolt.Open(path, 0666, nil) + assert(t, db != nil, "") + ok(t, err) + equals(t, db.Path(), path) + ok(t, db.Close()) +} + +// Ensure that opening an already open database file will timeout. +func TestOpen_Timeout(t *testing.T) { + if runtime.GOOS == "windows" { + t.Skip("timeout not supported on windows") + } + + path := tempfile() + defer os.Remove(path) + + // Open a data file. + db0, err := bolt.Open(path, 0666, nil) + assert(t, db0 != nil, "") + ok(t, err) + + // Attempt to open the database again. + start := time.Now() + db1, err := bolt.Open(path, 0666, &bolt.Options{Timeout: 100 * time.Millisecond}) + assert(t, db1 == nil, "") + equals(t, bolt.ErrTimeout, err) + assert(t, time.Since(start) > 100*time.Millisecond, "") + + db0.Close() +} + +// Ensure that opening an already open database file will wait until its closed. +func TestOpen_Wait(t *testing.T) { + if runtime.GOOS == "windows" { + t.Skip("timeout not supported on windows") + } + + path := tempfile() + defer os.Remove(path) + + // Open a data file. + db0, err := bolt.Open(path, 0666, nil) + assert(t, db0 != nil, "") + ok(t, err) + + // Close it in just a bit. + time.AfterFunc(100*time.Millisecond, func() { db0.Close() }) + + // Attempt to open the database again. + start := time.Now() + db1, err := bolt.Open(path, 0666, &bolt.Options{Timeout: 200 * time.Millisecond}) + assert(t, db1 != nil, "") + ok(t, err) + assert(t, time.Since(start) > 100*time.Millisecond, "") +} + +// Ensure that opening a database does not increase its size. +// https://github.com/boltdb/bolt/issues/291 +func TestOpen_Size(t *testing.T) { + // Open a data file. + db := NewTestDB() + path := db.Path() + defer db.Close() + + // Insert until we get above the minimum 4MB size. + ok(t, db.Update(func(tx *bolt.Tx) error { + b, _ := tx.CreateBucketIfNotExists([]byte("data")) + for i := 0; i < 10000; i++ { + ok(t, b.Put([]byte(fmt.Sprintf("%04d", i)), make([]byte, 1000))) + } + return nil + })) + + // Close database and grab the size. + db.DB.Close() + sz := fileSize(path) + if sz == 0 { + t.Fatalf("unexpected new file size: %d", sz) + } + + // Reopen database, update, and check size again. + db0, err := bolt.Open(path, 0666, nil) + ok(t, err) + ok(t, db0.Update(func(tx *bolt.Tx) error { return tx.Bucket([]byte("data")).Put([]byte{0}, []byte{0}) })) + ok(t, db0.Close()) + newSz := fileSize(path) + if newSz == 0 { + t.Fatalf("unexpected new file size: %d", newSz) + } + + // Compare the original size with the new size. + if sz != newSz { + t.Fatalf("unexpected file growth: %d => %d", sz, newSz) + } +} + +// Ensure that opening a database beyond the max step size does not increase its size. +// https://github.com/boltdb/bolt/issues/303 +func TestOpen_Size_Large(t *testing.T) { + if testing.Short() { + t.Skip("short mode") + } + + // Open a data file. + db := NewTestDB() + path := db.Path() + defer db.Close() + + // Insert until we get above the minimum 4MB size. + var index uint64 + for i := 0; i < 10000; i++ { + ok(t, db.Update(func(tx *bolt.Tx) error { + b, _ := tx.CreateBucketIfNotExists([]byte("data")) + for j := 0; j < 1000; j++ { + ok(t, b.Put(u64tob(index), make([]byte, 50))) + index++ + } + return nil + })) + } + + // Close database and grab the size. + db.DB.Close() + sz := fileSize(path) + if sz == 0 { + t.Fatalf("unexpected new file size: %d", sz) + } else if sz < (1 << 30) { + t.Fatalf("expected larger initial size: %d", sz) + } + + // Reopen database, update, and check size again. + db0, err := bolt.Open(path, 0666, nil) + ok(t, err) + ok(t, db0.Update(func(tx *bolt.Tx) error { return tx.Bucket([]byte("data")).Put([]byte{0}, []byte{0}) })) + ok(t, db0.Close()) + newSz := fileSize(path) + if newSz == 0 { + t.Fatalf("unexpected new file size: %d", newSz) + } + + // Compare the original size with the new size. + if sz != newSz { + t.Fatalf("unexpected file growth: %d => %d", sz, newSz) + } +} + +// Ensure that a re-opened database is consistent. +func TestOpen_Check(t *testing.T) { + path := tempfile() + defer os.Remove(path) + + db, err := bolt.Open(path, 0666, nil) + ok(t, err) + ok(t, db.View(func(tx *bolt.Tx) error { return <-tx.Check() })) + db.Close() + + db, err = bolt.Open(path, 0666, nil) + ok(t, err) + ok(t, db.View(func(tx *bolt.Tx) error { return <-tx.Check() })) + db.Close() +} + +// Ensure that the database returns an error if the file handle cannot be open. +func TestDB_Open_FileError(t *testing.T) { + path := tempfile() + defer os.Remove(path) + + _, err := bolt.Open(path+"/youre-not-my-real-parent", 0666, nil) + assert(t, err.(*os.PathError) != nil, "") + equals(t, path+"/youre-not-my-real-parent", err.(*os.PathError).Path) + equals(t, "open", err.(*os.PathError).Op) +} + +// Ensure that write errors to the meta file handler during initialization are returned. +func TestDB_Open_MetaInitWriteError(t *testing.T) { + t.Skip("pending") +} + +// Ensure that a database that is too small returns an error. +func TestDB_Open_FileTooSmall(t *testing.T) { + path := tempfile() + defer os.Remove(path) + + db, err := bolt.Open(path, 0666, nil) + ok(t, err) + db.Close() + + // corrupt the database + ok(t, os.Truncate(path, int64(os.Getpagesize()))) + + db, err = bolt.Open(path, 0666, nil) + equals(t, errors.New("file size too small"), err) +} + +// Ensure that a database can be opened in read-only mode by multiple processes +// and that a database can not be opened in read-write mode and in read-only +// mode at the same time. +func TestOpen_ReadOnly(t *testing.T) { + bucket, key, value := []byte(`bucket`), []byte(`key`), []byte(`value`) + + path := tempfile() + defer os.Remove(path) + + // Open in read-write mode. + db, err := bolt.Open(path, 0666, nil) + ok(t, db.Update(func(tx *bolt.Tx) error { + b, err := tx.CreateBucket(bucket) + if err != nil { + return err + } + return b.Put(key, value) + })) + assert(t, db != nil, "") + assert(t, !db.IsReadOnly(), "") + ok(t, err) + ok(t, db.Close()) + + // Open in read-only mode. + db0, err := bolt.Open(path, 0666, &bolt.Options{ReadOnly: true}) + ok(t, err) + defer db0.Close() + + // Opening in read-write mode should return an error. + _, err = bolt.Open(path, 0666, &bolt.Options{Timeout: time.Millisecond * 100}) + assert(t, err != nil, "") + + // And again (in read-only mode). + db1, err := bolt.Open(path, 0666, &bolt.Options{ReadOnly: true}) + ok(t, err) + defer db1.Close() + + // Verify both read-only databases are accessible. + for _, db := range []*bolt.DB{db0, db1} { + // Verify is is in read only mode indeed. + assert(t, db.IsReadOnly(), "") + + // Read-only databases should not allow updates. + assert(t, + bolt.ErrDatabaseReadOnly == db.Update(func(*bolt.Tx) error { + panic(`should never get here`) + }), + "") + + // Read-only databases should not allow beginning writable txns. + _, err = db.Begin(true) + assert(t, bolt.ErrDatabaseReadOnly == err, "") + + // Verify the data. + ok(t, db.View(func(tx *bolt.Tx) error { + b := tx.Bucket(bucket) + if b == nil { + return fmt.Errorf("expected bucket `%s`", string(bucket)) + } + + got := string(b.Get(key)) + expected := string(value) + if got != expected { + return fmt.Errorf("expected `%s`, got `%s`", expected, got) + } + return nil + })) + } +} + +// TODO(benbjohnson): Test corruption at every byte of the first two pages. + +// Ensure that a database cannot open a transaction when it's not open. +func TestDB_Begin_DatabaseNotOpen(t *testing.T) { + var db bolt.DB + tx, err := db.Begin(false) + assert(t, tx == nil, "") + equals(t, err, bolt.ErrDatabaseNotOpen) +} + +// Ensure that a read-write transaction can be retrieved. +func TestDB_BeginRW(t *testing.T) { + db := NewTestDB() + defer db.Close() + tx, err := db.Begin(true) + assert(t, tx != nil, "") + ok(t, err) + assert(t, tx.DB() == db.DB, "") + equals(t, tx.Writable(), true) + ok(t, tx.Commit()) +} + +// Ensure that opening a transaction while the DB is closed returns an error. +func TestDB_BeginRW_Closed(t *testing.T) { + var db bolt.DB + tx, err := db.Begin(true) + equals(t, err, bolt.ErrDatabaseNotOpen) + assert(t, tx == nil, "") +} + +func TestDB_Close_PendingTx_RW(t *testing.T) { testDB_Close_PendingTx(t, true) } +func TestDB_Close_PendingTx_RO(t *testing.T) { testDB_Close_PendingTx(t, false) } + +// Ensure that a database cannot close while transactions are open. +func testDB_Close_PendingTx(t *testing.T, writable bool) { + db := NewTestDB() + defer db.Close() + + // Start transaction. + tx, err := db.Begin(true) + if err != nil { + t.Fatal(err) + } + + // Open update in separate goroutine. + done := make(chan struct{}) + go func() { + db.Close() + close(done) + }() + + // Ensure database hasn't closed. + time.Sleep(100 * time.Millisecond) + select { + case <-done: + t.Fatal("database closed too early") + default: + } + + // Commit transaction. + if err := tx.Commit(); err != nil { + t.Fatal(err) + } + + // Ensure database closed now. + time.Sleep(100 * time.Millisecond) + select { + case <-done: + default: + t.Fatal("database did not close") + } +} + +// Ensure a database can provide a transactional block. +func TestDB_Update(t *testing.T) { + db := NewTestDB() + defer db.Close() + err := db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + b := tx.Bucket([]byte("widgets")) + b.Put([]byte("foo"), []byte("bar")) + b.Put([]byte("baz"), []byte("bat")) + b.Delete([]byte("foo")) + return nil + }) + ok(t, err) + err = db.View(func(tx *bolt.Tx) error { + assert(t, tx.Bucket([]byte("widgets")).Get([]byte("foo")) == nil, "") + equals(t, []byte("bat"), tx.Bucket([]byte("widgets")).Get([]byte("baz"))) + return nil + }) + ok(t, err) +} + +// Ensure a closed database returns an error while running a transaction block +func TestDB_Update_Closed(t *testing.T) { + var db bolt.DB + err := db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + return nil + }) + equals(t, err, bolt.ErrDatabaseNotOpen) +} + +// Ensure a panic occurs while trying to commit a managed transaction. +func TestDB_Update_ManualCommit(t *testing.T) { + db := NewTestDB() + defer db.Close() + + var ok bool + db.Update(func(tx *bolt.Tx) error { + func() { + defer func() { + if r := recover(); r != nil { + ok = true + } + }() + tx.Commit() + }() + return nil + }) + assert(t, ok, "expected panic") +} + +// Ensure a panic occurs while trying to rollback a managed transaction. +func TestDB_Update_ManualRollback(t *testing.T) { + db := NewTestDB() + defer db.Close() + + var ok bool + db.Update(func(tx *bolt.Tx) error { + func() { + defer func() { + if r := recover(); r != nil { + ok = true + } + }() + tx.Rollback() + }() + return nil + }) + assert(t, ok, "expected panic") +} + +// Ensure a panic occurs while trying to commit a managed transaction. +func TestDB_View_ManualCommit(t *testing.T) { + db := NewTestDB() + defer db.Close() + + var ok bool + db.Update(func(tx *bolt.Tx) error { + func() { + defer func() { + if r := recover(); r != nil { + ok = true + } + }() + tx.Commit() + }() + return nil + }) + assert(t, ok, "expected panic") +} + +// Ensure a panic occurs while trying to rollback a managed transaction. +func TestDB_View_ManualRollback(t *testing.T) { + db := NewTestDB() + defer db.Close() + + var ok bool + db.Update(func(tx *bolt.Tx) error { + func() { + defer func() { + if r := recover(); r != nil { + ok = true + } + }() + tx.Rollback() + }() + return nil + }) + assert(t, ok, "expected panic") +} + +// Ensure a write transaction that panics does not hold open locks. +func TestDB_Update_Panic(t *testing.T) { + db := NewTestDB() + defer db.Close() + + func() { + defer func() { + if r := recover(); r != nil { + t.Log("recover: update", r) + } + }() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + panic("omg") + }) + }() + + // Verify we can update again. + err := db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte("widgets")) + return err + }) + ok(t, err) + + // Verify that our change persisted. + err = db.Update(func(tx *bolt.Tx) error { + assert(t, tx.Bucket([]byte("widgets")) != nil, "") + return nil + }) +} + +// Ensure a database can return an error through a read-only transactional block. +func TestDB_View_Error(t *testing.T) { + db := NewTestDB() + defer db.Close() + err := db.View(func(tx *bolt.Tx) error { + return errors.New("xxx") + }) + equals(t, errors.New("xxx"), err) +} + +// Ensure a read transaction that panics does not hold open locks. +func TestDB_View_Panic(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + return nil + }) + + func() { + defer func() { + if r := recover(); r != nil { + t.Log("recover: view", r) + } + }() + db.View(func(tx *bolt.Tx) error { + assert(t, tx.Bucket([]byte("widgets")) != nil, "") + panic("omg") + }) + }() + + // Verify that we can still use read transactions. + db.View(func(tx *bolt.Tx) error { + assert(t, tx.Bucket([]byte("widgets")) != nil, "") + return nil + }) +} + +// Ensure that an error is returned when a database write fails. +func TestDB_Commit_WriteFail(t *testing.T) { + t.Skip("pending") // TODO(benbjohnson) +} + +// Ensure that DB stats can be returned. +func TestDB_Stats(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte("widgets")) + return err + }) + stats := db.Stats() + equals(t, 2, stats.TxStats.PageCount) + equals(t, 0, stats.FreePageN) + equals(t, 2, stats.PendingPageN) +} + +// Ensure that database pages are in expected order and type. +func TestDB_Consistency(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte("widgets")) + return err + }) + + for i := 0; i < 10; i++ { + db.Update(func(tx *bolt.Tx) error { + ok(t, tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar"))) + return nil + }) + } + db.Update(func(tx *bolt.Tx) error { + p, _ := tx.Page(0) + assert(t, p != nil, "") + equals(t, "meta", p.Type) + + p, _ = tx.Page(1) + assert(t, p != nil, "") + equals(t, "meta", p.Type) + + p, _ = tx.Page(2) + assert(t, p != nil, "") + equals(t, "free", p.Type) + + p, _ = tx.Page(3) + assert(t, p != nil, "") + equals(t, "free", p.Type) + + p, _ = tx.Page(4) + assert(t, p != nil, "") + equals(t, "leaf", p.Type) + + p, _ = tx.Page(5) + assert(t, p != nil, "") + equals(t, "freelist", p.Type) + + p, _ = tx.Page(6) + assert(t, p == nil, "") + return nil + }) +} + +// Ensure that DB stats can be substracted from one another. +func TestDBStats_Sub(t *testing.T) { + var a, b bolt.Stats + a.TxStats.PageCount = 3 + a.FreePageN = 4 + b.TxStats.PageCount = 10 + b.FreePageN = 14 + diff := b.Sub(&a) + equals(t, 7, diff.TxStats.PageCount) + // free page stats are copied from the receiver and not subtracted + equals(t, 14, diff.FreePageN) +} + +func ExampleDB_Update() { + // Open the database. + db, _ := bolt.Open(tempfile(), 0666, nil) + defer os.Remove(db.Path()) + defer db.Close() + + // Execute several commands within a write transaction. + err := db.Update(func(tx *bolt.Tx) error { + b, err := tx.CreateBucket([]byte("widgets")) + if err != nil { + return err + } + if err := b.Put([]byte("foo"), []byte("bar")); err != nil { + return err + } + return nil + }) + + // If our transactional block didn't return an error then our data is saved. + if err == nil { + db.View(func(tx *bolt.Tx) error { + value := tx.Bucket([]byte("widgets")).Get([]byte("foo")) + fmt.Printf("The value of 'foo' is: %s\n", value) + return nil + }) + } + + // Output: + // The value of 'foo' is: bar +} + +func ExampleDB_View() { + // Open the database. + db, _ := bolt.Open(tempfile(), 0666, nil) + defer os.Remove(db.Path()) + defer db.Close() + + // Insert data into a bucket. + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("people")) + b := tx.Bucket([]byte("people")) + b.Put([]byte("john"), []byte("doe")) + b.Put([]byte("susy"), []byte("que")) + return nil + }) + + // Access data from within a read-only transactional block. + db.View(func(tx *bolt.Tx) error { + v := tx.Bucket([]byte("people")).Get([]byte("john")) + fmt.Printf("John's last name is %s.\n", v) + return nil + }) + + // Output: + // John's last name is doe. +} + +func ExampleDB_Begin_ReadOnly() { + // Open the database. + db, _ := bolt.Open(tempfile(), 0666, nil) + defer os.Remove(db.Path()) + defer db.Close() + + // Create a bucket. + db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte("widgets")) + return err + }) + + // Create several keys in a transaction. + tx, _ := db.Begin(true) + b := tx.Bucket([]byte("widgets")) + b.Put([]byte("john"), []byte("blue")) + b.Put([]byte("abby"), []byte("red")) + b.Put([]byte("zephyr"), []byte("purple")) + tx.Commit() + + // Iterate over the values in sorted key order. + tx, _ = db.Begin(false) + c := tx.Bucket([]byte("widgets")).Cursor() + for k, v := c.First(); k != nil; k, v = c.Next() { + fmt.Printf("%s likes %s\n", k, v) + } + tx.Rollback() + + // Output: + // abby likes red + // john likes blue + // zephyr likes purple +} + +// TestDB represents a wrapper around a Bolt DB to handle temporary file +// creation and automatic cleanup on close. +type TestDB struct { + *bolt.DB +} + +// NewTestDB returns a new instance of TestDB. +func NewTestDB() *TestDB { + db, err := bolt.Open(tempfile(), 0666, nil) + if err != nil { + panic("cannot open db: " + err.Error()) + } + return &TestDB{db} +} + +// MustView executes a read-only function. Panic on error. +func (db *TestDB) MustView(fn func(tx *bolt.Tx) error) { + if err := db.DB.View(func(tx *bolt.Tx) error { + return fn(tx) + }); err != nil { + panic(err.Error()) + } +} + +// MustUpdate executes a read-write function. Panic on error. +func (db *TestDB) MustUpdate(fn func(tx *bolt.Tx) error) { + if err := db.DB.View(func(tx *bolt.Tx) error { + return fn(tx) + }); err != nil { + panic(err.Error()) + } +} + +// MustCreateBucket creates a new bucket. Panic on error. +func (db *TestDB) MustCreateBucket(name []byte) { + if err := db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte(name)) + return err + }); err != nil { + panic(err.Error()) + } +} + +// Close closes the database and deletes the underlying file. +func (db *TestDB) Close() { + // Log statistics. + if *statsFlag { + db.PrintStats() + } + + // Check database consistency after every test. + db.MustCheck() + + // Close database and remove file. + defer os.Remove(db.Path()) + db.DB.Close() +} + +// PrintStats prints the database stats +func (db *TestDB) PrintStats() { + var stats = db.Stats() + fmt.Printf("[db] %-20s %-20s %-20s\n", + fmt.Sprintf("pg(%d/%d)", stats.TxStats.PageCount, stats.TxStats.PageAlloc), + fmt.Sprintf("cur(%d)", stats.TxStats.CursorCount), + fmt.Sprintf("node(%d/%d)", stats.TxStats.NodeCount, stats.TxStats.NodeDeref), + ) + fmt.Printf(" %-20s %-20s %-20s\n", + fmt.Sprintf("rebal(%d/%v)", stats.TxStats.Rebalance, truncDuration(stats.TxStats.RebalanceTime)), + fmt.Sprintf("spill(%d/%v)", stats.TxStats.Spill, truncDuration(stats.TxStats.SpillTime)), + fmt.Sprintf("w(%d/%v)", stats.TxStats.Write, truncDuration(stats.TxStats.WriteTime)), + ) +} + +// MustCheck runs a consistency check on the database and panics if any errors are found. +func (db *TestDB) MustCheck() { + db.Update(func(tx *bolt.Tx) error { + // Collect all the errors. + var errors []error + for err := range tx.Check() { + errors = append(errors, err) + if len(errors) > 10 { + break + } + } + + // If errors occurred, copy the DB and print the errors. + if len(errors) > 0 { + var path = tempfile() + tx.CopyFile(path, 0600) + + // Print errors. + fmt.Print("\n\n") + fmt.Printf("consistency check failed (%d errors)\n", len(errors)) + for _, err := range errors { + fmt.Println(err) + } + fmt.Println("") + fmt.Println("db saved to:") + fmt.Println(path) + fmt.Print("\n\n") + os.Exit(-1) + } + + return nil + }) +} + +// CopyTempFile copies a database to a temporary file. +func (db *TestDB) CopyTempFile() { + path := tempfile() + db.View(func(tx *bolt.Tx) error { return tx.CopyFile(path, 0600) }) + fmt.Println("db copied to: ", path) +} + +// tempfile returns a temporary file path. +func tempfile() string { + f, _ := ioutil.TempFile("", "bolt-") + f.Close() + os.Remove(f.Name()) + return f.Name() +} + +// mustContainKeys checks that a bucket contains a given set of keys. +func mustContainKeys(b *bolt.Bucket, m map[string]string) { + found := make(map[string]string) + b.ForEach(func(k, _ []byte) error { + found[string(k)] = "" + return nil + }) + + // Check for keys found in bucket that shouldn't be there. + var keys []string + for k, _ := range found { + if _, ok := m[string(k)]; !ok { + keys = append(keys, k) + } + } + if len(keys) > 0 { + sort.Strings(keys) + panic(fmt.Sprintf("keys found(%d): %s", len(keys), strings.Join(keys, ","))) + } + + // Check for keys not found in bucket that should be there. + for k, _ := range m { + if _, ok := found[string(k)]; !ok { + keys = append(keys, k) + } + } + if len(keys) > 0 { + sort.Strings(keys) + panic(fmt.Sprintf("keys not found(%d): %s", len(keys), strings.Join(keys, ","))) + } +} + +func trunc(b []byte, length int) []byte { + if length < len(b) { + return b[:length] + } + return b +} + +func truncDuration(d time.Duration) string { + return regexp.MustCompile(`^(\d+)(\.\d+)`).ReplaceAllString(d.String(), "$1") +} + +func fileSize(path string) int64 { + fi, err := os.Stat(path) + if err != nil { + return 0 + } + return fi.Size() +} + +func warn(v ...interface{}) { fmt.Fprintln(os.Stderr, v...) } +func warnf(msg string, v ...interface{}) { fmt.Fprintf(os.Stderr, msg+"\n", v...) } + +// u64tob converts a uint64 into an 8-byte slice. +func u64tob(v uint64) []byte { + b := make([]byte, 8) + binary.BigEndian.PutUint64(b, v) + return b +} + +// btou64 converts an 8-byte slice into an uint64. +func btou64(b []byte) uint64 { return binary.BigEndian.Uint64(b) } diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/doc.go b/Godeps/_workspace/src/github.com/boltdb/bolt/doc.go new file mode 100644 index 0000000000000..cc937845dba0d --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/doc.go @@ -0,0 +1,44 @@ +/* +Package bolt implements a low-level key/value store in pure Go. It supports +fully serializable transactions, ACID semantics, and lock-free MVCC with +multiple readers and a single writer. Bolt can be used for projects that +want a simple data store without the need to add large dependencies such as +Postgres or MySQL. + +Bolt is a single-level, zero-copy, B+tree data store. This means that Bolt is +optimized for fast read access and does not require recovery in the event of a +system crash. Transactions which have not finished committing will simply be +rolled back in the event of a crash. + +The design of Bolt is based on Howard Chu's LMDB database project. + +Bolt currently works on Windows, Mac OS X, and Linux. + + +Basics + +There are only a few types in Bolt: DB, Bucket, Tx, and Cursor. The DB is +a collection of buckets and is represented by a single file on disk. A bucket is +a collection of unique keys that are associated with values. + +Transactions provide either read-only or read-write access to the database. +Read-only transactions can retrieve key/value pairs and can use Cursors to +iterate over the dataset sequentially. Read-write transactions can create and +delete buckets and can insert and remove keys. Only one read-write transaction +is allowed at a time. + + +Caveats + +The database uses a read-only, memory-mapped data file to ensure that +applications cannot corrupt the database, however, this means that keys and +values returned from Bolt cannot be changed. Writing to a read-only byte slice +will cause Go to panic. + +Keys and values retrieved from the database are only valid for the life of +the transaction. When used outside the transaction, these byte slices can +point to different data or can point to invalid memory which will cause a panic. + + +*/ +package bolt diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/errors.go b/Godeps/_workspace/src/github.com/boltdb/bolt/errors.go new file mode 100644 index 0000000000000..6883786d5da71 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/errors.go @@ -0,0 +1,70 @@ +package bolt + +import "errors" + +// These errors can be returned when opening or calling methods on a DB. +var ( + // ErrDatabaseNotOpen is returned when a DB instance is accessed before it + // is opened or after it is closed. + ErrDatabaseNotOpen = errors.New("database not open") + + // ErrDatabaseOpen is returned when opening a database that is + // already open. + ErrDatabaseOpen = errors.New("database already open") + + // ErrInvalid is returned when a data file is not a Bolt-formatted database. + ErrInvalid = errors.New("invalid database") + + // ErrVersionMismatch is returned when the data file was created with a + // different version of Bolt. + ErrVersionMismatch = errors.New("version mismatch") + + // ErrChecksum is returned when either meta page checksum does not match. + ErrChecksum = errors.New("checksum error") + + // ErrTimeout is returned when a database cannot obtain an exclusive lock + // on the data file after the timeout passed to Open(). + ErrTimeout = errors.New("timeout") +) + +// These errors can occur when beginning or committing a Tx. +var ( + // ErrTxNotWritable is returned when performing a write operation on a + // read-only transaction. + ErrTxNotWritable = errors.New("tx not writable") + + // ErrTxClosed is returned when committing or rolling back a transaction + // that has already been committed or rolled back. + ErrTxClosed = errors.New("tx closed") + + // ErrDatabaseReadOnly is returned when a mutating transaction is started on a + // read-only database. + ErrDatabaseReadOnly = errors.New("database is in read-only mode") +) + +// These errors can occur when putting or deleting a value or a bucket. +var ( + // ErrBucketNotFound is returned when trying to access a bucket that has + // not been created yet. + ErrBucketNotFound = errors.New("bucket not found") + + // ErrBucketExists is returned when creating a bucket that already exists. + ErrBucketExists = errors.New("bucket already exists") + + // ErrBucketNameRequired is returned when creating a bucket with a blank name. + ErrBucketNameRequired = errors.New("bucket name required") + + // ErrKeyRequired is returned when inserting a zero-length key. + ErrKeyRequired = errors.New("key required") + + // ErrKeyTooLarge is returned when inserting a key that is larger than MaxKeySize. + ErrKeyTooLarge = errors.New("key too large") + + // ErrValueTooLarge is returned when inserting a value that is larger than MaxValueSize. + ErrValueTooLarge = errors.New("value too large") + + // ErrIncompatibleValue is returned when trying create or delete a bucket + // on an existing non-bucket key or when trying to create or delete a + // non-bucket key on an existing bucket key. + ErrIncompatibleValue = errors.New("incompatible value") +) diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/freelist.go b/Godeps/_workspace/src/github.com/boltdb/bolt/freelist.go new file mode 100644 index 0000000000000..0161948fcf765 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/freelist.go @@ -0,0 +1,242 @@ +package bolt + +import ( + "fmt" + "sort" + "unsafe" +) + +// freelist represents a list of all pages that are available for allocation. +// It also tracks pages that have been freed but are still in use by open transactions. +type freelist struct { + ids []pgid // all free and available free page ids. + pending map[txid][]pgid // mapping of soon-to-be free page ids by tx. + cache map[pgid]bool // fast lookup of all free and pending page ids. +} + +// newFreelist returns an empty, initialized freelist. +func newFreelist() *freelist { + return &freelist{ + pending: make(map[txid][]pgid), + cache: make(map[pgid]bool), + } +} + +// size returns the size of the page after serialization. +func (f *freelist) size() int { + return pageHeaderSize + (int(unsafe.Sizeof(pgid(0))) * f.count()) +} + +// count returns count of pages on the freelist +func (f *freelist) count() int { + return f.free_count() + f.pending_count() +} + +// free_count returns count of free pages +func (f *freelist) free_count() int { + return len(f.ids) +} + +// pending_count returns count of pending pages +func (f *freelist) pending_count() int { + var count int + for _, list := range f.pending { + count += len(list) + } + return count +} + +// all returns a list of all free ids and all pending ids in one sorted list. +func (f *freelist) all() []pgid { + m := make(pgids, 0) + + for _, list := range f.pending { + m = append(m, list...) + } + + sort.Sort(m) + return pgids(f.ids).merge(m) +} + +// allocate returns the starting page id of a contiguous list of pages of a given size. +// If a contiguous block cannot be found then 0 is returned. +func (f *freelist) allocate(n int) pgid { + if len(f.ids) == 0 { + return 0 + } + + var initial, previd pgid + for i, id := range f.ids { + if id <= 1 { + panic(fmt.Sprintf("invalid page allocation: %d", id)) + } + + // Reset initial page if this is not contiguous. + if previd == 0 || id-previd != 1 { + initial = id + } + + // If we found a contiguous block then remove it and return it. + if (id-initial)+1 == pgid(n) { + // If we're allocating off the beginning then take the fast path + // and just adjust the existing slice. This will use extra memory + // temporarily but the append() in free() will realloc the slice + // as is necessary. + if (i + 1) == n { + f.ids = f.ids[i+1:] + } else { + copy(f.ids[i-n+1:], f.ids[i+1:]) + f.ids = f.ids[:len(f.ids)-n] + } + + // Remove from the free cache. + for i := pgid(0); i < pgid(n); i++ { + delete(f.cache, initial+i) + } + + return initial + } + + previd = id + } + return 0 +} + +// free releases a page and its overflow for a given transaction id. +// If the page is already free then a panic will occur. +func (f *freelist) free(txid txid, p *page) { + if p.id <= 1 { + panic(fmt.Sprintf("cannot free page 0 or 1: %d", p.id)) + } + + // Free page and all its overflow pages. + var ids = f.pending[txid] + for id := p.id; id <= p.id+pgid(p.overflow); id++ { + // Verify that page is not already free. + if f.cache[id] { + panic(fmt.Sprintf("page %d already freed", id)) + } + + // Add to the freelist and cache. + ids = append(ids, id) + f.cache[id] = true + } + f.pending[txid] = ids +} + +// release moves all page ids for a transaction id (or older) to the freelist. +func (f *freelist) release(txid txid) { + m := make(pgids, 0) + for tid, ids := range f.pending { + if tid <= txid { + // Move transaction's pending pages to the available freelist. + // Don't remove from the cache since the page is still free. + m = append(m, ids...) + delete(f.pending, tid) + } + } + sort.Sort(m) + f.ids = pgids(f.ids).merge(m) +} + +// rollback removes the pages from a given pending tx. +func (f *freelist) rollback(txid txid) { + // Remove page ids from cache. + for _, id := range f.pending[txid] { + delete(f.cache, id) + } + + // Remove pages from pending list. + delete(f.pending, txid) +} + +// freed returns whether a given page is in the free list. +func (f *freelist) freed(pgid pgid) bool { + return f.cache[pgid] +} + +// read initializes the freelist from a freelist page. +func (f *freelist) read(p *page) { + // If the page.count is at the max uint16 value (64k) then it's considered + // an overflow and the size of the freelist is stored as the first element. + idx, count := 0, int(p.count) + if count == 0xFFFF { + idx = 1 + count = int(((*[maxAllocSize]pgid)(unsafe.Pointer(&p.ptr)))[0]) + } + + // Copy the list of page ids from the freelist. + ids := ((*[maxAllocSize]pgid)(unsafe.Pointer(&p.ptr)))[idx:count] + f.ids = make([]pgid, len(ids)) + copy(f.ids, ids) + + // Make sure they're sorted. + sort.Sort(pgids(f.ids)) + + // Rebuild the page cache. + f.reindex() +} + +// write writes the page ids onto a freelist page. All free and pending ids are +// saved to disk since in the event of a program crash, all pending ids will +// become free. +func (f *freelist) write(p *page) error { + // Combine the old free pgids and pgids waiting on an open transaction. + ids := f.all() + + // Update the header flag. + p.flags |= freelistPageFlag + + // The page.count can only hold up to 64k elements so if we overflow that + // number then we handle it by putting the size in the first element. + if len(ids) < 0xFFFF { + p.count = uint16(len(ids)) + copy(((*[maxAllocSize]pgid)(unsafe.Pointer(&p.ptr)))[:], ids) + } else { + p.count = 0xFFFF + ((*[maxAllocSize]pgid)(unsafe.Pointer(&p.ptr)))[0] = pgid(len(ids)) + copy(((*[maxAllocSize]pgid)(unsafe.Pointer(&p.ptr)))[1:], ids) + } + + return nil +} + +// reload reads the freelist from a page and filters out pending items. +func (f *freelist) reload(p *page) { + f.read(p) + + // Build a cache of only pending pages. + pcache := make(map[pgid]bool) + for _, pendingIDs := range f.pending { + for _, pendingID := range pendingIDs { + pcache[pendingID] = true + } + } + + // Check each page in the freelist and build a new available freelist + // with any pages not in the pending lists. + var a []pgid + for _, id := range f.ids { + if !pcache[id] { + a = append(a, id) + } + } + f.ids = a + + // Once the available list is rebuilt then rebuild the free cache so that + // it includes the available and pending free pages. + f.reindex() +} + +// reindex rebuilds the free cache based on available and pending free lists. +func (f *freelist) reindex() { + f.cache = make(map[pgid]bool) + for _, id := range f.ids { + f.cache[id] = true + } + for _, pendingIDs := range f.pending { + for _, pendingID := range pendingIDs { + f.cache[pendingID] = true + } + } +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/freelist_test.go b/Godeps/_workspace/src/github.com/boltdb/bolt/freelist_test.go new file mode 100644 index 0000000000000..8caeab2ec4ec1 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/freelist_test.go @@ -0,0 +1,156 @@ +package bolt + +import ( + "math/rand" + "reflect" + "sort" + "testing" + "unsafe" +) + +// Ensure that a page is added to a transaction's freelist. +func TestFreelist_free(t *testing.T) { + f := newFreelist() + f.free(100, &page{id: 12}) + if !reflect.DeepEqual([]pgid{12}, f.pending[100]) { + t.Fatalf("exp=%v; got=%v", []pgid{12}, f.pending[100]) + } +} + +// Ensure that a page and its overflow is added to a transaction's freelist. +func TestFreelist_free_overflow(t *testing.T) { + f := newFreelist() + f.free(100, &page{id: 12, overflow: 3}) + if exp := []pgid{12, 13, 14, 15}; !reflect.DeepEqual(exp, f.pending[100]) { + t.Fatalf("exp=%v; got=%v", exp, f.pending[100]) + } +} + +// Ensure that a transaction's free pages can be released. +func TestFreelist_release(t *testing.T) { + f := newFreelist() + f.free(100, &page{id: 12, overflow: 1}) + f.free(100, &page{id: 9}) + f.free(102, &page{id: 39}) + f.release(100) + f.release(101) + if exp := []pgid{9, 12, 13}; !reflect.DeepEqual(exp, f.ids) { + t.Fatalf("exp=%v; got=%v", exp, f.ids) + } + + f.release(102) + if exp := []pgid{9, 12, 13, 39}; !reflect.DeepEqual(exp, f.ids) { + t.Fatalf("exp=%v; got=%v", exp, f.ids) + } +} + +// Ensure that a freelist can find contiguous blocks of pages. +func TestFreelist_allocate(t *testing.T) { + f := &freelist{ids: []pgid{3, 4, 5, 6, 7, 9, 12, 13, 18}} + if id := int(f.allocate(3)); id != 3 { + t.Fatalf("exp=3; got=%v", id) + } + if id := int(f.allocate(1)); id != 6 { + t.Fatalf("exp=6; got=%v", id) + } + if id := int(f.allocate(3)); id != 0 { + t.Fatalf("exp=0; got=%v", id) + } + if id := int(f.allocate(2)); id != 12 { + t.Fatalf("exp=12; got=%v", id) + } + if id := int(f.allocate(1)); id != 7 { + t.Fatalf("exp=7; got=%v", id) + } + if id := int(f.allocate(0)); id != 0 { + t.Fatalf("exp=0; got=%v", id) + } + if id := int(f.allocate(0)); id != 0 { + t.Fatalf("exp=0; got=%v", id) + } + if exp := []pgid{9, 18}; !reflect.DeepEqual(exp, f.ids) { + t.Fatalf("exp=%v; got=%v", exp, f.ids) + } + + if id := int(f.allocate(1)); id != 9 { + t.Fatalf("exp=9; got=%v", id) + } + if id := int(f.allocate(1)); id != 18 { + t.Fatalf("exp=18; got=%v", id) + } + if id := int(f.allocate(1)); id != 0 { + t.Fatalf("exp=0; got=%v", id) + } + if exp := []pgid{}; !reflect.DeepEqual(exp, f.ids) { + t.Fatalf("exp=%v; got=%v", exp, f.ids) + } +} + +// Ensure that a freelist can deserialize from a freelist page. +func TestFreelist_read(t *testing.T) { + // Create a page. + var buf [4096]byte + page := (*page)(unsafe.Pointer(&buf[0])) + page.flags = freelistPageFlag + page.count = 2 + + // Insert 2 page ids. + ids := (*[3]pgid)(unsafe.Pointer(&page.ptr)) + ids[0] = 23 + ids[1] = 50 + + // Deserialize page into a freelist. + f := newFreelist() + f.read(page) + + // Ensure that there are two page ids in the freelist. + if exp := []pgid{23, 50}; !reflect.DeepEqual(exp, f.ids) { + t.Fatalf("exp=%v; got=%v", exp, f.ids) + } +} + +// Ensure that a freelist can serialize into a freelist page. +func TestFreelist_write(t *testing.T) { + // Create a freelist and write it to a page. + var buf [4096]byte + f := &freelist{ids: []pgid{12, 39}, pending: make(map[txid][]pgid)} + f.pending[100] = []pgid{28, 11} + f.pending[101] = []pgid{3} + p := (*page)(unsafe.Pointer(&buf[0])) + f.write(p) + + // Read the page back out. + f2 := newFreelist() + f2.read(p) + + // Ensure that the freelist is correct. + // All pages should be present and in reverse order. + if exp := []pgid{3, 11, 12, 28, 39}; !reflect.DeepEqual(exp, f2.ids) { + t.Fatalf("exp=%v; got=%v", exp, f2.ids) + } +} + +func Benchmark_FreelistRelease10K(b *testing.B) { benchmark_FreelistRelease(b, 10000) } +func Benchmark_FreelistRelease100K(b *testing.B) { benchmark_FreelistRelease(b, 100000) } +func Benchmark_FreelistRelease1000K(b *testing.B) { benchmark_FreelistRelease(b, 1000000) } +func Benchmark_FreelistRelease10000K(b *testing.B) { benchmark_FreelistRelease(b, 10000000) } + +func benchmark_FreelistRelease(b *testing.B, size int) { + ids := randomPgids(size) + pending := randomPgids(len(ids) / 400) + b.ResetTimer() + for i := 0; i < b.N; i++ { + f := &freelist{ids: ids, pending: map[txid][]pgid{1: pending}} + f.release(1) + } +} + +func randomPgids(n int) []pgid { + rand.Seed(42) + pgids := make(pgids, n) + for i := range pgids { + pgids[i] = pgid(rand.Int63()) + } + sort.Sort(pgids) + return pgids +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/node.go b/Godeps/_workspace/src/github.com/boltdb/bolt/node.go new file mode 100644 index 0000000000000..c9fb21c73149f --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/node.go @@ -0,0 +1,636 @@ +package bolt + +import ( + "bytes" + "fmt" + "sort" + "unsafe" +) + +// node represents an in-memory, deserialized page. +type node struct { + bucket *Bucket + isLeaf bool + unbalanced bool + spilled bool + key []byte + pgid pgid + parent *node + children nodes + inodes inodes +} + +// root returns the top-level node this node is attached to. +func (n *node) root() *node { + if n.parent == nil { + return n + } + return n.parent.root() +} + +// minKeys returns the minimum number of inodes this node should have. +func (n *node) minKeys() int { + if n.isLeaf { + return 1 + } + return 2 +} + +// size returns the size of the node after serialization. +func (n *node) size() int { + sz, elsz := pageHeaderSize, n.pageElementSize() + for i := 0; i < len(n.inodes); i++ { + item := &n.inodes[i] + sz += elsz + len(item.key) + len(item.value) + } + return sz +} + +// sizeLessThan returns true if the node is less than a given size. +// This is an optimization to avoid calculating a large node when we only need +// to know if it fits inside a certain page size. +func (n *node) sizeLessThan(v int) bool { + sz, elsz := pageHeaderSize, n.pageElementSize() + for i := 0; i < len(n.inodes); i++ { + item := &n.inodes[i] + sz += elsz + len(item.key) + len(item.value) + if sz >= v { + return false + } + } + return true +} + +// pageElementSize returns the size of each page element based on the type of node. +func (n *node) pageElementSize() int { + if n.isLeaf { + return leafPageElementSize + } + return branchPageElementSize +} + +// childAt returns the child node at a given index. +func (n *node) childAt(index int) *node { + if n.isLeaf { + panic(fmt.Sprintf("invalid childAt(%d) on a leaf node", index)) + } + return n.bucket.node(n.inodes[index].pgid, n) +} + +// childIndex returns the index of a given child node. +func (n *node) childIndex(child *node) int { + index := sort.Search(len(n.inodes), func(i int) bool { return bytes.Compare(n.inodes[i].key, child.key) != -1 }) + return index +} + +// numChildren returns the number of children. +func (n *node) numChildren() int { + return len(n.inodes) +} + +// nextSibling returns the next node with the same parent. +func (n *node) nextSibling() *node { + if n.parent == nil { + return nil + } + index := n.parent.childIndex(n) + if index >= n.parent.numChildren()-1 { + return nil + } + return n.parent.childAt(index + 1) +} + +// prevSibling returns the previous node with the same parent. +func (n *node) prevSibling() *node { + if n.parent == nil { + return nil + } + index := n.parent.childIndex(n) + if index == 0 { + return nil + } + return n.parent.childAt(index - 1) +} + +// put inserts a key/value. +func (n *node) put(oldKey, newKey, value []byte, pgid pgid, flags uint32) { + if pgid >= n.bucket.tx.meta.pgid { + panic(fmt.Sprintf("pgid (%d) above high water mark (%d)", pgid, n.bucket.tx.meta.pgid)) + } else if len(oldKey) <= 0 { + panic("put: zero-length old key") + } else if len(newKey) <= 0 { + panic("put: zero-length new key") + } + + // Find insertion index. + index := sort.Search(len(n.inodes), func(i int) bool { return bytes.Compare(n.inodes[i].key, oldKey) != -1 }) + + // Add capacity and shift nodes if we don't have an exact match and need to insert. + exact := (len(n.inodes) > 0 && index < len(n.inodes) && bytes.Equal(n.inodes[index].key, oldKey)) + if !exact { + n.inodes = append(n.inodes, inode{}) + copy(n.inodes[index+1:], n.inodes[index:]) + } + + inode := &n.inodes[index] + inode.flags = flags + inode.key = newKey + inode.value = value + inode.pgid = pgid + _assert(len(inode.key) > 0, "put: zero-length inode key") +} + +// del removes a key from the node. +func (n *node) del(key []byte) { + // Find index of key. + index := sort.Search(len(n.inodes), func(i int) bool { return bytes.Compare(n.inodes[i].key, key) != -1 }) + + // Exit if the key isn't found. + if index >= len(n.inodes) || !bytes.Equal(n.inodes[index].key, key) { + return + } + + // Delete inode from the node. + n.inodes = append(n.inodes[:index], n.inodes[index+1:]...) + + // Mark the node as needing rebalancing. + n.unbalanced = true +} + +// read initializes the node from a page. +func (n *node) read(p *page) { + n.pgid = p.id + n.isLeaf = ((p.flags & leafPageFlag) != 0) + n.inodes = make(inodes, int(p.count)) + + for i := 0; i < int(p.count); i++ { + inode := &n.inodes[i] + if n.isLeaf { + elem := p.leafPageElement(uint16(i)) + inode.flags = elem.flags + inode.key = elem.key() + inode.value = elem.value() + } else { + elem := p.branchPageElement(uint16(i)) + inode.pgid = elem.pgid + inode.key = elem.key() + } + _assert(len(inode.key) > 0, "read: zero-length inode key") + } + + // Save first key so we can find the node in the parent when we spill. + if len(n.inodes) > 0 { + n.key = n.inodes[0].key + _assert(len(n.key) > 0, "read: zero-length node key") + } else { + n.key = nil + } +} + +// write writes the items onto one or more pages. +func (n *node) write(p *page) { + // Initialize page. + if n.isLeaf { + p.flags |= leafPageFlag + } else { + p.flags |= branchPageFlag + } + + if len(n.inodes) >= 0xFFFF { + panic(fmt.Sprintf("inode overflow: %d (pgid=%d)", len(n.inodes), p.id)) + } + p.count = uint16(len(n.inodes)) + + // Loop over each item and write it to the page. + b := (*[maxAllocSize]byte)(unsafe.Pointer(&p.ptr))[n.pageElementSize()*len(n.inodes):] + for i, item := range n.inodes { + _assert(len(item.key) > 0, "write: zero-length inode key") + + // Write the page element. + if n.isLeaf { + elem := p.leafPageElement(uint16(i)) + elem.pos = uint32(uintptr(unsafe.Pointer(&b[0])) - uintptr(unsafe.Pointer(elem))) + elem.flags = item.flags + elem.ksize = uint32(len(item.key)) + elem.vsize = uint32(len(item.value)) + } else { + elem := p.branchPageElement(uint16(i)) + elem.pos = uint32(uintptr(unsafe.Pointer(&b[0])) - uintptr(unsafe.Pointer(elem))) + elem.ksize = uint32(len(item.key)) + elem.pgid = item.pgid + _assert(elem.pgid != p.id, "write: circular dependency occurred") + } + + // If the length of key+value is larger than the max allocation size + // then we need to reallocate the byte array pointer. + // + // See: https://github.com/boltdb/bolt/pull/335 + klen, vlen := len(item.key), len(item.value) + if len(b) < klen+vlen { + b = (*[maxAllocSize]byte)(unsafe.Pointer(&b[0]))[:] + } + + // Write data for the element to the end of the page. + copy(b[0:], item.key) + b = b[klen:] + copy(b[0:], item.value) + b = b[vlen:] + } + + // DEBUG ONLY: n.dump() +} + +// split breaks up a node into multiple smaller nodes, if appropriate. +// This should only be called from the spill() function. +func (n *node) split(pageSize int) []*node { + var nodes []*node + + node := n + for { + // Split node into two. + a, b := node.splitTwo(pageSize) + nodes = append(nodes, a) + + // If we can't split then exit the loop. + if b == nil { + break + } + + // Set node to b so it gets split on the next iteration. + node = b + } + + return nodes +} + +// splitTwo breaks up a node into two smaller nodes, if appropriate. +// This should only be called from the split() function. +func (n *node) splitTwo(pageSize int) (*node, *node) { + // Ignore the split if the page doesn't have at least enough nodes for + // two pages or if the nodes can fit in a single page. + if len(n.inodes) <= (minKeysPerPage*2) || n.sizeLessThan(pageSize) { + return n, nil + } + + // Determine the threshold before starting a new node. + var fillPercent = n.bucket.FillPercent + if fillPercent < minFillPercent { + fillPercent = minFillPercent + } else if fillPercent > maxFillPercent { + fillPercent = maxFillPercent + } + threshold := int(float64(pageSize) * fillPercent) + + // Determine split position and sizes of the two pages. + splitIndex, _ := n.splitIndex(threshold) + + // Split node into two separate nodes. + // If there's no parent then we'll need to create one. + if n.parent == nil { + n.parent = &node{bucket: n.bucket, children: []*node{n}} + } + + // Create a new node and add it to the parent. + next := &node{bucket: n.bucket, isLeaf: n.isLeaf, parent: n.parent} + n.parent.children = append(n.parent.children, next) + + // Split inodes across two nodes. + next.inodes = n.inodes[splitIndex:] + n.inodes = n.inodes[:splitIndex] + + // Update the statistics. + n.bucket.tx.stats.Split++ + + return n, next +} + +// splitIndex finds the position where a page will fill a given threshold. +// It returns the index as well as the size of the first page. +// This is only be called from split(). +func (n *node) splitIndex(threshold int) (index, sz int) { + sz = pageHeaderSize + + // Loop until we only have the minimum number of keys required for the second page. + for i := 0; i < len(n.inodes)-minKeysPerPage; i++ { + index = i + inode := n.inodes[i] + elsize := n.pageElementSize() + len(inode.key) + len(inode.value) + + // If we have at least the minimum number of keys and adding another + // node would put us over the threshold then exit and return. + if i >= minKeysPerPage && sz+elsize > threshold { + break + } + + // Add the element size to the total size. + sz += elsize + } + + return +} + +// spill writes the nodes to dirty pages and splits nodes as it goes. +// Returns an error if dirty pages cannot be allocated. +func (n *node) spill() error { + var tx = n.bucket.tx + if n.spilled { + return nil + } + + // Spill child nodes first. Child nodes can materialize sibling nodes in + // the case of split-merge so we cannot use a range loop. We have to check + // the children size on every loop iteration. + sort.Sort(n.children) + for i := 0; i < len(n.children); i++ { + if err := n.children[i].spill(); err != nil { + return err + } + } + + // We no longer need the child list because it's only used for spill tracking. + n.children = nil + + // Split nodes into appropriate sizes. The first node will always be n. + var nodes = n.split(tx.db.pageSize) + for _, node := range nodes { + // Add node's page to the freelist if it's not new. + if node.pgid > 0 { + tx.db.freelist.free(tx.meta.txid, tx.page(node.pgid)) + node.pgid = 0 + } + + // Allocate contiguous space for the node. + p, err := tx.allocate((node.size() / tx.db.pageSize) + 1) + if err != nil { + return err + } + + // Write the node. + if p.id >= tx.meta.pgid { + panic(fmt.Sprintf("pgid (%d) above high water mark (%d)", p.id, tx.meta.pgid)) + } + node.pgid = p.id + node.write(p) + node.spilled = true + + // Insert into parent inodes. + if node.parent != nil { + var key = node.key + if key == nil { + key = node.inodes[0].key + } + + node.parent.put(key, node.inodes[0].key, nil, node.pgid, 0) + node.key = node.inodes[0].key + _assert(len(node.key) > 0, "spill: zero-length node key") + } + + // Update the statistics. + tx.stats.Spill++ + } + + // If the root node split and created a new root then we need to spill that + // as well. We'll clear out the children to make sure it doesn't try to respill. + if n.parent != nil && n.parent.pgid == 0 { + n.children = nil + return n.parent.spill() + } + + return nil +} + +// rebalance attempts to combine the node with sibling nodes if the node fill +// size is below a threshold or if there are not enough keys. +func (n *node) rebalance() { + if !n.unbalanced { + return + } + n.unbalanced = false + + // Update statistics. + n.bucket.tx.stats.Rebalance++ + + // Ignore if node is above threshold (25%) and has enough keys. + var threshold = n.bucket.tx.db.pageSize / 4 + if n.size() > threshold && len(n.inodes) > n.minKeys() { + return + } + + // Root node has special handling. + if n.parent == nil { + // If root node is a branch and only has one node then collapse it. + if !n.isLeaf && len(n.inodes) == 1 { + // Move root's child up. + child := n.bucket.node(n.inodes[0].pgid, n) + n.isLeaf = child.isLeaf + n.inodes = child.inodes[:] + n.children = child.children + + // Reparent all child nodes being moved. + for _, inode := range n.inodes { + if child, ok := n.bucket.nodes[inode.pgid]; ok { + child.parent = n + } + } + + // Remove old child. + child.parent = nil + delete(n.bucket.nodes, child.pgid) + child.free() + } + + return + } + + // If node has no keys then just remove it. + if n.numChildren() == 0 { + n.parent.del(n.key) + n.parent.removeChild(n) + delete(n.bucket.nodes, n.pgid) + n.free() + n.parent.rebalance() + return + } + + _assert(n.parent.numChildren() > 1, "parent must have at least 2 children") + + // Destination node is right sibling if idx == 0, otherwise left sibling. + var target *node + var useNextSibling = (n.parent.childIndex(n) == 0) + if useNextSibling { + target = n.nextSibling() + } else { + target = n.prevSibling() + } + + // If target node has extra nodes then just move one over. + if target.numChildren() > target.minKeys() { + if useNextSibling { + // Reparent and move node. + if child, ok := n.bucket.nodes[target.inodes[0].pgid]; ok { + child.parent.removeChild(child) + child.parent = n + child.parent.children = append(child.parent.children, child) + } + n.inodes = append(n.inodes, target.inodes[0]) + target.inodes = target.inodes[1:] + + // Update target key on parent. + target.parent.put(target.key, target.inodes[0].key, nil, target.pgid, 0) + target.key = target.inodes[0].key + _assert(len(target.key) > 0, "rebalance(1): zero-length node key") + } else { + // Reparent and move node. + if child, ok := n.bucket.nodes[target.inodes[len(target.inodes)-1].pgid]; ok { + child.parent.removeChild(child) + child.parent = n + child.parent.children = append(child.parent.children, child) + } + n.inodes = append(n.inodes, inode{}) + copy(n.inodes[1:], n.inodes) + n.inodes[0] = target.inodes[len(target.inodes)-1] + target.inodes = target.inodes[:len(target.inodes)-1] + } + + // Update parent key for node. + n.parent.put(n.key, n.inodes[0].key, nil, n.pgid, 0) + n.key = n.inodes[0].key + _assert(len(n.key) > 0, "rebalance(2): zero-length node key") + + return + } + + // If both this node and the target node are too small then merge them. + if useNextSibling { + // Reparent all child nodes being moved. + for _, inode := range target.inodes { + if child, ok := n.bucket.nodes[inode.pgid]; ok { + child.parent.removeChild(child) + child.parent = n + child.parent.children = append(child.parent.children, child) + } + } + + // Copy over inodes from target and remove target. + n.inodes = append(n.inodes, target.inodes...) + n.parent.del(target.key) + n.parent.removeChild(target) + delete(n.bucket.nodes, target.pgid) + target.free() + } else { + // Reparent all child nodes being moved. + for _, inode := range n.inodes { + if child, ok := n.bucket.nodes[inode.pgid]; ok { + child.parent.removeChild(child) + child.parent = target + child.parent.children = append(child.parent.children, child) + } + } + + // Copy over inodes to target and remove node. + target.inodes = append(target.inodes, n.inodes...) + n.parent.del(n.key) + n.parent.removeChild(n) + delete(n.bucket.nodes, n.pgid) + n.free() + } + + // Either this node or the target node was deleted from the parent so rebalance it. + n.parent.rebalance() +} + +// removes a node from the list of in-memory children. +// This does not affect the inodes. +func (n *node) removeChild(target *node) { + for i, child := range n.children { + if child == target { + n.children = append(n.children[:i], n.children[i+1:]...) + return + } + } +} + +// dereference causes the node to copy all its inode key/value references to heap memory. +// This is required when the mmap is reallocated so inodes are not pointing to stale data. +func (n *node) dereference() { + if n.key != nil { + key := make([]byte, len(n.key)) + copy(key, n.key) + n.key = key + _assert(n.pgid == 0 || len(n.key) > 0, "dereference: zero-length node key on existing node") + } + + for i := range n.inodes { + inode := &n.inodes[i] + + key := make([]byte, len(inode.key)) + copy(key, inode.key) + inode.key = key + _assert(len(inode.key) > 0, "dereference: zero-length inode key") + + value := make([]byte, len(inode.value)) + copy(value, inode.value) + inode.value = value + } + + // Recursively dereference children. + for _, child := range n.children { + child.dereference() + } + + // Update statistics. + n.bucket.tx.stats.NodeDeref++ +} + +// free adds the node's underlying page to the freelist. +func (n *node) free() { + if n.pgid != 0 { + n.bucket.tx.db.freelist.free(n.bucket.tx.meta.txid, n.bucket.tx.page(n.pgid)) + n.pgid = 0 + } +} + +// dump writes the contents of the node to STDERR for debugging purposes. +/* +func (n *node) dump() { + // Write node header. + var typ = "branch" + if n.isLeaf { + typ = "leaf" + } + warnf("[NODE %d {type=%s count=%d}]", n.pgid, typ, len(n.inodes)) + + // Write out abbreviated version of each item. + for _, item := range n.inodes { + if n.isLeaf { + if item.flags&bucketLeafFlag != 0 { + bucket := (*bucket)(unsafe.Pointer(&item.value[0])) + warnf("+L %08x -> (bucket root=%d)", trunc(item.key, 4), bucket.root) + } else { + warnf("+L %08x -> %08x", trunc(item.key, 4), trunc(item.value, 4)) + } + } else { + warnf("+B %08x -> pgid=%d", trunc(item.key, 4), item.pgid) + } + } + warn("") +} +*/ + +type nodes []*node + +func (s nodes) Len() int { return len(s) } +func (s nodes) Swap(i, j int) { s[i], s[j] = s[j], s[i] } +func (s nodes) Less(i, j int) bool { return bytes.Compare(s[i].inodes[0].key, s[j].inodes[0].key) == -1 } + +// inode represents an internal node inside of a node. +// It can be used to point to elements in a page or point +// to an element which hasn't been added to a page yet. +type inode struct { + flags uint32 + pgid pgid + key []byte + value []byte +} + +type inodes []inode diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/node_test.go b/Godeps/_workspace/src/github.com/boltdb/bolt/node_test.go new file mode 100644 index 0000000000000..fa5d10f9990ec --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/node_test.go @@ -0,0 +1,156 @@ +package bolt + +import ( + "testing" + "unsafe" +) + +// Ensure that a node can insert a key/value. +func TestNode_put(t *testing.T) { + n := &node{inodes: make(inodes, 0), bucket: &Bucket{tx: &Tx{meta: &meta{pgid: 1}}}} + n.put([]byte("baz"), []byte("baz"), []byte("2"), 0, 0) + n.put([]byte("foo"), []byte("foo"), []byte("0"), 0, 0) + n.put([]byte("bar"), []byte("bar"), []byte("1"), 0, 0) + n.put([]byte("foo"), []byte("foo"), []byte("3"), 0, leafPageFlag) + + if len(n.inodes) != 3 { + t.Fatalf("exp=3; got=%d", len(n.inodes)) + } + if k, v := n.inodes[0].key, n.inodes[0].value; string(k) != "bar" || string(v) != "1" { + t.Fatalf("exp=; got=<%s,%s>", k, v) + } + if k, v := n.inodes[1].key, n.inodes[1].value; string(k) != "baz" || string(v) != "2" { + t.Fatalf("exp=; got=<%s,%s>", k, v) + } + if k, v := n.inodes[2].key, n.inodes[2].value; string(k) != "foo" || string(v) != "3" { + t.Fatalf("exp=; got=<%s,%s>", k, v) + } + if n.inodes[2].flags != uint32(leafPageFlag) { + t.Fatalf("not a leaf: %d", n.inodes[2].flags) + } +} + +// Ensure that a node can deserialize from a leaf page. +func TestNode_read_LeafPage(t *testing.T) { + // Create a page. + var buf [4096]byte + page := (*page)(unsafe.Pointer(&buf[0])) + page.flags = leafPageFlag + page.count = 2 + + // Insert 2 elements at the beginning. sizeof(leafPageElement) == 16 + nodes := (*[3]leafPageElement)(unsafe.Pointer(&page.ptr)) + nodes[0] = leafPageElement{flags: 0, pos: 32, ksize: 3, vsize: 4} // pos = sizeof(leafPageElement) * 2 + nodes[1] = leafPageElement{flags: 0, pos: 23, ksize: 10, vsize: 3} // pos = sizeof(leafPageElement) + 3 + 4 + + // Write data for the nodes at the end. + data := (*[4096]byte)(unsafe.Pointer(&nodes[2])) + copy(data[:], []byte("barfooz")) + copy(data[7:], []byte("helloworldbye")) + + // Deserialize page into a leaf. + n := &node{} + n.read(page) + + // Check that there are two inodes with correct data. + if !n.isLeaf { + t.Fatal("expected leaf") + } + if len(n.inodes) != 2 { + t.Fatalf("exp=2; got=%d", len(n.inodes)) + } + if k, v := n.inodes[0].key, n.inodes[0].value; string(k) != "bar" || string(v) != "fooz" { + t.Fatalf("exp=; got=<%s,%s>", k, v) + } + if k, v := n.inodes[1].key, n.inodes[1].value; string(k) != "helloworld" || string(v) != "bye" { + t.Fatalf("exp=; got=<%s,%s>", k, v) + } +} + +// Ensure that a node can serialize into a leaf page. +func TestNode_write_LeafPage(t *testing.T) { + // Create a node. + n := &node{isLeaf: true, inodes: make(inodes, 0), bucket: &Bucket{tx: &Tx{db: &DB{}, meta: &meta{pgid: 1}}}} + n.put([]byte("susy"), []byte("susy"), []byte("que"), 0, 0) + n.put([]byte("ricki"), []byte("ricki"), []byte("lake"), 0, 0) + n.put([]byte("john"), []byte("john"), []byte("johnson"), 0, 0) + + // Write it to a page. + var buf [4096]byte + p := (*page)(unsafe.Pointer(&buf[0])) + n.write(p) + + // Read the page back in. + n2 := &node{} + n2.read(p) + + // Check that the two pages are the same. + if len(n2.inodes) != 3 { + t.Fatalf("exp=3; got=%d", len(n2.inodes)) + } + if k, v := n2.inodes[0].key, n2.inodes[0].value; string(k) != "john" || string(v) != "johnson" { + t.Fatalf("exp=; got=<%s,%s>", k, v) + } + if k, v := n2.inodes[1].key, n2.inodes[1].value; string(k) != "ricki" || string(v) != "lake" { + t.Fatalf("exp=; got=<%s,%s>", k, v) + } + if k, v := n2.inodes[2].key, n2.inodes[2].value; string(k) != "susy" || string(v) != "que" { + t.Fatalf("exp=; got=<%s,%s>", k, v) + } +} + +// Ensure that a node can split into appropriate subgroups. +func TestNode_split(t *testing.T) { + // Create a node. + n := &node{inodes: make(inodes, 0), bucket: &Bucket{tx: &Tx{db: &DB{}, meta: &meta{pgid: 1}}}} + n.put([]byte("00000001"), []byte("00000001"), []byte("0123456701234567"), 0, 0) + n.put([]byte("00000002"), []byte("00000002"), []byte("0123456701234567"), 0, 0) + n.put([]byte("00000003"), []byte("00000003"), []byte("0123456701234567"), 0, 0) + n.put([]byte("00000004"), []byte("00000004"), []byte("0123456701234567"), 0, 0) + n.put([]byte("00000005"), []byte("00000005"), []byte("0123456701234567"), 0, 0) + + // Split between 2 & 3. + n.split(100) + + var parent = n.parent + if len(parent.children) != 2 { + t.Fatalf("exp=2; got=%d", len(parent.children)) + } + if len(parent.children[0].inodes) != 2 { + t.Fatalf("exp=2; got=%d", len(parent.children[0].inodes)) + } + if len(parent.children[1].inodes) != 3 { + t.Fatalf("exp=3; got=%d", len(parent.children[1].inodes)) + } +} + +// Ensure that a page with the minimum number of inodes just returns a single node. +func TestNode_split_MinKeys(t *testing.T) { + // Create a node. + n := &node{inodes: make(inodes, 0), bucket: &Bucket{tx: &Tx{db: &DB{}, meta: &meta{pgid: 1}}}} + n.put([]byte("00000001"), []byte("00000001"), []byte("0123456701234567"), 0, 0) + n.put([]byte("00000002"), []byte("00000002"), []byte("0123456701234567"), 0, 0) + + // Split. + n.split(20) + if n.parent != nil { + t.Fatalf("expected nil parent") + } +} + +// Ensure that a node that has keys that all fit on a page just returns one leaf. +func TestNode_split_SinglePage(t *testing.T) { + // Create a node. + n := &node{inodes: make(inodes, 0), bucket: &Bucket{tx: &Tx{db: &DB{}, meta: &meta{pgid: 1}}}} + n.put([]byte("00000001"), []byte("00000001"), []byte("0123456701234567"), 0, 0) + n.put([]byte("00000002"), []byte("00000002"), []byte("0123456701234567"), 0, 0) + n.put([]byte("00000003"), []byte("00000003"), []byte("0123456701234567"), 0, 0) + n.put([]byte("00000004"), []byte("00000004"), []byte("0123456701234567"), 0, 0) + n.put([]byte("00000005"), []byte("00000005"), []byte("0123456701234567"), 0, 0) + + // Split. + n.split(4096) + if n.parent != nil { + t.Fatalf("expected nil parent") + } +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/page.go b/Godeps/_workspace/src/github.com/boltdb/bolt/page.go new file mode 100644 index 0000000000000..818aa1b1531de --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/page.go @@ -0,0 +1,172 @@ +package bolt + +import ( + "fmt" + "os" + "sort" + "unsafe" +) + +const pageHeaderSize = int(unsafe.Offsetof(((*page)(nil)).ptr)) + +const minKeysPerPage = 2 + +const branchPageElementSize = int(unsafe.Sizeof(branchPageElement{})) +const leafPageElementSize = int(unsafe.Sizeof(leafPageElement{})) + +const ( + branchPageFlag = 0x01 + leafPageFlag = 0x02 + metaPageFlag = 0x04 + freelistPageFlag = 0x10 +) + +const ( + bucketLeafFlag = 0x01 +) + +type pgid uint64 + +type page struct { + id pgid + flags uint16 + count uint16 + overflow uint32 + ptr uintptr +} + +// typ returns a human readable page type string used for debugging. +func (p *page) typ() string { + if (p.flags & branchPageFlag) != 0 { + return "branch" + } else if (p.flags & leafPageFlag) != 0 { + return "leaf" + } else if (p.flags & metaPageFlag) != 0 { + return "meta" + } else if (p.flags & freelistPageFlag) != 0 { + return "freelist" + } + return fmt.Sprintf("unknown<%02x>", p.flags) +} + +// meta returns a pointer to the metadata section of the page. +func (p *page) meta() *meta { + return (*meta)(unsafe.Pointer(&p.ptr)) +} + +// leafPageElement retrieves the leaf node by index +func (p *page) leafPageElement(index uint16) *leafPageElement { + n := &((*[0x7FFFFFF]leafPageElement)(unsafe.Pointer(&p.ptr)))[index] + return n +} + +// leafPageElements retrieves a list of leaf nodes. +func (p *page) leafPageElements() []leafPageElement { + return ((*[0x7FFFFFF]leafPageElement)(unsafe.Pointer(&p.ptr)))[:] +} + +// branchPageElement retrieves the branch node by index +func (p *page) branchPageElement(index uint16) *branchPageElement { + return &((*[0x7FFFFFF]branchPageElement)(unsafe.Pointer(&p.ptr)))[index] +} + +// branchPageElements retrieves a list of branch nodes. +func (p *page) branchPageElements() []branchPageElement { + return ((*[0x7FFFFFF]branchPageElement)(unsafe.Pointer(&p.ptr)))[:] +} + +// dump writes n bytes of the page to STDERR as hex output. +func (p *page) hexdump(n int) { + buf := (*[maxAllocSize]byte)(unsafe.Pointer(p))[:n] + fmt.Fprintf(os.Stderr, "%x\n", buf) +} + +type pages []*page + +func (s pages) Len() int { return len(s) } +func (s pages) Swap(i, j int) { s[i], s[j] = s[j], s[i] } +func (s pages) Less(i, j int) bool { return s[i].id < s[j].id } + +// branchPageElement represents a node on a branch page. +type branchPageElement struct { + pos uint32 + ksize uint32 + pgid pgid +} + +// key returns a byte slice of the node key. +func (n *branchPageElement) key() []byte { + buf := (*[maxAllocSize]byte)(unsafe.Pointer(n)) + return (*[maxAllocSize]byte)(unsafe.Pointer(&buf[n.pos]))[:n.ksize] +} + +// leafPageElement represents a node on a leaf page. +type leafPageElement struct { + flags uint32 + pos uint32 + ksize uint32 + vsize uint32 +} + +// key returns a byte slice of the node key. +func (n *leafPageElement) key() []byte { + buf := (*[maxAllocSize]byte)(unsafe.Pointer(n)) + return (*[maxAllocSize]byte)(unsafe.Pointer(&buf[n.pos]))[:n.ksize] +} + +// value returns a byte slice of the node value. +func (n *leafPageElement) value() []byte { + buf := (*[maxAllocSize]byte)(unsafe.Pointer(n)) + return (*[maxAllocSize]byte)(unsafe.Pointer(&buf[n.pos+n.ksize]))[:n.vsize] +} + +// PageInfo represents human readable information about a page. +type PageInfo struct { + ID int + Type string + Count int + OverflowCount int +} + +type pgids []pgid + +func (s pgids) Len() int { return len(s) } +func (s pgids) Swap(i, j int) { s[i], s[j] = s[j], s[i] } +func (s pgids) Less(i, j int) bool { return s[i] < s[j] } + +// merge returns the sorted union of a and b. +func (a pgids) merge(b pgids) pgids { + // Return the opposite slice if one is nil. + if len(a) == 0 { + return b + } else if len(b) == 0 { + return a + } + + // Create a list to hold all elements from both lists. + merged := make(pgids, 0, len(a)+len(b)) + + // Assign lead to the slice with a lower starting value, follow to the higher value. + lead, follow := a, b + if b[0] < a[0] { + lead, follow = b, a + } + + // Continue while there are elements in the lead. + for len(lead) > 0 { + // Merge largest prefix of lead that is ahead of follow[0]. + n := sort.Search(len(lead), func(i int) bool { return lead[i] > follow[0] }) + merged = append(merged, lead[:n]...) + if n >= len(lead) { + break + } + + // Swap lead and follow. + lead, follow = follow, lead[n:] + } + + // Append what's left in follow. + merged = append(merged, follow...) + + return merged +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/page_test.go b/Godeps/_workspace/src/github.com/boltdb/bolt/page_test.go new file mode 100644 index 0000000000000..59f4a30ed8350 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/page_test.go @@ -0,0 +1,72 @@ +package bolt + +import ( + "reflect" + "sort" + "testing" + "testing/quick" +) + +// Ensure that the page type can be returned in human readable format. +func TestPage_typ(t *testing.T) { + if typ := (&page{flags: branchPageFlag}).typ(); typ != "branch" { + t.Fatalf("exp=branch; got=%v", typ) + } + if typ := (&page{flags: leafPageFlag}).typ(); typ != "leaf" { + t.Fatalf("exp=leaf; got=%v", typ) + } + if typ := (&page{flags: metaPageFlag}).typ(); typ != "meta" { + t.Fatalf("exp=meta; got=%v", typ) + } + if typ := (&page{flags: freelistPageFlag}).typ(); typ != "freelist" { + t.Fatalf("exp=freelist; got=%v", typ) + } + if typ := (&page{flags: 20000}).typ(); typ != "unknown<4e20>" { + t.Fatalf("exp=unknown<4e20>; got=%v", typ) + } +} + +// Ensure that the hexdump debugging function doesn't blow up. +func TestPage_dump(t *testing.T) { + (&page{id: 256}).hexdump(16) +} + +func TestPgids_merge(t *testing.T) { + a := pgids{4, 5, 6, 10, 11, 12, 13, 27} + b := pgids{1, 3, 8, 9, 25, 30} + c := a.merge(b) + if !reflect.DeepEqual(c, pgids{1, 3, 4, 5, 6, 8, 9, 10, 11, 12, 13, 25, 27, 30}) { + t.Errorf("mismatch: %v", c) + } + + a = pgids{4, 5, 6, 10, 11, 12, 13, 27, 35, 36} + b = pgids{8, 9, 25, 30} + c = a.merge(b) + if !reflect.DeepEqual(c, pgids{4, 5, 6, 8, 9, 10, 11, 12, 13, 25, 27, 30, 35, 36}) { + t.Errorf("mismatch: %v", c) + } +} + +func TestPgids_merge_quick(t *testing.T) { + if err := quick.Check(func(a, b pgids) bool { + // Sort incoming lists. + sort.Sort(a) + sort.Sort(b) + + // Merge the two lists together. + got := a.merge(b) + + // The expected value should be the two lists combined and sorted. + exp := append(a, b...) + sort.Sort(exp) + + if !reflect.DeepEqual(exp, got) { + t.Errorf("\nexp=%+v\ngot=%+v\n", exp, got) + return false + } + + return true + }, nil); err != nil { + t.Fatal(err) + } +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/quick_test.go b/Godeps/_workspace/src/github.com/boltdb/bolt/quick_test.go new file mode 100644 index 0000000000000..4da581775a351 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/quick_test.go @@ -0,0 +1,79 @@ +package bolt_test + +import ( + "bytes" + "flag" + "fmt" + "math/rand" + "os" + "reflect" + "testing/quick" + "time" +) + +// testing/quick defaults to 5 iterations and a random seed. +// You can override these settings from the command line: +// +// -quick.count The number of iterations to perform. +// -quick.seed The seed to use for randomizing. +// -quick.maxitems The maximum number of items to insert into a DB. +// -quick.maxksize The maximum size of a key. +// -quick.maxvsize The maximum size of a value. +// + +var qcount, qseed, qmaxitems, qmaxksize, qmaxvsize int + +func init() { + flag.IntVar(&qcount, "quick.count", 5, "") + flag.IntVar(&qseed, "quick.seed", int(time.Now().UnixNano())%100000, "") + flag.IntVar(&qmaxitems, "quick.maxitems", 1000, "") + flag.IntVar(&qmaxksize, "quick.maxksize", 1024, "") + flag.IntVar(&qmaxvsize, "quick.maxvsize", 1024, "") + flag.Parse() + fmt.Fprintln(os.Stderr, "seed:", qseed) + fmt.Fprintf(os.Stderr, "quick settings: count=%v, items=%v, ksize=%v, vsize=%v\n", qcount, qmaxitems, qmaxksize, qmaxvsize) +} + +func qconfig() *quick.Config { + return &quick.Config{ + MaxCount: qcount, + Rand: rand.New(rand.NewSource(int64(qseed))), + } +} + +type testdata []testdataitem + +func (t testdata) Len() int { return len(t) } +func (t testdata) Swap(i, j int) { t[i], t[j] = t[j], t[i] } +func (t testdata) Less(i, j int) bool { return bytes.Compare(t[i].Key, t[j].Key) == -1 } + +func (t testdata) Generate(rand *rand.Rand, size int) reflect.Value { + n := rand.Intn(qmaxitems-1) + 1 + items := make(testdata, n) + for i := 0; i < n; i++ { + item := &items[i] + item.Key = randByteSlice(rand, 1, qmaxksize) + item.Value = randByteSlice(rand, 0, qmaxvsize) + } + return reflect.ValueOf(items) +} + +type revtestdata []testdataitem + +func (t revtestdata) Len() int { return len(t) } +func (t revtestdata) Swap(i, j int) { t[i], t[j] = t[j], t[i] } +func (t revtestdata) Less(i, j int) bool { return bytes.Compare(t[i].Key, t[j].Key) == 1 } + +type testdataitem struct { + Key []byte + Value []byte +} + +func randByteSlice(rand *rand.Rand, minSize, maxSize int) []byte { + n := rand.Intn(maxSize-minSize) + minSize + b := make([]byte, n) + for i := 0; i < n; i++ { + b[i] = byte(rand.Intn(255)) + } + return b +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/simulation_test.go b/Godeps/_workspace/src/github.com/boltdb/bolt/simulation_test.go new file mode 100644 index 0000000000000..ceb8baef0c92a --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/simulation_test.go @@ -0,0 +1,327 @@ +package bolt_test + +import ( + "bytes" + "fmt" + "math/rand" + "sync" + "testing" + + "github.com/boltdb/bolt" +) + +func TestSimulate_1op_1p(t *testing.T) { testSimulate(t, 100, 1) } +func TestSimulate_10op_1p(t *testing.T) { testSimulate(t, 10, 1) } +func TestSimulate_100op_1p(t *testing.T) { testSimulate(t, 100, 1) } +func TestSimulate_1000op_1p(t *testing.T) { testSimulate(t, 1000, 1) } +func TestSimulate_10000op_1p(t *testing.T) { testSimulate(t, 10000, 1) } + +func TestSimulate_10op_10p(t *testing.T) { testSimulate(t, 10, 10) } +func TestSimulate_100op_10p(t *testing.T) { testSimulate(t, 100, 10) } +func TestSimulate_1000op_10p(t *testing.T) { testSimulate(t, 1000, 10) } +func TestSimulate_10000op_10p(t *testing.T) { testSimulate(t, 10000, 10) } + +func TestSimulate_100op_100p(t *testing.T) { testSimulate(t, 100, 100) } +func TestSimulate_1000op_100p(t *testing.T) { testSimulate(t, 1000, 100) } +func TestSimulate_10000op_100p(t *testing.T) { testSimulate(t, 10000, 100) } + +func TestSimulate_10000op_1000p(t *testing.T) { testSimulate(t, 10000, 1000) } + +// Randomly generate operations on a given database with multiple clients to ensure consistency and thread safety. +func testSimulate(t *testing.T, threadCount, parallelism int) { + if testing.Short() { + t.Skip("skipping test in short mode.") + } + + rand.Seed(int64(qseed)) + + // A list of operations that readers and writers can perform. + var readerHandlers = []simulateHandler{simulateGetHandler} + var writerHandlers = []simulateHandler{simulateGetHandler, simulatePutHandler} + + var versions = make(map[int]*QuickDB) + versions[1] = NewQuickDB() + + db := NewTestDB() + defer db.Close() + + var mutex sync.Mutex + + // Run n threads in parallel, each with their own operation. + var wg sync.WaitGroup + var threads = make(chan bool, parallelism) + var i int + for { + threads <- true + wg.Add(1) + writable := ((rand.Int() % 100) < 20) // 20% writers + + // Choose an operation to execute. + var handler simulateHandler + if writable { + handler = writerHandlers[rand.Intn(len(writerHandlers))] + } else { + handler = readerHandlers[rand.Intn(len(readerHandlers))] + } + + // Execute a thread for the given operation. + go func(writable bool, handler simulateHandler) { + defer wg.Done() + + // Start transaction. + tx, err := db.Begin(writable) + if err != nil { + t.Fatal("tx begin: ", err) + } + + // Obtain current state of the dataset. + mutex.Lock() + var qdb = versions[tx.ID()] + if writable { + qdb = versions[tx.ID()-1].Copy() + } + mutex.Unlock() + + // Make sure we commit/rollback the tx at the end and update the state. + if writable { + defer func() { + mutex.Lock() + versions[tx.ID()] = qdb + mutex.Unlock() + + ok(t, tx.Commit()) + }() + } else { + defer tx.Rollback() + } + + // Ignore operation if we don't have data yet. + if qdb == nil { + return + } + + // Execute handler. + handler(tx, qdb) + + // Release a thread back to the scheduling loop. + <-threads + }(writable, handler) + + i++ + if i > threadCount { + break + } + } + + // Wait until all threads are done. + wg.Wait() +} + +type simulateHandler func(tx *bolt.Tx, qdb *QuickDB) + +// Retrieves a key from the database and verifies that it is what is expected. +func simulateGetHandler(tx *bolt.Tx, qdb *QuickDB) { + // Randomly retrieve an existing exist. + keys := qdb.Rand() + if len(keys) == 0 { + return + } + + // Retrieve root bucket. + b := tx.Bucket(keys[0]) + if b == nil { + panic(fmt.Sprintf("bucket[0] expected: %08x\n", trunc(keys[0], 4))) + } + + // Drill into nested buckets. + for _, key := range keys[1 : len(keys)-1] { + b = b.Bucket(key) + if b == nil { + panic(fmt.Sprintf("bucket[n] expected: %v -> %v\n", keys, key)) + } + } + + // Verify key/value on the final bucket. + expected := qdb.Get(keys) + actual := b.Get(keys[len(keys)-1]) + if !bytes.Equal(actual, expected) { + fmt.Println("=== EXPECTED ===") + fmt.Println(expected) + fmt.Println("=== ACTUAL ===") + fmt.Println(actual) + fmt.Println("=== END ===") + panic("value mismatch") + } +} + +// Inserts a key into the database. +func simulatePutHandler(tx *bolt.Tx, qdb *QuickDB) { + var err error + keys, value := randKeys(), randValue() + + // Retrieve root bucket. + b := tx.Bucket(keys[0]) + if b == nil { + b, err = tx.CreateBucket(keys[0]) + if err != nil { + panic("create bucket: " + err.Error()) + } + } + + // Create nested buckets, if necessary. + for _, key := range keys[1 : len(keys)-1] { + child := b.Bucket(key) + if child != nil { + b = child + } else { + b, err = b.CreateBucket(key) + if err != nil { + panic("create bucket: " + err.Error()) + } + } + } + + // Insert into database. + if err := b.Put(keys[len(keys)-1], value); err != nil { + panic("put: " + err.Error()) + } + + // Insert into in-memory database. + qdb.Put(keys, value) +} + +// QuickDB is an in-memory database that replicates the functionality of the +// Bolt DB type except that it is entirely in-memory. It is meant for testing +// that the Bolt database is consistent. +type QuickDB struct { + sync.RWMutex + m map[string]interface{} +} + +// NewQuickDB returns an instance of QuickDB. +func NewQuickDB() *QuickDB { + return &QuickDB{m: make(map[string]interface{})} +} + +// Get retrieves the value at a key path. +func (db *QuickDB) Get(keys [][]byte) []byte { + db.RLock() + defer db.RUnlock() + + m := db.m + for _, key := range keys[:len(keys)-1] { + value := m[string(key)] + if value == nil { + return nil + } + switch value := value.(type) { + case map[string]interface{}: + m = value + case []byte: + return nil + } + } + + // Only return if it's a simple value. + if value, ok := m[string(keys[len(keys)-1])].([]byte); ok { + return value + } + return nil +} + +// Put inserts a value into a key path. +func (db *QuickDB) Put(keys [][]byte, value []byte) { + db.Lock() + defer db.Unlock() + + // Build buckets all the way down the key path. + m := db.m + for _, key := range keys[:len(keys)-1] { + if _, ok := m[string(key)].([]byte); ok { + return // Keypath intersects with a simple value. Do nothing. + } + + if m[string(key)] == nil { + m[string(key)] = make(map[string]interface{}) + } + m = m[string(key)].(map[string]interface{}) + } + + // Insert value into the last key. + m[string(keys[len(keys)-1])] = value +} + +// Rand returns a random key path that points to a simple value. +func (db *QuickDB) Rand() [][]byte { + db.RLock() + defer db.RUnlock() + if len(db.m) == 0 { + return nil + } + var keys [][]byte + db.rand(db.m, &keys) + return keys +} + +func (db *QuickDB) rand(m map[string]interface{}, keys *[][]byte) { + i, index := 0, rand.Intn(len(m)) + for k, v := range m { + if i == index { + *keys = append(*keys, []byte(k)) + if v, ok := v.(map[string]interface{}); ok { + db.rand(v, keys) + } + return + } + i++ + } + panic("quickdb rand: out-of-range") +} + +// Copy copies the entire database. +func (db *QuickDB) Copy() *QuickDB { + db.RLock() + defer db.RUnlock() + return &QuickDB{m: db.copy(db.m)} +} + +func (db *QuickDB) copy(m map[string]interface{}) map[string]interface{} { + clone := make(map[string]interface{}, len(m)) + for k, v := range m { + switch v := v.(type) { + case map[string]interface{}: + clone[k] = db.copy(v) + default: + clone[k] = v + } + } + return clone +} + +func randKey() []byte { + var min, max = 1, 1024 + n := rand.Intn(max-min) + min + b := make([]byte, n) + for i := 0; i < n; i++ { + b[i] = byte(rand.Intn(255)) + } + return b +} + +func randKeys() [][]byte { + var keys [][]byte + var count = rand.Intn(2) + 2 + for i := 0; i < count; i++ { + keys = append(keys, randKey()) + } + return keys +} + +func randValue() []byte { + n := rand.Intn(8192) + b := make([]byte, n) + for i := 0; i < n; i++ { + b[i] = byte(rand.Intn(255)) + } + return b +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/tx.go b/Godeps/_workspace/src/github.com/boltdb/bolt/tx.go new file mode 100644 index 0000000000000..6b52b2c896452 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/tx.go @@ -0,0 +1,611 @@ +package bolt + +import ( + "fmt" + "io" + "os" + "sort" + "time" + "unsafe" +) + +// txid represents the internal transaction identifier. +type txid uint64 + +// Tx represents a read-only or read/write transaction on the database. +// Read-only transactions can be used for retrieving values for keys and creating cursors. +// Read/write transactions can create and remove buckets and create and remove keys. +// +// IMPORTANT: You must commit or rollback transactions when you are done with +// them. Pages can not be reclaimed by the writer until no more transactions +// are using them. A long running read transaction can cause the database to +// quickly grow. +type Tx struct { + writable bool + managed bool + db *DB + meta *meta + root Bucket + pages map[pgid]*page + stats TxStats + commitHandlers []func() +} + +// init initializes the transaction. +func (tx *Tx) init(db *DB) { + tx.db = db + tx.pages = nil + + // Copy the meta page since it can be changed by the writer. + tx.meta = &meta{} + db.meta().copy(tx.meta) + + // Copy over the root bucket. + tx.root = newBucket(tx) + tx.root.bucket = &bucket{} + *tx.root.bucket = tx.meta.root + + // Increment the transaction id and add a page cache for writable transactions. + if tx.writable { + tx.pages = make(map[pgid]*page) + tx.meta.txid += txid(1) + } +} + +// ID returns the transaction id. +func (tx *Tx) ID() int { + return int(tx.meta.txid) +} + +// DB returns a reference to the database that created the transaction. +func (tx *Tx) DB() *DB { + return tx.db +} + +// Size returns current database size in bytes as seen by this transaction. +func (tx *Tx) Size() int64 { + return int64(tx.meta.pgid) * int64(tx.db.pageSize) +} + +// Writable returns whether the transaction can perform write operations. +func (tx *Tx) Writable() bool { + return tx.writable +} + +// Cursor creates a cursor associated with the root bucket. +// All items in the cursor will return a nil value because all root bucket keys point to buckets. +// The cursor is only valid as long as the transaction is open. +// Do not use a cursor after the transaction is closed. +func (tx *Tx) Cursor() *Cursor { + return tx.root.Cursor() +} + +// Stats retrieves a copy of the current transaction statistics. +func (tx *Tx) Stats() TxStats { + return tx.stats +} + +// Bucket retrieves a bucket by name. +// Returns nil if the bucket does not exist. +func (tx *Tx) Bucket(name []byte) *Bucket { + return tx.root.Bucket(name) +} + +// CreateBucket creates a new bucket. +// Returns an error if the bucket already exists, if the bucket name is blank, or if the bucket name is too long. +func (tx *Tx) CreateBucket(name []byte) (*Bucket, error) { + return tx.root.CreateBucket(name) +} + +// CreateBucketIfNotExists creates a new bucket if it doesn't already exist. +// Returns an error if the bucket name is blank, or if the bucket name is too long. +func (tx *Tx) CreateBucketIfNotExists(name []byte) (*Bucket, error) { + return tx.root.CreateBucketIfNotExists(name) +} + +// DeleteBucket deletes a bucket. +// Returns an error if the bucket cannot be found or if the key represents a non-bucket value. +func (tx *Tx) DeleteBucket(name []byte) error { + return tx.root.DeleteBucket(name) +} + +// ForEach executes a function for each bucket in the root. +// If the provided function returns an error then the iteration is stopped and +// the error is returned to the caller. +func (tx *Tx) ForEach(fn func(name []byte, b *Bucket) error) error { + return tx.root.ForEach(func(k, v []byte) error { + if err := fn(k, tx.root.Bucket(k)); err != nil { + return err + } + return nil + }) +} + +// OnCommit adds a handler function to be executed after the transaction successfully commits. +func (tx *Tx) OnCommit(fn func()) { + tx.commitHandlers = append(tx.commitHandlers, fn) +} + +// Commit writes all changes to disk and updates the meta page. +// Returns an error if a disk write error occurs, or if Commit is +// called on a read-only transaction. +func (tx *Tx) Commit() error { + _assert(!tx.managed, "managed tx commit not allowed") + if tx.db == nil { + return ErrTxClosed + } else if !tx.writable { + return ErrTxNotWritable + } + + // TODO(benbjohnson): Use vectorized I/O to write out dirty pages. + + // Rebalance nodes which have had deletions. + var startTime = time.Now() + tx.root.rebalance() + if tx.stats.Rebalance > 0 { + tx.stats.RebalanceTime += time.Since(startTime) + } + + // spill data onto dirty pages. + startTime = time.Now() + if err := tx.root.spill(); err != nil { + tx.rollback() + return err + } + tx.stats.SpillTime += time.Since(startTime) + + // Free the old root bucket. + tx.meta.root.root = tx.root.root + + // Free the freelist and allocate new pages for it. This will overestimate + // the size of the freelist but not underestimate the size (which would be bad). + tx.db.freelist.free(tx.meta.txid, tx.db.page(tx.meta.freelist)) + p, err := tx.allocate((tx.db.freelist.size() / tx.db.pageSize) + 1) + if err != nil { + tx.rollback() + return err + } + if err := tx.db.freelist.write(p); err != nil { + tx.rollback() + return err + } + tx.meta.freelist = p.id + + // Write dirty pages to disk. + startTime = time.Now() + if err := tx.write(); err != nil { + tx.rollback() + return err + } + + // If strict mode is enabled then perform a consistency check. + // Only the first consistency error is reported in the panic. + if tx.db.StrictMode { + if err, ok := <-tx.Check(); ok { + panic("check fail: " + err.Error()) + } + } + + // Write meta to disk. + if err := tx.writeMeta(); err != nil { + tx.rollback() + return err + } + tx.stats.WriteTime += time.Since(startTime) + + // Finalize the transaction. + tx.close() + + // Execute commit handlers now that the locks have been removed. + for _, fn := range tx.commitHandlers { + fn() + } + + return nil +} + +// Rollback closes the transaction and ignores all previous updates. Read-only +// transactions must be rolled back and not committed. +func (tx *Tx) Rollback() error { + _assert(!tx.managed, "managed tx rollback not allowed") + if tx.db == nil { + return ErrTxClosed + } + tx.rollback() + return nil +} + +func (tx *Tx) rollback() { + if tx.db == nil { + return + } + if tx.writable { + tx.db.freelist.rollback(tx.meta.txid) + tx.db.freelist.reload(tx.db.page(tx.db.meta().freelist)) + } + tx.close() +} + +func (tx *Tx) close() { + if tx.db == nil { + return + } + if tx.writable { + // Grab freelist stats. + var freelistFreeN = tx.db.freelist.free_count() + var freelistPendingN = tx.db.freelist.pending_count() + var freelistAlloc = tx.db.freelist.size() + + // Remove writer lock. + tx.db.rwlock.Unlock() + + // Merge statistics. + tx.db.statlock.Lock() + tx.db.stats.FreePageN = freelistFreeN + tx.db.stats.PendingPageN = freelistPendingN + tx.db.stats.FreeAlloc = (freelistFreeN + freelistPendingN) * tx.db.pageSize + tx.db.stats.FreelistInuse = freelistAlloc + tx.db.stats.TxStats.add(&tx.stats) + tx.db.statlock.Unlock() + } else { + tx.db.removeTx(tx) + } + tx.db = nil +} + +// Copy writes the entire database to a writer. +// This function exists for backwards compatibility. Use WriteTo() in +func (tx *Tx) Copy(w io.Writer) error { + _, err := tx.WriteTo(w) + return err +} + +// WriteTo writes the entire database to a writer. +// If err == nil then exactly tx.Size() bytes will be written into the writer. +func (tx *Tx) WriteTo(w io.Writer) (n int64, err error) { + // Attempt to open reader directly. + var f *os.File + if f, err = os.OpenFile(tx.db.path, os.O_RDONLY|odirect, 0); err != nil { + // Fallback to a regular open if that doesn't work. + if f, err = os.OpenFile(tx.db.path, os.O_RDONLY, 0); err != nil { + return 0, err + } + } + + // Copy the meta pages. + tx.db.metalock.Lock() + n, err = io.CopyN(w, f, int64(tx.db.pageSize*2)) + tx.db.metalock.Unlock() + if err != nil { + _ = f.Close() + return n, fmt.Errorf("meta copy: %s", err) + } + + // Copy data pages. + wn, err := io.CopyN(w, f, tx.Size()-int64(tx.db.pageSize*2)) + n += wn + if err != nil { + _ = f.Close() + return n, err + } + + return n, f.Close() +} + +// CopyFile copies the entire database to file at the given path. +// A reader transaction is maintained during the copy so it is safe to continue +// using the database while a copy is in progress. +func (tx *Tx) CopyFile(path string, mode os.FileMode) error { + f, err := os.OpenFile(path, os.O_RDWR|os.O_CREATE|os.O_TRUNC, mode) + if err != nil { + return err + } + + err = tx.Copy(f) + if err != nil { + _ = f.Close() + return err + } + return f.Close() +} + +// Check performs several consistency checks on the database for this transaction. +// An error is returned if any inconsistency is found. +// +// It can be safely run concurrently on a writable transaction. However, this +// incurs a high cost for large databases and databases with a lot of subbuckets +// because of caching. This overhead can be removed if running on a read-only +// transaction, however, it is not safe to execute other writer transactions at +// the same time. +func (tx *Tx) Check() <-chan error { + ch := make(chan error) + go tx.check(ch) + return ch +} + +func (tx *Tx) check(ch chan error) { + // Check if any pages are double freed. + freed := make(map[pgid]bool) + for _, id := range tx.db.freelist.all() { + if freed[id] { + ch <- fmt.Errorf("page %d: already freed", id) + } + freed[id] = true + } + + // Track every reachable page. + reachable := make(map[pgid]*page) + reachable[0] = tx.page(0) // meta0 + reachable[1] = tx.page(1) // meta1 + for i := uint32(0); i <= tx.page(tx.meta.freelist).overflow; i++ { + reachable[tx.meta.freelist+pgid(i)] = tx.page(tx.meta.freelist) + } + + // Recursively check buckets. + tx.checkBucket(&tx.root, reachable, freed, ch) + + // Ensure all pages below high water mark are either reachable or freed. + for i := pgid(0); i < tx.meta.pgid; i++ { + _, isReachable := reachable[i] + if !isReachable && !freed[i] { + ch <- fmt.Errorf("page %d: unreachable unfreed", int(i)) + } + } + + // Close the channel to signal completion. + close(ch) +} + +func (tx *Tx) checkBucket(b *Bucket, reachable map[pgid]*page, freed map[pgid]bool, ch chan error) { + // Ignore inline buckets. + if b.root == 0 { + return + } + + // Check every page used by this bucket. + b.tx.forEachPage(b.root, 0, func(p *page, _ int) { + if p.id > tx.meta.pgid { + ch <- fmt.Errorf("page %d: out of bounds: %d", int(p.id), int(b.tx.meta.pgid)) + } + + // Ensure each page is only referenced once. + for i := pgid(0); i <= pgid(p.overflow); i++ { + var id = p.id + i + if _, ok := reachable[id]; ok { + ch <- fmt.Errorf("page %d: multiple references", int(id)) + } + reachable[id] = p + } + + // We should only encounter un-freed leaf and branch pages. + if freed[p.id] { + ch <- fmt.Errorf("page %d: reachable freed", int(p.id)) + } else if (p.flags&branchPageFlag) == 0 && (p.flags&leafPageFlag) == 0 { + ch <- fmt.Errorf("page %d: invalid type: %s", int(p.id), p.typ()) + } + }) + + // Check each bucket within this bucket. + _ = b.ForEach(func(k, v []byte) error { + if child := b.Bucket(k); child != nil { + tx.checkBucket(child, reachable, freed, ch) + } + return nil + }) +} + +// allocate returns a contiguous block of memory starting at a given page. +func (tx *Tx) allocate(count int) (*page, error) { + p, err := tx.db.allocate(count) + if err != nil { + return nil, err + } + + // Save to our page cache. + tx.pages[p.id] = p + + // Update statistics. + tx.stats.PageCount++ + tx.stats.PageAlloc += count * tx.db.pageSize + + return p, nil +} + +// write writes any dirty pages to disk. +func (tx *Tx) write() error { + // Sort pages by id. + pages := make(pages, 0, len(tx.pages)) + for _, p := range tx.pages { + pages = append(pages, p) + } + sort.Sort(pages) + + // Write pages to disk in order. + for _, p := range pages { + size := (int(p.overflow) + 1) * tx.db.pageSize + offset := int64(p.id) * int64(tx.db.pageSize) + + // Write out page in "max allocation" sized chunks. + ptr := (*[maxAllocSize]byte)(unsafe.Pointer(p)) + for { + // Limit our write to our max allocation size. + sz := size + if sz > maxAllocSize-1 { + sz = maxAllocSize - 1 + } + + // Write chunk to disk. + buf := ptr[:sz] + if _, err := tx.db.ops.writeAt(buf, offset); err != nil { + return err + } + + // Update statistics. + tx.stats.Write++ + + // Exit inner for loop if we've written all the chunks. + size -= sz + if size == 0 { + break + } + + // Otherwise move offset forward and move pointer to next chunk. + offset += int64(sz) + ptr = (*[maxAllocSize]byte)(unsafe.Pointer(&ptr[sz])) + } + } + + // Ignore file sync if flag is set on DB. + if !tx.db.NoSync || IgnoreNoSync { + if err := fdatasync(tx.db); err != nil { + return err + } + } + + // Clear out page cache. + tx.pages = make(map[pgid]*page) + + return nil +} + +// writeMeta writes the meta to the disk. +func (tx *Tx) writeMeta() error { + // Create a temporary buffer for the meta page. + buf := make([]byte, tx.db.pageSize) + p := tx.db.pageInBuffer(buf, 0) + tx.meta.write(p) + + // Write the meta page to file. + if _, err := tx.db.ops.writeAt(buf, int64(p.id)*int64(tx.db.pageSize)); err != nil { + return err + } + if !tx.db.NoSync || IgnoreNoSync { + if err := fdatasync(tx.db); err != nil { + return err + } + } + + // Update statistics. + tx.stats.Write++ + + return nil +} + +// page returns a reference to the page with a given id. +// If page has been written to then a temporary bufferred page is returned. +func (tx *Tx) page(id pgid) *page { + // Check the dirty pages first. + if tx.pages != nil { + if p, ok := tx.pages[id]; ok { + return p + } + } + + // Otherwise return directly from the mmap. + return tx.db.page(id) +} + +// forEachPage iterates over every page within a given page and executes a function. +func (tx *Tx) forEachPage(pgid pgid, depth int, fn func(*page, int)) { + p := tx.page(pgid) + + // Execute function. + fn(p, depth) + + // Recursively loop over children. + if (p.flags & branchPageFlag) != 0 { + for i := 0; i < int(p.count); i++ { + elem := p.branchPageElement(uint16(i)) + tx.forEachPage(elem.pgid, depth+1, fn) + } + } +} + +// Page returns page information for a given page number. +// This is only safe for concurrent use when used by a writable transaction. +func (tx *Tx) Page(id int) (*PageInfo, error) { + if tx.db == nil { + return nil, ErrTxClosed + } else if pgid(id) >= tx.meta.pgid { + return nil, nil + } + + // Build the page info. + p := tx.db.page(pgid(id)) + info := &PageInfo{ + ID: id, + Count: int(p.count), + OverflowCount: int(p.overflow), + } + + // Determine the type (or if it's free). + if tx.db.freelist.freed(pgid(id)) { + info.Type = "free" + } else { + info.Type = p.typ() + } + + return info, nil +} + +// TxStats represents statistics about the actions performed by the transaction. +type TxStats struct { + // Page statistics. + PageCount int // number of page allocations + PageAlloc int // total bytes allocated + + // Cursor statistics. + CursorCount int // number of cursors created + + // Node statistics + NodeCount int // number of node allocations + NodeDeref int // number of node dereferences + + // Rebalance statistics. + Rebalance int // number of node rebalances + RebalanceTime time.Duration // total time spent rebalancing + + // Split/Spill statistics. + Split int // number of nodes split + Spill int // number of nodes spilled + SpillTime time.Duration // total time spent spilling + + // Write statistics. + Write int // number of writes performed + WriteTime time.Duration // total time spent writing to disk +} + +func (s *TxStats) add(other *TxStats) { + s.PageCount += other.PageCount + s.PageAlloc += other.PageAlloc + s.CursorCount += other.CursorCount + s.NodeCount += other.NodeCount + s.NodeDeref += other.NodeDeref + s.Rebalance += other.Rebalance + s.RebalanceTime += other.RebalanceTime + s.Split += other.Split + s.Spill += other.Spill + s.SpillTime += other.SpillTime + s.Write += other.Write + s.WriteTime += other.WriteTime +} + +// Sub calculates and returns the difference between two sets of transaction stats. +// This is useful when obtaining stats at two different points and time and +// you need the performance counters that occurred within that time span. +func (s *TxStats) Sub(other *TxStats) TxStats { + var diff TxStats + diff.PageCount = s.PageCount - other.PageCount + diff.PageAlloc = s.PageAlloc - other.PageAlloc + diff.CursorCount = s.CursorCount - other.CursorCount + diff.NodeCount = s.NodeCount - other.NodeCount + diff.NodeDeref = s.NodeDeref - other.NodeDeref + diff.Rebalance = s.Rebalance - other.Rebalance + diff.RebalanceTime = s.RebalanceTime - other.RebalanceTime + diff.Split = s.Split - other.Split + diff.Spill = s.Spill - other.Spill + diff.SpillTime = s.SpillTime - other.SpillTime + diff.Write = s.Write - other.Write + diff.WriteTime = s.WriteTime - other.WriteTime + return diff +} diff --git a/Godeps/_workspace/src/github.com/boltdb/bolt/tx_test.go b/Godeps/_workspace/src/github.com/boltdb/bolt/tx_test.go new file mode 100644 index 0000000000000..6c8271a608c58 --- /dev/null +++ b/Godeps/_workspace/src/github.com/boltdb/bolt/tx_test.go @@ -0,0 +1,456 @@ +package bolt_test + +import ( + "errors" + "fmt" + "os" + "testing" + + "github.com/boltdb/bolt" +) + +// Ensure that committing a closed transaction returns an error. +func TestTx_Commit_Closed(t *testing.T) { + db := NewTestDB() + defer db.Close() + tx, _ := db.Begin(true) + tx.CreateBucket([]byte("foo")) + ok(t, tx.Commit()) + equals(t, tx.Commit(), bolt.ErrTxClosed) +} + +// Ensure that rolling back a closed transaction returns an error. +func TestTx_Rollback_Closed(t *testing.T) { + db := NewTestDB() + defer db.Close() + tx, _ := db.Begin(true) + ok(t, tx.Rollback()) + equals(t, tx.Rollback(), bolt.ErrTxClosed) +} + +// Ensure that committing a read-only transaction returns an error. +func TestTx_Commit_ReadOnly(t *testing.T) { + db := NewTestDB() + defer db.Close() + tx, _ := db.Begin(false) + equals(t, tx.Commit(), bolt.ErrTxNotWritable) +} + +// Ensure that a transaction can retrieve a cursor on the root bucket. +func TestTx_Cursor(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.CreateBucket([]byte("woojits")) + c := tx.Cursor() + + k, v := c.First() + equals(t, "widgets", string(k)) + assert(t, v == nil, "") + + k, v = c.Next() + equals(t, "woojits", string(k)) + assert(t, v == nil, "") + + k, v = c.Next() + assert(t, k == nil, "") + assert(t, v == nil, "") + + return nil + }) +} + +// Ensure that creating a bucket with a read-only transaction returns an error. +func TestTx_CreateBucket_ReadOnly(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.View(func(tx *bolt.Tx) error { + b, err := tx.CreateBucket([]byte("foo")) + assert(t, b == nil, "") + equals(t, bolt.ErrTxNotWritable, err) + return nil + }) +} + +// Ensure that creating a bucket on a closed transaction returns an error. +func TestTx_CreateBucket_Closed(t *testing.T) { + db := NewTestDB() + defer db.Close() + tx, _ := db.Begin(true) + tx.Commit() + b, err := tx.CreateBucket([]byte("foo")) + assert(t, b == nil, "") + equals(t, bolt.ErrTxClosed, err) +} + +// Ensure that a Tx can retrieve a bucket. +func TestTx_Bucket(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + b := tx.Bucket([]byte("widgets")) + assert(t, b != nil, "") + return nil + }) +} + +// Ensure that a Tx retrieving a non-existent key returns nil. +func TestTx_Get_Missing(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar")) + value := tx.Bucket([]byte("widgets")).Get([]byte("no_such_key")) + assert(t, value == nil, "") + return nil + }) +} + +// Ensure that a bucket can be created and retrieved. +func TestTx_CreateBucket(t *testing.T) { + db := NewTestDB() + defer db.Close() + + // Create a bucket. + db.Update(func(tx *bolt.Tx) error { + b, err := tx.CreateBucket([]byte("widgets")) + assert(t, b != nil, "") + ok(t, err) + return nil + }) + + // Read the bucket through a separate transaction. + db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("widgets")) + assert(t, b != nil, "") + return nil + }) +} + +// Ensure that a bucket can be created if it doesn't already exist. +func TestTx_CreateBucketIfNotExists(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + b, err := tx.CreateBucketIfNotExists([]byte("widgets")) + assert(t, b != nil, "") + ok(t, err) + + b, err = tx.CreateBucketIfNotExists([]byte("widgets")) + assert(t, b != nil, "") + ok(t, err) + + b, err = tx.CreateBucketIfNotExists([]byte{}) + assert(t, b == nil, "") + equals(t, bolt.ErrBucketNameRequired, err) + + b, err = tx.CreateBucketIfNotExists(nil) + assert(t, b == nil, "") + equals(t, bolt.ErrBucketNameRequired, err) + return nil + }) + + // Read the bucket through a separate transaction. + db.View(func(tx *bolt.Tx) error { + b := tx.Bucket([]byte("widgets")) + assert(t, b != nil, "") + return nil + }) +} + +// Ensure that a bucket cannot be created twice. +func TestTx_CreateBucket_Exists(t *testing.T) { + db := NewTestDB() + defer db.Close() + // Create a bucket. + db.Update(func(tx *bolt.Tx) error { + b, err := tx.CreateBucket([]byte("widgets")) + assert(t, b != nil, "") + ok(t, err) + return nil + }) + + // Create the same bucket again. + db.Update(func(tx *bolt.Tx) error { + b, err := tx.CreateBucket([]byte("widgets")) + assert(t, b == nil, "") + equals(t, bolt.ErrBucketExists, err) + return nil + }) +} + +// Ensure that a bucket is created with a non-blank name. +func TestTx_CreateBucket_NameRequired(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + b, err := tx.CreateBucket(nil) + assert(t, b == nil, "") + equals(t, bolt.ErrBucketNameRequired, err) + return nil + }) +} + +// Ensure that a bucket can be deleted. +func TestTx_DeleteBucket(t *testing.T) { + db := NewTestDB() + defer db.Close() + + // Create a bucket and add a value. + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar")) + return nil + }) + + // Delete the bucket and make sure we can't get the value. + db.Update(func(tx *bolt.Tx) error { + ok(t, tx.DeleteBucket([]byte("widgets"))) + assert(t, tx.Bucket([]byte("widgets")) == nil, "") + return nil + }) + + db.Update(func(tx *bolt.Tx) error { + // Create the bucket again and make sure there's not a phantom value. + b, err := tx.CreateBucket([]byte("widgets")) + assert(t, b != nil, "") + ok(t, err) + assert(t, tx.Bucket([]byte("widgets")).Get([]byte("foo")) == nil, "") + return nil + }) +} + +// Ensure that deleting a bucket on a closed transaction returns an error. +func TestTx_DeleteBucket_Closed(t *testing.T) { + db := NewTestDB() + defer db.Close() + tx, _ := db.Begin(true) + tx.Commit() + equals(t, tx.DeleteBucket([]byte("foo")), bolt.ErrTxClosed) +} + +// Ensure that deleting a bucket with a read-only transaction returns an error. +func TestTx_DeleteBucket_ReadOnly(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.View(func(tx *bolt.Tx) error { + equals(t, tx.DeleteBucket([]byte("foo")), bolt.ErrTxNotWritable) + return nil + }) +} + +// Ensure that nothing happens when deleting a bucket that doesn't exist. +func TestTx_DeleteBucket_NotFound(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + equals(t, bolt.ErrBucketNotFound, tx.DeleteBucket([]byte("widgets"))) + return nil + }) +} + +// Ensure that no error is returned when a tx.ForEach function does not return +// an error. +func TestTx_ForEach_NoError(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar")) + + equals(t, nil, tx.ForEach(func(name []byte, b *bolt.Bucket) error { + return nil + })) + return nil + }) +} + +// Ensure that an error is returned when a tx.ForEach function returns an error. +func TestTx_ForEach_WithError(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar")) + + err := errors.New("foo") + equals(t, err, tx.ForEach(func(name []byte, b *bolt.Bucket) error { + return err + })) + return nil + }) +} + +// Ensure that Tx commit handlers are called after a transaction successfully commits. +func TestTx_OnCommit(t *testing.T) { + var x int + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.OnCommit(func() { x += 1 }) + tx.OnCommit(func() { x += 2 }) + _, err := tx.CreateBucket([]byte("widgets")) + return err + }) + equals(t, 3, x) +} + +// Ensure that Tx commit handlers are NOT called after a transaction rolls back. +func TestTx_OnCommit_Rollback(t *testing.T) { + var x int + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.OnCommit(func() { x += 1 }) + tx.OnCommit(func() { x += 2 }) + tx.CreateBucket([]byte("widgets")) + return errors.New("rollback this commit") + }) + equals(t, 0, x) +} + +// Ensure that the database can be copied to a file path. +func TestTx_CopyFile(t *testing.T) { + db := NewTestDB() + defer db.Close() + var dest = tempfile() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar")) + tx.Bucket([]byte("widgets")).Put([]byte("baz"), []byte("bat")) + return nil + }) + + ok(t, db.View(func(tx *bolt.Tx) error { return tx.CopyFile(dest, 0600) })) + + db2, err := bolt.Open(dest, 0600, nil) + ok(t, err) + defer db2.Close() + + db2.View(func(tx *bolt.Tx) error { + equals(t, []byte("bar"), tx.Bucket([]byte("widgets")).Get([]byte("foo"))) + equals(t, []byte("bat"), tx.Bucket([]byte("widgets")).Get([]byte("baz"))) + return nil + }) +} + +type failWriterError struct{} + +func (failWriterError) Error() string { + return "error injected for tests" +} + +type failWriter struct { + // fail after this many bytes + After int +} + +func (f *failWriter) Write(p []byte) (n int, err error) { + n = len(p) + if n > f.After { + n = f.After + err = failWriterError{} + } + f.After -= n + return n, err +} + +// Ensure that Copy handles write errors right. +func TestTx_CopyFile_Error_Meta(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar")) + tx.Bucket([]byte("widgets")).Put([]byte("baz"), []byte("bat")) + return nil + }) + + err := db.View(func(tx *bolt.Tx) error { return tx.Copy(&failWriter{}) }) + equals(t, err.Error(), "meta copy: error injected for tests") +} + +// Ensure that Copy handles write errors right. +func TestTx_CopyFile_Error_Normal(t *testing.T) { + db := NewTestDB() + defer db.Close() + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar")) + tx.Bucket([]byte("widgets")).Put([]byte("baz"), []byte("bat")) + return nil + }) + + err := db.View(func(tx *bolt.Tx) error { return tx.Copy(&failWriter{3 * db.Info().PageSize}) }) + equals(t, err.Error(), "error injected for tests") +} + +func ExampleTx_Rollback() { + // Open the database. + db, _ := bolt.Open(tempfile(), 0666, nil) + defer os.Remove(db.Path()) + defer db.Close() + + // Create a bucket. + db.Update(func(tx *bolt.Tx) error { + _, err := tx.CreateBucket([]byte("widgets")) + return err + }) + + // Set a value for a key. + db.Update(func(tx *bolt.Tx) error { + return tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar")) + }) + + // Update the key but rollback the transaction so it never saves. + tx, _ := db.Begin(true) + b := tx.Bucket([]byte("widgets")) + b.Put([]byte("foo"), []byte("baz")) + tx.Rollback() + + // Ensure that our original value is still set. + db.View(func(tx *bolt.Tx) error { + value := tx.Bucket([]byte("widgets")).Get([]byte("foo")) + fmt.Printf("The value for 'foo' is still: %s\n", value) + return nil + }) + + // Output: + // The value for 'foo' is still: bar +} + +func ExampleTx_CopyFile() { + // Open the database. + db, _ := bolt.Open(tempfile(), 0666, nil) + defer os.Remove(db.Path()) + defer db.Close() + + // Create a bucket and a key. + db.Update(func(tx *bolt.Tx) error { + tx.CreateBucket([]byte("widgets")) + tx.Bucket([]byte("widgets")).Put([]byte("foo"), []byte("bar")) + return nil + }) + + // Copy the database to another file. + toFile := tempfile() + db.View(func(tx *bolt.Tx) error { return tx.CopyFile(toFile, 0666) }) + defer os.Remove(toFile) + + // Open the cloned database. + db2, _ := bolt.Open(toFile, 0666, nil) + defer db2.Close() + + // Ensure that the key exists in the copy. + db2.View(func(tx *bolt.Tx) error { + value := tx.Bucket([]byte("widgets")).Get([]byte("foo")) + fmt.Printf("The value for 'foo' in the clone is: %s\n", value) + return nil + }) + + // Output: + // The value for 'foo' in the clone is: bar +} diff --git a/Godeps/_workspace/src/github.com/cenkalti/backoff/.gitignore b/Godeps/_workspace/src/github.com/cenkalti/backoff/.gitignore new file mode 100644 index 0000000000000..00268614f0456 --- /dev/null +++ b/Godeps/_workspace/src/github.com/cenkalti/backoff/.gitignore @@ -0,0 +1,22 @@ +# Compiled Object files, Static and Dynamic libs (Shared Objects) +*.o +*.a +*.so + +# Folders +_obj +_test + +# Architecture specific extensions/prefixes +*.[568vq] +[568vq].out + +*.cgo1.go +*.cgo2.c +_cgo_defun.c +_cgo_gotypes.go +_cgo_export.* + +_testmain.go + +*.exe diff --git a/Godeps/_workspace/src/github.com/cenkalti/backoff/.travis.yml b/Godeps/_workspace/src/github.com/cenkalti/backoff/.travis.yml new file mode 100644 index 0000000000000..ce9cb62333543 --- /dev/null +++ b/Godeps/_workspace/src/github.com/cenkalti/backoff/.travis.yml @@ -0,0 +1,2 @@ +language: go +go: 1.3.3 diff --git a/Godeps/_workspace/src/github.com/cenkalti/backoff/LICENSE b/Godeps/_workspace/src/github.com/cenkalti/backoff/LICENSE new file mode 100644 index 0000000000000..89b8179965581 --- /dev/null +++ b/Godeps/_workspace/src/github.com/cenkalti/backoff/LICENSE @@ -0,0 +1,20 @@ +The MIT License (MIT) + +Copyright (c) 2014 Cenk Altı + +Permission is hereby granted, free of charge, to any person obtaining a copy of +this software and associated documentation files (the "Software"), to deal in +the Software without restriction, including without limitation the rights to +use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of +the Software, and to permit persons to whom the Software is furnished to do so, +subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS +FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR +COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER +IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN +CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/Godeps/_workspace/src/github.com/cenkalti/backoff/README.md b/Godeps/_workspace/src/github.com/cenkalti/backoff/README.md new file mode 100644 index 0000000000000..020b8fbf339e9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/cenkalti/backoff/README.md @@ -0,0 +1,116 @@ +# Exponential Backoff [![GoDoc][godoc image]][godoc] [![Build Status][travis image]][travis] + +This is a Go port of the exponential backoff algorithm from [Google's HTTP Client Library for Java][google-http-java-client]. + +[Exponential backoff][exponential backoff wiki] +is an algorithm that uses feedback to multiplicatively decrease the rate of some process, +in order to gradually find an acceptable rate. +The retries exponentially increase and stop increasing when a certain threshold is met. + +## How To + +We define two functions, `Retry()` and `RetryNotify()`. +They receive an `Operation` to execute, a `BackOff` algorithm, +and an optional `Notify` error handler. + +The operation will be executed, and will be retried on failure with delay +as given by the backoff algorithm. The backoff algorithm can also decide when to stop +retrying. +In addition, the notify error handler will be called after each failed attempt, +except for the last time, whose error should be handled by the caller. + +```go +// An Operation is executing by Retry() or RetryNotify(). +// The operation will be retried using a backoff policy if it returns an error. +type Operation func() error + +// Notify is a notify-on-error function. It receives an operation error and +// backoff delay if the operation failed (with an error). +// +// NOTE that if the backoff policy stated to stop retrying, +// the notify function isn't called. +type Notify func(error, time.Duration) + +func Retry(Operation, BackOff) error +func RetryNotify(Operation, BackOff, Notify) +``` + +## Examples + +See more advanced examples in the [godoc][advanced example]. + +### Retry + +Simple retry helper that uses the default exponential backoff algorithm: + +```go +operation := func() error { + // An operation that might fail. + return nil // or return errors.New("some error") +} + +err := Retry(operation, NewExponentialBackOff()) +if err != nil { + // Handle error. + return err +} + +// Operation is successful. +return nil +``` + +### Ticker + +```go +operation := func() error { + // An operation that might fail + return nil // or return errors.New("some error") +} + +b := NewExponentialBackOff() +ticker := NewTicker(b) + +var err error + +// Ticks will continue to arrive when the previous operation is still running, +// so operations that take a while to fail could run in quick succession. +for range ticker.C { + if err = operation(); err != nil { + log.Println(err, "will retry...") + continue + } + + ticker.Stop() + break +} + +if err != nil { + // Operation has failed. + return err +} + +// Operation is successful. +return nil +``` + +## Getting Started + +```bash +# install +$ go get github.com/cenkalti/backoff + +# test +$ cd $GOPATH/src/github.com/cenkalti/backoff +$ go get -t ./... +$ go test -v -cover +``` + +[godoc]: https://godoc.org/github.com/cenkalti/backoff +[godoc image]: https://godoc.org/github.com/cenkalti/backoff?status.png +[travis]: https://travis-ci.org/cenkalti/backoff +[travis image]: https://travis-ci.org/cenkalti/backoff.png + +[google-http-java-client]: https://github.com/google/google-http-java-client +[exponential backoff wiki]: http://en.wikipedia.org/wiki/Exponential_backoff + +[advanced example]: https://godoc.org/github.com/cenkalti/backoff#example_ diff --git a/Godeps/_workspace/src/github.com/cenkalti/backoff/adv_example_test.go b/Godeps/_workspace/src/github.com/cenkalti/backoff/adv_example_test.go new file mode 100644 index 0000000000000..3fe6783b8238c --- /dev/null +++ b/Godeps/_workspace/src/github.com/cenkalti/backoff/adv_example_test.go @@ -0,0 +1,117 @@ +package backoff + +import ( + "io/ioutil" + "log" + "net/http" + "time" +) + +// This is an example that demonstrates how this package could be used +// to perform various advanced operations. +// +// It executes an HTTP GET request with exponential backoff, +// while errors are logged and failed responses are closed, as required by net/http package. +// +// Note we define a condition function which is used inside the operation to +// determine whether the operation succeeded or failed. +func Example() error { + res, err := GetWithRetry( + "http://localhost:9999", + ErrorIfStatusCodeIsNot(http.StatusOK), + NewExponentialBackOff()) + + if err != nil { + // Close response body of last (failed) attempt. + // The Last attempt isn't handled by the notify-on-error function, + // which closes the body of all the previous attempts. + if e := res.Body.Close(); e != nil { + log.Printf("error closing last attempt's response body: %s", e) + } + log.Printf("too many failed request attempts: %s", err) + return err + } + defer res.Body.Close() // The response's Body must be closed. + + // Read body + _, _ = ioutil.ReadAll(res.Body) + + // Do more stuff + return nil +} + +// GetWithRetry is a helper function that performs an HTTP GET request +// to the given URL, and retries with the given backoff using the given condition function. +// +// It also uses a notify-on-error function which logs +// and closes the response body of the failed request. +func GetWithRetry(url string, condition Condition, bck BackOff) (*http.Response, error) { + var res *http.Response + err := RetryNotify( + func() error { + var err error + res, err = http.Get(url) + if err != nil { + return err + } + return condition(res) + }, + bck, + LogAndClose()) + + return res, err +} + +// Condition is a retry condition function. +// It receives a response, and returns an error +// if the response failed the condition. +type Condition func(*http.Response) error + +// ErrorIfStatusCodeIsNot returns a retry condition function. +// The condition returns an error +// if the given response's status code is not the given HTTP status code. +func ErrorIfStatusCodeIsNot(status int) Condition { + return func(res *http.Response) error { + if res.StatusCode != status { + return NewError(res) + } + return nil + } +} + +// Error is returned on ErrorIfX() condition functions throughout this package. +type Error struct { + Response *http.Response +} + +func NewError(res *http.Response) *Error { + // Sanity check + if res == nil { + panic("response object is nil") + } + return &Error{Response: res} +} +func (err *Error) Error() string { return "request failed" } + +// LogAndClose is a notify-on-error function. +// It logs the error and closes the response body. +func LogAndClose() Notify { + return func(err error, wait time.Duration) { + switch e := err.(type) { + case *Error: + defer e.Response.Body.Close() + + b, err := ioutil.ReadAll(e.Response.Body) + var body string + if err != nil { + body = "can't read body" + } else { + body = string(b) + } + + log.Printf("%s: %s", e.Response.Status, body) + default: + log.Println(err) + } + } +} diff --git a/Godeps/_workspace/src/github.com/cenkalti/backoff/backoff.go b/Godeps/_workspace/src/github.com/cenkalti/backoff/backoff.go new file mode 100644 index 0000000000000..61bd6df66c08a --- /dev/null +++ b/Godeps/_workspace/src/github.com/cenkalti/backoff/backoff.go @@ -0,0 +1,59 @@ +// Package backoff implements backoff algorithms for retrying operations. +// +// Also has a Retry() helper for retrying operations that may fail. +package backoff + +import "time" + +// BackOff is a backoff policy for retrying an operation. +type BackOff interface { + // NextBackOff returns the duration to wait before retrying the operation, + // or backoff.Stop to indicate that no more retries should be made. + // + // Example usage: + // + // duration := backoff.NextBackOff(); + // if (duration == backoff.Stop) { + // // Do not retry operation. + // } else { + // // Sleep for duration and retry operation. + // } + // + NextBackOff() time.Duration + + // Reset to initial state. + Reset() +} + +// Indicates that no more retries should be made for use in NextBackOff(). +const Stop time.Duration = -1 + +// ZeroBackOff is a fixed backoff policy whose backoff time is always zero, +// meaning that the operation is retried immediately without waiting, indefinitely. +type ZeroBackOff struct{} + +func (b *ZeroBackOff) Reset() {} + +func (b *ZeroBackOff) NextBackOff() time.Duration { return 0 } + +// StopBackOff is a fixed backoff policy that always returns backoff.Stop for +// NextBackOff(), meaning that the operation should never be retried. +type StopBackOff struct{} + +func (b *StopBackOff) Reset() {} + +func (b *StopBackOff) NextBackOff() time.Duration { return Stop } + +// ConstantBackOff is a backoff policy that always returns the same backoff delay. +// This is in contrast to an exponential backoff policy, +// which returns a delay that grows longer as you call NextBackOff() over and over again. +type ConstantBackOff struct { + Interval time.Duration +} + +func (b *ConstantBackOff) Reset() {} +func (b *ConstantBackOff) NextBackOff() time.Duration { return b.Interval } + +func NewConstantBackOff(d time.Duration) *ConstantBackOff { + return &ConstantBackOff{Interval: d} +} diff --git a/Godeps/_workspace/src/github.com/cenkalti/backoff/backoff_test.go b/Godeps/_workspace/src/github.com/cenkalti/backoff/backoff_test.go new file mode 100644 index 0000000000000..91f27c4f19010 --- /dev/null +++ b/Godeps/_workspace/src/github.com/cenkalti/backoff/backoff_test.go @@ -0,0 +1,27 @@ +package backoff + +import ( + "testing" + "time" +) + +func TestNextBackOffMillis(t *testing.T) { + subtestNextBackOff(t, 0, new(ZeroBackOff)) + subtestNextBackOff(t, Stop, new(StopBackOff)) +} + +func subtestNextBackOff(t *testing.T, expectedValue time.Duration, backOffPolicy BackOff) { + for i := 0; i < 10; i++ { + next := backOffPolicy.NextBackOff() + if next != expectedValue { + t.Errorf("got: %d expected: %d", next, expectedValue) + } + } +} + +func TestConstantBackOff(t *testing.T) { + backoff := NewConstantBackOff(time.Second) + if backoff.NextBackOff() != time.Second { + t.Error("invalid interval") + } +} diff --git a/Godeps/_workspace/src/github.com/cenkalti/backoff/example_test.go b/Godeps/_workspace/src/github.com/cenkalti/backoff/example_test.go new file mode 100644 index 0000000000000..0d1852e45cdc7 --- /dev/null +++ b/Godeps/_workspace/src/github.com/cenkalti/backoff/example_test.go @@ -0,0 +1,51 @@ +package backoff + +import "log" + +func ExampleRetry() error { + operation := func() error { + // An operation that might fail. + return nil // or return errors.New("some error") + } + + err := Retry(operation, NewExponentialBackOff()) + if err != nil { + // Handle error. + return err + } + + // Operation is successful. + return nil +} + +func ExampleTicker() error { + operation := func() error { + // An operation that might fail + return nil // or return errors.New("some error") + } + + b := NewExponentialBackOff() + ticker := NewTicker(b) + + var err error + + // Ticks will continue to arrive when the previous operation is still running, + // so operations that take a while to fail could run in quick succession. + for _ = range ticker.C { + if err = operation(); err != nil { + log.Println(err, "will retry...") + continue + } + + ticker.Stop() + break + } + + if err != nil { + // Operation has failed. + return err + } + + // Operation is successful. + return nil +} diff --git a/Godeps/_workspace/src/github.com/cenkalti/backoff/exponential.go b/Godeps/_workspace/src/github.com/cenkalti/backoff/exponential.go new file mode 100644 index 0000000000000..cc2a164f288f8 --- /dev/null +++ b/Godeps/_workspace/src/github.com/cenkalti/backoff/exponential.go @@ -0,0 +1,151 @@ +package backoff + +import ( + "math/rand" + "time" +) + +/* +ExponentialBackOff is a backoff implementation that increases the backoff +period for each retry attempt using a randomization function that grows exponentially. + +NextBackOff() is calculated using the following formula: + + randomized interval = + RetryInterval * (random value in range [1 - RandomizationFactor, 1 + RandomizationFactor]) + +In other words NextBackOff() will range between the randomization factor +percentage below and above the retry interval. + +For example, given the following parameters: + + RetryInterval = 2 + RandomizationFactor = 0.5 + Multiplier = 2 + +the actual backoff period used in the next retry attempt will range between 1 and 3 seconds, +multiplied by the exponential, that is, between 2 and 6 seconds. + +Note: MaxInterval caps the RetryInterval and not the randomized interval. + +If the time elapsed since an ExponentialBackOff instance is created goes past the +MaxElapsedTime, then the method NextBackOff() starts returning backoff.Stop. + +The elapsed time can be reset by calling Reset(). + +Example: Given the following default arguments, for 10 tries the sequence will be, +and assuming we go over the MaxElapsedTime on the 10th try: + + Request # RetryInterval (seconds) Randomized Interval (seconds) + + 1 0.5 [0.25, 0.75] + 2 0.75 [0.375, 1.125] + 3 1.125 [0.562, 1.687] + 4 1.687 [0.8435, 2.53] + 5 2.53 [1.265, 3.795] + 6 3.795 [1.897, 5.692] + 7 5.692 [2.846, 8.538] + 8 8.538 [4.269, 12.807] + 9 12.807 [6.403, 19.210] + 10 19.210 backoff.Stop + +Note: Implementation is not thread-safe. +*/ +type ExponentialBackOff struct { + InitialInterval time.Duration + RandomizationFactor float64 + Multiplier float64 + MaxInterval time.Duration + // After MaxElapsedTime the ExponentialBackOff stops. + // It never stops if MaxElapsedTime == 0. + MaxElapsedTime time.Duration + Clock Clock + + currentInterval time.Duration + startTime time.Time +} + +// Clock is an interface that returns current time for BackOff. +type Clock interface { + Now() time.Time +} + +// Default values for ExponentialBackOff. +const ( + DefaultInitialInterval = 500 * time.Millisecond + DefaultRandomizationFactor = 0.5 + DefaultMultiplier = 1.5 + DefaultMaxInterval = 60 * time.Second + DefaultMaxElapsedTime = 15 * time.Minute +) + +// NewExponentialBackOff creates an instance of ExponentialBackOff using default values. +func NewExponentialBackOff() *ExponentialBackOff { + b := &ExponentialBackOff{ + InitialInterval: DefaultInitialInterval, + RandomizationFactor: DefaultRandomizationFactor, + Multiplier: DefaultMultiplier, + MaxInterval: DefaultMaxInterval, + MaxElapsedTime: DefaultMaxElapsedTime, + Clock: SystemClock, + } + b.Reset() + return b +} + +type systemClock struct{} + +func (t systemClock) Now() time.Time { + return time.Now() +} + +// SystemClock implements Clock interface that uses time.Now(). +var SystemClock = systemClock{} + +// Reset the interval back to the initial retry interval and restarts the timer. +func (b *ExponentialBackOff) Reset() { + b.currentInterval = b.InitialInterval + b.startTime = b.Clock.Now() +} + +// NextBackOff calculates the next backoff interval using the formula: +// Randomized interval = RetryInterval +/- (RandomizationFactor * RetryInterval) +func (b *ExponentialBackOff) NextBackOff() time.Duration { + // Make sure we have not gone over the maximum elapsed time. + if b.MaxElapsedTime != 0 && b.GetElapsedTime() > b.MaxElapsedTime { + return Stop + } + defer b.incrementCurrentInterval() + return getRandomValueFromInterval(b.RandomizationFactor, rand.Float64(), b.currentInterval) +} + +// GetElapsedTime returns the elapsed time since an ExponentialBackOff instance +// is created and is reset when Reset() is called. +// +// The elapsed time is computed using time.Now().UnixNano(). +func (b *ExponentialBackOff) GetElapsedTime() time.Duration { + return b.Clock.Now().Sub(b.startTime) +} + +// Increments the current interval by multiplying it with the multiplier. +func (b *ExponentialBackOff) incrementCurrentInterval() { + // Check for overflow, if overflow is detected set the current interval to the max interval. + if float64(b.currentInterval) >= float64(b.MaxInterval)/b.Multiplier { + b.currentInterval = b.MaxInterval + } else { + b.currentInterval = time.Duration(float64(b.currentInterval) * b.Multiplier) + } +} + +// Returns a random value from the following interval: +// [randomizationFactor * currentInterval, randomizationFactor * currentInterval]. +func getRandomValueFromInterval(randomizationFactor, random float64, currentInterval time.Duration) time.Duration { + var delta = randomizationFactor * float64(currentInterval) + var minInterval = float64(currentInterval) - delta + var maxInterval = float64(currentInterval) + delta + + // Get a random value from the range [minInterval, maxInterval]. + // The formula used below has a +1 because if the minInterval is 1 and the maxInterval is 3 then + // we want a 33% chance for selecting either 1, 2 or 3. + return time.Duration(minInterval + (random * (maxInterval - minInterval + 1))) +} diff --git a/Godeps/_workspace/src/github.com/cenkalti/backoff/exponential_test.go b/Godeps/_workspace/src/github.com/cenkalti/backoff/exponential_test.go new file mode 100644 index 0000000000000..11b95e4f61d43 --- /dev/null +++ b/Godeps/_workspace/src/github.com/cenkalti/backoff/exponential_test.go @@ -0,0 +1,108 @@ +package backoff + +import ( + "math" + "testing" + "time" +) + +func TestBackOff(t *testing.T) { + var ( + testInitialInterval = 500 * time.Millisecond + testRandomizationFactor = 0.1 + testMultiplier = 2.0 + testMaxInterval = 5 * time.Second + testMaxElapsedTime = 15 * time.Minute + ) + + exp := NewExponentialBackOff() + exp.InitialInterval = testInitialInterval + exp.RandomizationFactor = testRandomizationFactor + exp.Multiplier = testMultiplier + exp.MaxInterval = testMaxInterval + exp.MaxElapsedTime = testMaxElapsedTime + exp.Reset() + + var expectedResults = []time.Duration{500, 1000, 2000, 4000, 5000, 5000, 5000, 5000, 5000, 5000} + for i, d := range expectedResults { + expectedResults[i] = d * time.Millisecond + } + + for _, expected := range expectedResults { + assertEquals(t, expected, exp.currentInterval) + // Assert that the next backoff falls in the expected range. + var minInterval = expected - time.Duration(testRandomizationFactor*float64(expected)) + var maxInterval = expected + time.Duration(testRandomizationFactor*float64(expected)) + var actualInterval = exp.NextBackOff() + if !(minInterval <= actualInterval && actualInterval <= maxInterval) { + t.Error("error") + } + } +} + +func TestGetRandomizedInterval(t *testing.T) { + // 33% chance of being 1. + assertEquals(t, 1, getRandomValueFromInterval(0.5, 0, 2)) + assertEquals(t, 1, getRandomValueFromInterval(0.5, 0.33, 2)) + // 33% chance of being 2. + assertEquals(t, 2, getRandomValueFromInterval(0.5, 0.34, 2)) + assertEquals(t, 2, getRandomValueFromInterval(0.5, 0.66, 2)) + // 33% chance of being 3. + assertEquals(t, 3, getRandomValueFromInterval(0.5, 0.67, 2)) + assertEquals(t, 3, getRandomValueFromInterval(0.5, 0.99, 2)) +} + +type TestClock struct { + i time.Duration + start time.Time +} + +func (c *TestClock) Now() time.Time { + t := c.start.Add(c.i) + c.i += time.Second + return t +} + +func TestGetElapsedTime(t *testing.T) { + var exp = NewExponentialBackOff() + exp.Clock = &TestClock{} + exp.Reset() + + var elapsedTime = exp.GetElapsedTime() + if elapsedTime != time.Second { + t.Errorf("elapsedTime=%d", elapsedTime) + } +} + +func TestMaxElapsedTime(t *testing.T) { + var exp = NewExponentialBackOff() + exp.Clock = &TestClock{start: time.Time{}.Add(10000 * time.Second)} + // Change the currentElapsedTime to be 0 ensuring that the elapsed time will be greater + // than the max elapsed time. + exp.startTime = time.Time{} + assertEquals(t, Stop, exp.NextBackOff()) +} + +func TestBackOffOverflow(t *testing.T) { + var ( + testInitialInterval time.Duration = math.MaxInt64 / 2 + testMaxInterval time.Duration = math.MaxInt64 + testMultiplier = 2.1 + ) + + exp := NewExponentialBackOff() + exp.InitialInterval = testInitialInterval + exp.Multiplier = testMultiplier + exp.MaxInterval = testMaxInterval + exp.Reset() + + exp.NextBackOff() + // Assert that when an overflow is possible the current varerval time.Duration is set to the max varerval time.Duration . + assertEquals(t, testMaxInterval, exp.currentInterval) +} + +func assertEquals(t *testing.T, expected, value time.Duration) { + if expected != value { + t.Errorf("got: %d, expected: %d", value, expected) + } +} diff --git a/Godeps/_workspace/src/github.com/cenkalti/backoff/retry.go b/Godeps/_workspace/src/github.com/cenkalti/backoff/retry.go new file mode 100644 index 0000000000000..f01f2bbd07ea2 --- /dev/null +++ b/Godeps/_workspace/src/github.com/cenkalti/backoff/retry.go @@ -0,0 +1,46 @@ +package backoff + +import "time" + +// An Operation is executing by Retry() or RetryNotify(). +// The operation will be retried using a backoff policy if it returns an error. +type Operation func() error + +// Notify is a notify-on-error function. It receives an operation error and +// backoff delay if the operation failed (with an error). +// +// NOTE that if the backoff policy stated to stop retrying, +// the notify function isn't called. +type Notify func(error, time.Duration) + +// Retry the function f until it does not return error or BackOff stops. +// f is guaranteed to be run at least once. +// It is the caller's responsibility to reset b after Retry returns. +// +// Retry sleeps the goroutine for the duration returned by BackOff after a +// failed operation returns. +func Retry(o Operation, b BackOff) error { return RetryNotify(o, b, nil) } + +// RetryNotify calls notify function with the error and wait duration +// for each failed attempt before sleep. +func RetryNotify(operation Operation, b BackOff, notify Notify) error { + var err error + var next time.Duration + + b.Reset() + for { + if err = operation(); err == nil { + return nil + } + + if next = b.NextBackOff(); next == Stop { + return err + } + + if notify != nil { + notify(err, next) + } + + time.Sleep(next) + } +} diff --git a/Godeps/_workspace/src/github.com/cenkalti/backoff/retry_test.go b/Godeps/_workspace/src/github.com/cenkalti/backoff/retry_test.go new file mode 100644 index 0000000000000..c0d25ab766e60 --- /dev/null +++ b/Godeps/_workspace/src/github.com/cenkalti/backoff/retry_test.go @@ -0,0 +1,34 @@ +package backoff + +import ( + "errors" + "log" + "testing" +) + +func TestRetry(t *testing.T) { + const successOn = 3 + var i = 0 + + // This function is successfull on "successOn" calls. + f := func() error { + i++ + log.Printf("function is called %d. time\n", i) + + if i == successOn { + log.Println("OK") + return nil + } + + log.Println("error") + return errors.New("error") + } + + err := Retry(f, NewExponentialBackOff()) + if err != nil { + t.Errorf("unexpected error: %s", err.Error()) + } + if i != successOn { + t.Errorf("invalid number of retries: %d", i) + } +} diff --git a/Godeps/_workspace/src/github.com/cenkalti/backoff/ticker.go b/Godeps/_workspace/src/github.com/cenkalti/backoff/ticker.go new file mode 100644 index 0000000000000..7a5ff4ed1fa93 --- /dev/null +++ b/Godeps/_workspace/src/github.com/cenkalti/backoff/ticker.go @@ -0,0 +1,79 @@ +package backoff + +import ( + "runtime" + "sync" + "time" +) + +// Ticker holds a channel that delivers `ticks' of a clock at times reported by a BackOff. +// +// Ticks will continue to arrive when the previous operation is still running, +// so operations that take a while to fail could run in quick succession. +type Ticker struct { + C <-chan time.Time + c chan time.Time + b BackOff + stop chan struct{} + stopOnce sync.Once +} + +// NewTicker returns a new Ticker containing a channel that will send the time at times +// specified by the BackOff argument. Ticker is guaranteed to tick at least once. +// The channel is closed when Stop method is called or BackOff stops. +func NewTicker(b BackOff) *Ticker { + c := make(chan time.Time) + t := &Ticker{ + C: c, + c: c, + b: b, + stop: make(chan struct{}), + } + go t.run() + runtime.SetFinalizer(t, (*Ticker).Stop) + return t +} + +// Stop turns off a ticker. After Stop, no more ticks will be sent. +func (t *Ticker) Stop() { + t.stopOnce.Do(func() { close(t.stop) }) +} + +func (t *Ticker) run() { + c := t.c + defer close(c) + t.b.Reset() + + // Ticker is guaranteed to tick at least once. + afterC := t.send(time.Now()) + + for { + if afterC == nil { + return + } + + select { + case tick := <-afterC: + afterC = t.send(tick) + case <-t.stop: + t.c = nil // Prevent future ticks from being sent to the channel. + return + } + } +} + +func (t *Ticker) send(tick time.Time) <-chan time.Time { + select { + case t.c <- tick: + case <-t.stop: + return nil + } + + next := t.b.NextBackOff() + if next == Stop { + t.Stop() + return nil + } + + return time.After(next) +} diff --git a/Godeps/_workspace/src/github.com/cenkalti/backoff/ticker_test.go b/Godeps/_workspace/src/github.com/cenkalti/backoff/ticker_test.go new file mode 100644 index 0000000000000..7c392df46ce46 --- /dev/null +++ b/Godeps/_workspace/src/github.com/cenkalti/backoff/ticker_test.go @@ -0,0 +1,45 @@ +package backoff + +import ( + "errors" + "log" + "testing" +) + +func TestTicker(t *testing.T) { + const successOn = 3 + var i = 0 + + // This function is successfull on "successOn" calls. + f := func() error { + i++ + log.Printf("function is called %d. time\n", i) + + if i == successOn { + log.Println("OK") + return nil + } + + log.Println("error") + return errors.New("error") + } + + b := NewExponentialBackOff() + ticker := NewTicker(b) + + var err error + for _ = range ticker.C { + if err = f(); err != nil { + t.Log(err) + continue + } + + break + } + if err != nil { + t.Errorf("unexpected error: %s", err.Error()) + } + if i != successOn { + t.Errorf("invalid number of retries: %d", i) + } +} diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/cache.go b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/cache.go new file mode 100644 index 0000000000000..feb28f2a54e5a --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/cache.go @@ -0,0 +1,258 @@ +// This code is based on encoding/json and gorilla/schema + +package encoding + +import ( + "reflect" + "sort" + "sync" +) + +// A field represents a single field found in a struct. +type field struct { + name string + nameBytes []byte // []byte(name) + equalFold func(s, t []byte) bool + + tag bool + index []int + typ reflect.Type + omitEmpty bool + quoted bool +} + +func fillField(f field) field { + f.nameBytes = []byte(f.name) + f.equalFold = foldFunc(f.nameBytes) + + return f +} + +// byName sorts field by name, breaking ties with depth, +// then breaking ties with "name came from tag", then +// breaking ties with index sequence. +type byName []field + +func (x byName) Len() int { return len(x) } + +func (x byName) Swap(i, j int) { x[i], x[j] = x[j], x[i] } + +func (x byName) Less(i, j int) bool { + if x[i].name != x[j].name { + return x[i].name < x[j].name + } + if len(x[i].index) != len(x[j].index) { + return len(x[i].index) < len(x[j].index) + } + if x[i].tag != x[j].tag { + return x[i].tag + } + return byIndex(x).Less(i, j) +} + +// byIndex sorts field by index sequence. +type byIndex []field + +func (x byIndex) Len() int { return len(x) } + +func (x byIndex) Swap(i, j int) { x[i], x[j] = x[j], x[i] } + +func (x byIndex) Less(i, j int) bool { + for k, xik := range x[i].index { + if k >= len(x[j].index) { + return false + } + if xik != x[j].index[k] { + return xik < x[j].index[k] + } + } + return len(x[i].index) < len(x[j].index) +} + +// typeFields returns a list of fields that should be recognized for the given type. +// The algorithm is breadth-first search over the set of structs to include - the top struct +// and then any reachable anonymous structs. +func typeFields(t reflect.Type) []field { + // Anonymous fields to explore at the current level and the next. + current := []field{} + next := []field{{typ: t}} + + // Count of queued names for current level and the next. + count := map[reflect.Type]int{} + nextCount := map[reflect.Type]int{} + + // Types already visited at an earlier level. + visited := map[reflect.Type]bool{} + + // Fields found. + var fields []field + + for len(next) > 0 { + current, next = next, current[:0] + count, nextCount = nextCount, map[reflect.Type]int{} + + for _, f := range current { + if visited[f.typ] { + continue + } + visited[f.typ] = true + + // Scan f.typ for fields to include. + for i := 0; i < f.typ.NumField(); i++ { + sf := f.typ.Field(i) + if sf.PkgPath != "" { // unexported + continue + } + tag := getTag(sf) + if tag == "-" { + continue + } + name, opts := parseTag(tag) + if !isValidTag(name) { + name = "" + } + index := make([]int, len(f.index)+1) + copy(index, f.index) + index[len(f.index)] = i + + ft := sf.Type + if ft.Name() == "" && ft.Kind() == reflect.Ptr { + // Follow pointer. + ft = ft.Elem() + } + + // Record found field and index sequence. + if name != "" || !sf.Anonymous || ft.Kind() != reflect.Struct { + tagged := name != "" + if name == "" { + name = sf.Name + } + fields = append(fields, fillField(field{ + name: name, + tag: tagged, + index: index, + typ: ft, + omitEmpty: opts.Contains("omitempty"), + })) + if count[f.typ] > 1 { + // If there were multiple instances, add a second, + // so that the annihilation code will see a duplicate. + // It only cares about the distinction between 1 or 2, + // so don't bother generating any more copies. + fields = append(fields, fields[len(fields)-1]) + } + continue + } + + // Record new anonymous struct to explore in next round. + nextCount[ft]++ + if nextCount[ft] == 1 { + next = append(next, fillField(field{name: ft.Name(), index: index, typ: ft})) + } + } + } + } + + sort.Sort(byName(fields)) + + // Delete all fields that are hidden by the Go rules for embedded fields, + // except that fields with valid tags are promoted. + + // The fields are sorted in primary order of name, secondary order + // of field index length. Loop over names; for each name, delete + // hidden fields by choosing the one dominant field that survives. + out := fields[:0] + for advance, i := 0, 0; i < len(fields); i += advance { + // One iteration per name. + // Find the sequence of fields with the name of this first field. + fi := fields[i] + name := fi.name + for advance = 1; i+advance < len(fields); advance++ { + fj := fields[i+advance] + if fj.name != name { + break + } + } + if advance == 1 { // Only one field with this name + out = append(out, fi) + continue + } + dominant, ok := dominantField(fields[i : i+advance]) + if ok { + out = append(out, dominant) + } + } + + fields = out + sort.Sort(byIndex(fields)) + + return fields +} + +// dominantField looks through the fields, all of which are known to +// have the same name, to find the single field that dominates the +// others using Go's embedding rules, modified by the presence of +// valid tags. If there are multiple top-level fields, the boolean +// will be false: This condition is an error in Go and we skip all +// the fields. +func dominantField(fields []field) (field, bool) { + // The fields are sorted in increasing index-length order. The winner + // must therefore be one with the shortest index length. Drop all + // longer entries, which is easy: just truncate the slice. + length := len(fields[0].index) + tagged := -1 // Index of first tagged field. + for i, f := range fields { + if len(f.index) > length { + fields = fields[:i] + break + } + if f.tag { + if tagged >= 0 { + // Multiple tagged fields at the same level: conflict. + // Return no field. + return field{}, false + } + tagged = i + } + } + if tagged >= 0 { + return fields[tagged], true + } + // All remaining fields have the same length. If there's more than one, + // we have a conflict (two fields named "X" at the same level) and we + // return no field. + if len(fields) > 1 { + return field{}, false + } + return fields[0], true +} + +var fieldCache struct { + sync.RWMutex + m map[reflect.Type][]field +} + +// cachedTypeFields is like typeFields but uses a cache to avoid repeated work. +func cachedTypeFields(t reflect.Type) []field { + fieldCache.RLock() + f := fieldCache.m[t] + fieldCache.RUnlock() + if f != nil { + return f + } + + // Compute fields without lock. + // Might duplicate effort but won't hold other computations back. + f = typeFields(t) + if f == nil { + f = []field{} + } + + fieldCache.Lock() + if fieldCache.m == nil { + fieldCache.m = map[reflect.Type][]field{} + } + fieldCache.m[t] = f + fieldCache.Unlock() + return f +} diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/decoder.go b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/decoder.go new file mode 100644 index 0000000000000..f50478abb7ed0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/decoder.go @@ -0,0 +1,141 @@ +package encoding + +import ( + "errors" + "reflect" + "runtime" + "sync" +) + +var byteSliceType = reflect.TypeOf([]byte(nil)) + +type decoderFunc func(dv reflect.Value, sv reflect.Value) + +// Decode decodes map[string]interface{} into a struct. The first parameter +// must be a pointer. +func Decode(dst interface{}, src interface{}) (err error) { + defer func() { + if r := recover(); r != nil { + if _, ok := r.(runtime.Error); ok { + panic(r) + } + if v, ok := r.(string); ok { + err = errors.New(v) + } else { + err = r.(error) + } + } + }() + + dv := reflect.ValueOf(dst) + sv := reflect.ValueOf(src) + if dv.Kind() != reflect.Ptr { + return &DecodeTypeError{ + DestType: dv.Type(), + SrcType: sv.Type(), + Reason: "must be a pointer", + } + } + + dv = dv.Elem() + if !dv.CanAddr() { + return &DecodeTypeError{ + DestType: dv.Type(), + SrcType: sv.Type(), + Reason: "must be addressable", + } + } + + decode(dv, sv) + return nil +} + +// decode decodes the source value into the destination value +func decode(dv, sv reflect.Value) { + valueDecoder(dv, sv)(dv, sv) +} + +type decoderCacheKey struct { + dt, st reflect.Type +} + +var decoderCache struct { + sync.RWMutex + m map[decoderCacheKey]decoderFunc +} + +func valueDecoder(dv, sv reflect.Value) decoderFunc { + if !sv.IsValid() { + return invalidValueDecoder + } + + if dv.IsValid() { + dv = indirect(dv, false) + dv.Set(reflect.Zero(dv.Type())) + } + + return typeDecoder(dv.Type(), sv.Type()) +} + +func typeDecoder(dt, st reflect.Type) decoderFunc { + decoderCache.RLock() + f := decoderCache.m[decoderCacheKey{dt, st}] + decoderCache.RUnlock() + if f != nil { + return f + } + + // To deal with recursive types, populate the map with an + // indirect func before we build it. This type waits on the + // real func (f) to be ready and then calls it. This indirect + // func is only used for recursive types. + decoderCache.Lock() + var wg sync.WaitGroup + wg.Add(1) + decoderCache.m[decoderCacheKey{dt, st}] = func(dv, sv reflect.Value) { + wg.Wait() + f(dv, sv) + } + decoderCache.Unlock() + + // Compute fields without lock. + // Might duplicate effort but won't hold other computations back. + f = newTypeDecoder(dt, st) + wg.Done() + decoderCache.Lock() + decoderCache.m[decoderCacheKey{dt, st}] = f + decoderCache.Unlock() + return f +} + +// indirect walks down v allocating pointers as needed, +// until it gets to a non-pointer. +func indirect(v reflect.Value, decodeNull bool) reflect.Value { + // If v is a named type and is addressable, + // start with its address, so that if the type has pointer methods, + // we find them. + if v.Kind() != reflect.Ptr && v.Type().Name() != "" && v.CanAddr() { + v = v.Addr() + } + for { + // Load value from interface, but only if the result will be + // usefully addressable. + if v.Kind() == reflect.Interface && !v.IsNil() { + e := v.Elem() + if e.Kind() == reflect.Ptr && !e.IsNil() && (!decodeNull || e.Elem().Kind() == reflect.Ptr) { + v = e + continue + } + } + + if v.Kind() != reflect.Ptr { + break + } + + if v.IsNil() { + v.Set(reflect.New(v.Type().Elem())) + } + v = v.Elem() + } + return v +} diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/decoder_test.go b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/decoder_test.go new file mode 100644 index 0000000000000..909b2cc2d203a --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/decoder_test.go @@ -0,0 +1,426 @@ +package encoding + +import ( + "bytes" + "encoding/json" + "image" + "reflect" + "testing" +) + +type T struct { + X string + Y int + Z int `gorethink:"-"` +} + +type U struct { + Alphabet string `gorethink:"alpha"` +} + +type V struct { + F1 interface{} + F2 int32 + F3 string +} + +type tx struct { + x int +} + +var txType = reflect.TypeOf((*tx)(nil)).Elem() + +// Test data structures for anonymous fields. + +type Point struct { + Z int +} + +type Top struct { + Level0 int + Embed0 + *Embed0a + *Embed0b `gorethink:"e,omitempty"` // treated as named + Embed0c `gorethink:"-"` // ignored + Loop + Embed0p // has Point with X, Y, used + Embed0q // has Point with Z, used +} + +type Embed0 struct { + Level1a int // overridden by Embed0a's Level1a with tag + Level1b int // used because Embed0a's Level1b is renamed + Level1c int // used because Embed0a's Level1c is ignored + Level1d int // annihilated by Embed0a's Level1d + Level1e int `gorethink:"x"` // annihilated by Embed0a.Level1e +} + +type Embed0a struct { + Level1a int `gorethink:"Level1a,omitempty"` + Level1b int `gorethink:"LEVEL1B,omitempty"` + Level1c int `gorethink:"-"` + Level1d int // annihilated by Embed0's Level1d + Level1f int `gorethink:"x"` // annihilated by Embed0's Level1e +} + +type Embed0b Embed0 + +type Embed0c Embed0 + +type Embed0p struct { + image.Point +} + +type Embed0q struct { + Point +} + +type Loop struct { + Loop1 int `gorethink:",omitempty"` + Loop2 int `gorethink:",omitempty"` + *Loop +} + +// From reflect test: +// The X in S6 and S7 annihilate, but they also block the X in S8.S9. +type S5 struct { + S6 + S7 + S8 +} + +type S6 struct { + X int +} + +type S7 S6 + +type S8 struct { + S9 +} + +type S9 struct { + X int + Y int +} + +// From reflect test: +// The X in S11.S6 and S12.S6 annihilate, but they also block the X in S13.S8.S9. +type S10 struct { + S11 + S12 + S13 +} + +type S11 struct { + S6 +} + +type S12 struct { + S6 +} + +type S13 struct { + S8 +} + +type Pointer struct { + PPoint *Point + Point Point +} + +type decodeTest struct { + in interface{} + ptr interface{} + out interface{} + err error +} + +type Ambig struct { + // Given "hello", the first match should win. + First int `gorethink:"HELLO"` + Second int `gorethink:"Hello"` +} + +var decodeTests = []decodeTest{ + // basic types + {in: true, ptr: new(bool), out: true}, + {in: 1, ptr: new(int), out: 1}, + {in: 1.2, ptr: new(float64), out: 1.2}, + {in: -5, ptr: new(int16), out: int16(-5)}, + {in: 2, ptr: new(string), out: string("2")}, + {in: float64(2.0), ptr: new(interface{}), out: float64(2.0)}, + {in: string("2"), ptr: new(interface{}), out: string("2")}, + {in: "a\u1234", ptr: new(string), out: "a\u1234"}, + {in: []interface{}{}, ptr: new([]string), out: []string{}}, + {in: map[string]interface{}{"X": []interface{}{1, 2, 3}, "Y": 4}, ptr: new(T), out: T{}, err: &DecodeTypeError{reflect.TypeOf(""), reflect.TypeOf([]interface{}{}), ""}}, + {in: map[string]interface{}{"x": 1}, ptr: new(tx), out: tx{}}, + {in: map[string]interface{}{"F1": float64(1), "F2": 2, "F3": 3}, ptr: new(V), out: V{F1: float64(1), F2: int32(2), F3: string("3")}}, + {in: map[string]interface{}{"F1": string("1"), "F2": 2, "F3": 3}, ptr: new(V), out: V{F1: string("1"), F2: int32(2), F3: string("3")}}, + { + in: map[string]interface{}{"k1": int64(1), "k2": "s", "k3": []interface{}{int64(1), 2.0, 3e-3}, "k4": map[string]interface{}{"kk1": "s", "kk2": int64(2)}}, + out: map[string]interface{}{"k1": int64(1), "k2": "s", "k3": []interface{}{int64(1), 2.0, 3e-3}, "k4": map[string]interface{}{"kk1": "s", "kk2": int64(2)}}, + ptr: new(interface{}), + }, + + // Z has a "-" tag. + {in: map[string]interface{}{"Y": 1, "Z": 2}, ptr: new(T), out: T{Y: 1}}, + + {in: map[string]interface{}{"alpha": "abc", "alphabet": "xyz"}, ptr: new(U), out: U{Alphabet: "abc"}}, + {in: map[string]interface{}{"alpha": "abc"}, ptr: new(U), out: U{Alphabet: "abc"}}, + {in: map[string]interface{}{"alphabet": "xyz"}, ptr: new(U), out: U{}}, + + // array tests + {in: []interface{}{1, 2, 3}, ptr: new([3]int), out: [3]int{1, 2, 3}}, + {in: []interface{}{1, 2, 3}, ptr: new([1]int), out: [1]int{1}}, + {in: []interface{}{1, 2, 3}, ptr: new([5]int), out: [5]int{1, 2, 3, 0, 0}}, + + // empty array to interface test + {in: map[string]interface{}{"T": []interface{}{}}, ptr: new(map[string]interface{}), out: map[string]interface{}{"T": []interface{}{}}}, + + { + in: map[string]interface{}{ + "Level0": 1, + "Level1b": 2, + "Level1c": 3, + "level1d": 4, + "Level1a": 5, + "LEVEL1B": 6, + "e": map[string]interface{}{ + "Level1a": 8, + "Level1b": 9, + "Level1c": 10, + "Level1d": 11, + "x": 12, + }, + "Loop1": 13, + "Loop2": 14, + "X": 15, + "Y": 16, + "Z": 17, + }, + ptr: new(Top), + out: Top{ + Level0: 1, + Embed0: Embed0{ + Level1b: 2, + Level1c: 3, + }, + Embed0a: &Embed0a{ + Level1a: 5, + Level1b: 6, + }, + Embed0b: &Embed0b{ + Level1a: 8, + Level1b: 9, + Level1c: 10, + Level1d: 11, + }, + Loop: Loop{ + Loop1: 13, + Loop2: 14, + }, + Embed0p: Embed0p{ + Point: image.Point{X: 15, Y: 16}, + }, + Embed0q: Embed0q{ + Point: Point{Z: 17}, + }, + }, + }, + { + in: map[string]interface{}{"hello": 1}, + ptr: new(Ambig), + out: Ambig{First: 1}, + }, + { + in: map[string]interface{}{"X": 1, "Y": 2}, + ptr: new(S5), + out: S5{S8: S8{S9: S9{Y: 2}}}, + }, + { + in: map[string]interface{}{"X": 1, "Y": 2}, + ptr: new(S10), + out: S10{S13: S13{S8: S8{S9: S9{Y: 2}}}}, + }, + { + in: map[string]interface{}{"PPoint": map[string]interface{}{"Z": 1}, "Point": map[string]interface{}{"Z": 2}}, + ptr: new(Pointer), + out: Pointer{PPoint: &Point{Z: 1}, Point: Point{Z: 2}}, + }, + { + in: map[string]interface{}{"Point": map[string]interface{}{"Z": 2}}, + ptr: new(Pointer), + out: Pointer{PPoint: nil, Point: Point{Z: 2}}, + }, +} + +func TestDecode(t *testing.T) { + for i, tt := range decodeTests { + if tt.ptr == nil { + continue + } + + // v = new(right-type) + v := reflect.New(reflect.TypeOf(tt.ptr).Elem()) + + err := Decode(v.Interface(), tt.in) + if !jsonEqual(err, tt.err) { + t.Errorf("#%d: got error %v want %v", i, err, tt.err) + continue + } + + if tt.err == nil && !jsonEqual(v.Elem().Interface(), tt.out) { + t.Errorf("#%d: mismatch\nhave: %+v\nwant: %+v", i, v.Elem().Interface(), tt.out) + continue + } + + // Check round trip. + if tt.err == nil { + enc, err := Encode(v.Interface()) + if err != nil { + t.Errorf("#%d: error re-marshaling: %v", i, err) + continue + } + vv := reflect.New(reflect.TypeOf(tt.ptr).Elem()) + + if err := Decode(vv.Interface(), enc); err != nil { + t.Errorf("#%d: error re-decodeing: %v", i, err) + continue + } + if !jsonEqual(v.Elem().Interface(), vv.Elem().Interface()) { + t.Errorf("#%d: mismatch\nhave: %#+v\nwant: %#+v", i, v.Elem().Interface(), vv.Elem().Interface()) + continue + } + } + } +} + +func TestStringKind(t *testing.T) { + type aMap map[string]int + + var m1, m2 map[string]int + m1 = map[string]int{ + "foo": 42, + } + + data, err := Encode(m1) + if err != nil { + t.Errorf("Unexpected error encoding: %v", err) + } + + err = Decode(&m2, data) + if err != nil { + t.Errorf("Unexpected error decoding: %v", err) + } + + if !jsonEqual(m1, m2) { + t.Error("Items should be equal after encoding and then decoding") + } + +} + +// Test handling of unexported fields that should be ignored. +type unexportedFields struct { + Name string + m map[string]interface{} `gorethink:"-"` + m2 map[string]interface{} `gorethink:"abcd"` +} + +func TestDecodeUnexported(t *testing.T) { + input := map[string]interface{}{ + "Name": "Bob", + "m": map[string]interface{}{ + "x": 123, + }, + "m2": map[string]interface{}{ + "y": 123, + }, + "abcd": map[string]interface{}{ + "z": 789, + }, + } + want := &unexportedFields{Name: "Bob"} + + out := &unexportedFields{} + err := Decode(out, input) + if err != nil { + t.Errorf("got error %v, expected nil", err) + } + if !jsonEqual(out, want) { + t.Errorf("got %q, want %q", out, want) + } +} + +type Foo struct { + FooBar interface{} `gorethink:"foobar"` +} +type Bar struct { + Baz int `gorethink:"baz"` +} + +type UnmarshalerPointer struct { + Value *UnmarshalerValue +} + +type UnmarshalerValue struct { + ValueInt int64 + ValueString string +} + +func (v *UnmarshalerValue) MarshalRQL() (interface{}, error) { + if v.ValueInt != int64(0) { + return Encode(v.ValueInt) + } + if v.ValueString != "" { + return Encode(v.ValueString) + } + + return Encode(nil) +} + +func (v *UnmarshalerValue) UnmarshalRQL(b interface{}) (err error) { + n, s := int64(0), "" + + if err = Decode(&s, b); err == nil { + v.ValueString = s + return + } + if err = Decode(&n, b); err == nil { + v.ValueInt = n + + } + + return +} + +func TestDecodeUnmarshalerPointer(t *testing.T) { + input := map[string]interface{}{ + "Value": "abc", + } + want := &UnmarshalerPointer{ + Value: &UnmarshalerValue{ValueString: "abc"}, + } + + out := &UnmarshalerPointer{} + err := Decode(out, input) + if err != nil { + t.Errorf("got error %v, expected nil", err) + } + if !jsonEqual(out, want) { + t.Errorf("got %q, want %q", out, want) + } +} + +func jsonEqual(a, b interface{}) bool { + ba, err := json.Marshal(a) + if err != nil { + panic(err) + } + bb, err := json.Marshal(b) + if err != nil { + panic(err) + } + + return bytes.Compare(ba, bb) == 0 +} diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/decoder_types.go b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/decoder_types.go new file mode 100644 index 0000000000000..61d268f83e571 --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/decoder_types.go @@ -0,0 +1,524 @@ +package encoding + +import ( + "bytes" + "fmt" + "reflect" + "strconv" +) + +// newTypeDecoder constructs an decoderFunc for a type. +func newTypeDecoder(dt, st reflect.Type) decoderFunc { + if reflect.PtrTo(dt).Implements(unmarshalerType) || + dt.Implements(unmarshalerType) { + return unmarshalerDecoder + } + + if st.Kind() == reflect.Interface { + return interfaceAsTypeDecoder + } + + switch dt.Kind() { + case reflect.Bool: + switch st.Kind() { + case reflect.Bool: + return boolAsBoolDecoder + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return intAsBoolDecoder + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + return uintAsBoolDecoder + case reflect.Float32, reflect.Float64: + return floatAsBoolDecoder + case reflect.String: + return stringAsBoolDecoder + default: + return decodeTypeError + } + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + switch st.Kind() { + case reflect.Bool: + return boolAsIntDecoder + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return intAsIntDecoder + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + return uintAsIntDecoder + case reflect.Float32, reflect.Float64: + return floatAsIntDecoder + case reflect.String: + return stringAsIntDecoder + default: + return decodeTypeError + } + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + switch st.Kind() { + case reflect.Bool: + return boolAsUintDecoder + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return intAsUintDecoder + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + return uintAsUintDecoder + case reflect.Float32, reflect.Float64: + return floatAsUintDecoder + case reflect.String: + return stringAsUintDecoder + default: + return decodeTypeError + } + case reflect.Float32, reflect.Float64: + switch st.Kind() { + case reflect.Bool: + return boolAsFloatDecoder + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return intAsFloatDecoder + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + return uintAsFloatDecoder + case reflect.Float32, reflect.Float64: + return floatAsFloatDecoder + case reflect.String: + return stringAsFloatDecoder + default: + return decodeTypeError + } + case reflect.String: + switch st.Kind() { + case reflect.Bool: + return boolAsStringDecoder + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return intAsStringDecoder + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + return uintAsStringDecoder + case reflect.Float32, reflect.Float64: + return floatAsStringDecoder + case reflect.String: + return stringAsStringDecoder + default: + return decodeTypeError + } + case reflect.Interface: + if !st.AssignableTo(dt) { + return decodeTypeError + } + + return interfaceDecoder + case reflect.Ptr: + return newPtrDecoder(dt, st) + case reflect.Map: + if st.AssignableTo(dt) { + return interfaceDecoder + } + + switch st.Kind() { + case reflect.Map: + return newMapAsMapDecoder(dt, st) + default: + return decodeTypeError + } + case reflect.Struct: + if st.AssignableTo(dt) { + return interfaceDecoder + } + + switch st.Kind() { + case reflect.Map: + if kind := st.Key().Kind(); kind != reflect.String && kind != reflect.Interface { + return newDecodeTypeError(fmt.Errorf("map needs string keys")) + } + + return newMapAsStructDecoder(dt, st) + default: + return decodeTypeError + } + case reflect.Slice: + if st.AssignableTo(dt) { + return interfaceDecoder + } + + switch st.Kind() { + case reflect.Array, reflect.Slice: + return newSliceDecoder(dt, st) + default: + return decodeTypeError + } + case reflect.Array: + if st.AssignableTo(dt) { + return interfaceDecoder + } + + switch st.Kind() { + case reflect.Array, reflect.Slice: + return newArrayDecoder(dt, st) + default: + return decodeTypeError + } + default: + return unsupportedTypeDecoder + } +} + +func invalidValueDecoder(dv, sv reflect.Value) { + dv.Set(reflect.Zero(dv.Type())) +} + +func unsupportedTypeDecoder(dv, sv reflect.Value) { + panic(&UnsupportedTypeError{dv.Type()}) +} + +func decodeTypeError(dv, sv reflect.Value) { + panic(&DecodeTypeError{ + DestType: dv.Type(), + SrcType: sv.Type(), + }) +} + +func newDecodeTypeError(err error) decoderFunc { + return func(dv, sv reflect.Value) { + panic(&DecodeTypeError{ + DestType: dv.Type(), + SrcType: sv.Type(), + Reason: err.Error(), + }) + } +} + +func interfaceDecoder(dv, sv reflect.Value) { + dv.Set(sv) +} + +func interfaceAsTypeDecoder(dv, sv reflect.Value) { + decode(dv, sv.Elem()) +} + +type ptrDecoder struct { + elemDec decoderFunc +} + +func (d *ptrDecoder) decode(dv, sv reflect.Value) { + v := reflect.New(dv.Type().Elem()) + d.elemDec(v, sv) + dv.Set(v) +} + +func newPtrDecoder(dt, st reflect.Type) decoderFunc { + dec := &ptrDecoder{typeDecoder(dt.Elem(), st)} + + return dec.decode +} + +func unmarshalerDecoder(dv, sv reflect.Value) { + // modeled off of https://golang.org/src/encoding/json/decode.go?#L325 + if dv.Kind() != reflect.Ptr && dv.Type().Name() != "" && dv.CanAddr() { + dv = dv.Addr() + } + + if dv.IsNil() { + dv.Set(reflect.New(dv.Type().Elem())) + } + + u := dv.Interface().(Unmarshaler) + err := u.UnmarshalRQL(sv.Interface()) + if err != nil { + panic(&DecodeTypeError{dv.Type(), sv.Type(), err.Error()}) + } +} + +// Boolean decoders + +func boolAsBoolDecoder(dv, sv reflect.Value) { + dv.SetBool(sv.Bool()) +} +func boolAsIntDecoder(dv, sv reflect.Value) { + if sv.Bool() { + dv.SetInt(1) + } else { + dv.SetInt(0) + } +} +func boolAsUintDecoder(dv, sv reflect.Value) { + if sv.Bool() { + dv.SetUint(1) + } else { + dv.SetUint(0) + } +} +func boolAsFloatDecoder(dv, sv reflect.Value) { + if sv.Bool() { + dv.SetFloat(1) + } else { + dv.SetFloat(0) + } +} +func boolAsStringDecoder(dv, sv reflect.Value) { + if sv.Bool() { + dv.SetString("1") + } else { + dv.SetString("0") + } +} + +// Int decoders + +func intAsBoolDecoder(dv, sv reflect.Value) { + dv.SetBool(sv.Int() != 0) +} +func intAsIntDecoder(dv, sv reflect.Value) { + dv.SetInt(sv.Int()) +} +func intAsUintDecoder(dv, sv reflect.Value) { + dv.SetUint(uint64(sv.Int())) +} +func intAsFloatDecoder(dv, sv reflect.Value) { + dv.SetFloat(float64(sv.Int())) +} +func intAsStringDecoder(dv, sv reflect.Value) { + dv.SetString(strconv.FormatInt(sv.Int(), 10)) +} + +// Uint decoders + +func uintAsBoolDecoder(dv, sv reflect.Value) { + dv.SetBool(sv.Uint() != 0) +} +func uintAsIntDecoder(dv, sv reflect.Value) { + dv.SetInt(int64(sv.Uint())) +} +func uintAsUintDecoder(dv, sv reflect.Value) { + dv.SetUint(sv.Uint()) +} +func uintAsFloatDecoder(dv, sv reflect.Value) { + dv.SetFloat(float64(sv.Uint())) +} +func uintAsStringDecoder(dv, sv reflect.Value) { + dv.SetString(strconv.FormatUint(sv.Uint(), 10)) +} + +// Float decoders + +func floatAsBoolDecoder(dv, sv reflect.Value) { + dv.SetBool(sv.Float() != 0) +} +func floatAsIntDecoder(dv, sv reflect.Value) { + dv.SetInt(int64(sv.Float())) +} +func floatAsUintDecoder(dv, sv reflect.Value) { + dv.SetUint(uint64(sv.Float())) +} +func floatAsFloatDecoder(dv, sv reflect.Value) { + dv.SetFloat(float64(sv.Float())) +} +func floatAsStringDecoder(dv, sv reflect.Value) { + dv.SetString(strconv.FormatFloat(sv.Float(), 'f', -1, 64)) +} + +// String decoders + +func stringAsBoolDecoder(dv, sv reflect.Value) { + b, err := strconv.ParseBool(sv.String()) + if err == nil { + dv.SetBool(b) + } else if sv.String() == "" { + dv.SetBool(false) + } else { + panic(&DecodeTypeError{dv.Type(), sv.Type(), err.Error()}) + } +} +func stringAsIntDecoder(dv, sv reflect.Value) { + i, err := strconv.ParseInt(sv.String(), 0, dv.Type().Bits()) + if err == nil { + dv.SetInt(i) + } else { + panic(&DecodeTypeError{dv.Type(), sv.Type(), err.Error()}) + } +} +func stringAsUintDecoder(dv, sv reflect.Value) { + i, err := strconv.ParseUint(sv.String(), 0, dv.Type().Bits()) + if err == nil { + dv.SetUint(i) + } else { + panic(&DecodeTypeError{dv.Type(), sv.Type(), err.Error()}) + } +} +func stringAsFloatDecoder(dv, sv reflect.Value) { + f, err := strconv.ParseFloat(sv.String(), dv.Type().Bits()) + if err == nil { + dv.SetFloat(f) + } else { + panic(&DecodeTypeError{dv.Type(), sv.Type(), err.Error()}) + } +} +func stringAsStringDecoder(dv, sv reflect.Value) { + dv.SetString(sv.String()) +} + +// Slice/Array decoder + +type sliceDecoder struct { + arrayDec decoderFunc +} + +func (d *sliceDecoder) decode(dv, sv reflect.Value) { + if dv.Kind() == reflect.Slice { + dv.Set(reflect.MakeSlice(dv.Type(), dv.Len(), dv.Cap())) + } + + if !sv.IsNil() { + d.arrayDec(dv, sv) + } +} + +func newSliceDecoder(dt, st reflect.Type) decoderFunc { + // Byte slices get special treatment; arrays don't. + // if t.Elem().Kind() == reflect.Uint8 { + // return decodeByteSlice + // } + dec := &sliceDecoder{newArrayDecoder(dt, st)} + return dec.decode +} + +type arrayDecoder struct { + elemDec decoderFunc +} + +func (d *arrayDecoder) decode(dv, sv reflect.Value) { + // Iterate through the slice/array and decode each element before adding it + // to the dest slice/array + i := 0 + for i < sv.Len() { + if dv.Kind() == reflect.Slice { + // Get element of array, growing if necessary. + if i >= dv.Cap() { + newcap := dv.Cap() + dv.Cap()/2 + if newcap < 4 { + newcap = 4 + } + newdv := reflect.MakeSlice(dv.Type(), dv.Len(), newcap) + reflect.Copy(newdv, dv) + dv.Set(newdv) + } + if i >= dv.Len() { + dv.SetLen(i + 1) + } + } + + if i < dv.Len() { + // Decode into element. + d.elemDec(dv.Index(i), sv.Index(i)) + } + + i++ + } + + // Ensure that the destination is the correct size + if i < dv.Len() { + if dv.Kind() == reflect.Array { + // Array. Zero the rest. + z := reflect.Zero(dv.Type().Elem()) + for ; i < dv.Len(); i++ { + dv.Index(i).Set(z) + } + } else { + dv.SetLen(i) + } + } +} + +func newArrayDecoder(dt, st reflect.Type) decoderFunc { + dec := &arrayDecoder{typeDecoder(dt.Elem(), st.Elem())} + return dec.decode +} + +// Map decoder + +type mapAsMapDecoder struct { + keyDec, elemDec decoderFunc +} + +func (d *mapAsMapDecoder) decode(dv, sv reflect.Value) { + dt := dv.Type() + dv.Set(reflect.MakeMap(reflect.MapOf(dt.Key(), dt.Elem()))) + + var mapKey reflect.Value + var mapElem reflect.Value + + keyType := dv.Type().Key() + elemType := dv.Type().Elem() + + for _, sElemKey := range sv.MapKeys() { + var dElemKey reflect.Value + var dElemVal reflect.Value + + if !mapKey.IsValid() { + mapKey = reflect.New(keyType).Elem() + } else { + mapKey.Set(reflect.Zero(keyType)) + } + dElemKey = mapKey + + if !mapElem.IsValid() { + mapElem = reflect.New(elemType).Elem() + } else { + mapElem.Set(reflect.Zero(elemType)) + } + dElemVal = mapElem + + d.keyDec(dElemKey, sElemKey) + d.elemDec(dElemVal, sv.MapIndex(sElemKey)) + + dv.SetMapIndex(dElemKey, dElemVal) + } +} + +func newMapAsMapDecoder(dt, st reflect.Type) decoderFunc { + d := &mapAsMapDecoder{typeDecoder(dt.Key(), st.Key()), typeDecoder(dt.Elem(), st.Elem())} + return d.decode +} + +type mapAsStructDecoder struct { + fields []field + fieldDecs []decoderFunc +} + +func (d *mapAsStructDecoder) decode(dv, sv reflect.Value) { + for _, kv := range sv.MapKeys() { + var f *field + var fieldDec decoderFunc + key := []byte(kv.String()) + for i := range d.fields { + ff := &d.fields[i] + ffd := d.fieldDecs[i] + if bytes.Equal(ff.nameBytes, key) { + f = ff + fieldDec = ffd + break + } + if f == nil && ff.equalFold(ff.nameBytes, key) { + f = ff + fieldDec = ffd + } + } + + if f == nil { + continue + } + + dElemVal := fieldByIndex(dv, f.index) + sElemVal := sv.MapIndex(kv) + + if !sElemVal.IsValid() || !dElemVal.CanSet() { + continue + } + + fieldDec(dElemVal, sElemVal) + } +} + +func newMapAsStructDecoder(dt, st reflect.Type) decoderFunc { + fields := cachedTypeFields(dt) + se := &mapAsStructDecoder{ + fields: fields, + fieldDecs: make([]decoderFunc, len(fields)), + } + for i, f := range fields { + se.fieldDecs[i] = typeDecoder(typeByIndex(dt, f.index), st.Elem()) + } + return se.decode +} diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/encoder.go b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/encoder.go new file mode 100644 index 0000000000000..3b0d3508deeef --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/encoder.go @@ -0,0 +1,89 @@ +// This code is based on encoding/json and gorilla/schema + +package encoding + +import ( + "errors" + "reflect" + "runtime" + "sync" +) + +type encoderFunc func(v reflect.Value) interface{} + +// Encode returns the encoded value of v. +// +// Encode traverses the value v recursively and looks for structs. If a struct +// is found then it is checked for tagged fields and convert to +// map[string]interface{} +func Encode(v interface{}) (ev interface{}, err error) { + defer func() { + if r := recover(); r != nil { + if _, ok := r.(runtime.Error); ok { + panic(r) + } + if v, ok := r.(string); ok { + err = errors.New(v) + } else { + err = r.(error) + } + } + }() + + return encode(reflect.ValueOf(v)), nil +} + +func encode(v reflect.Value) interface{} { + return valueEncoder(v)(v) +} + +var encoderCache struct { + sync.RWMutex + m map[reflect.Type]encoderFunc +} + +func valueEncoder(v reflect.Value) encoderFunc { + if !v.IsValid() { + return invalidValueEncoder + } + return typeEncoder(v.Type()) +} + +func typeEncoder(t reflect.Type) encoderFunc { + encoderCache.RLock() + f := encoderCache.m[t] + encoderCache.RUnlock() + if f != nil { + return f + } + + // To deal with recursive types, populate the map with an + // indirect func before we build it. This type waits on the + // real func (f) to + // be ready and then calls it. This indirect + // func is only used for recursive types. + encoderCache.Lock() + var wg sync.WaitGroup + wg.Add(1) + encoderCache.m[t] = func(v reflect.Value) interface{} { + wg.Wait() + return f(v) + } + encoderCache.Unlock() + + // Compute fields without lock. + // Might duplicate effort but won't hold other computations back. + f = newTypeEncoder(t, true) + wg.Done() + encoderCache.Lock() + encoderCache.m[t] = f + encoderCache.Unlock() + return f +} + +// IgnoreType causes the encoder to ignore a type when encoding +func IgnoreType(t reflect.Type) { + encoderCache.Lock() + encoderCache.m[t] = doNothingEncoder + encoderCache.Unlock() +} diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/encoder_test.go b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/encoder_test.go new file mode 100644 index 0000000000000..7b1ee0614fab3 --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/encoder_test.go @@ -0,0 +1,262 @@ +package encoding + +import ( + "image" + "reflect" + "testing" + "time" +) + +var encodeExpected = map[string]interface{}{ + "Level0": int64(1), + "Level1b": int64(2), + "Level1c": int64(3), + "Level1a": int64(5), + "LEVEL1B": int64(6), + "e": map[string]interface{}{ + "Level1a": int64(8), + "Level1b": int64(9), + "Level1c": int64(10), + "Level1d": int64(11), + "x": int64(12), + }, + "Loop1": int64(13), + "Loop2": int64(14), + "X": int64(15), + "Y": int64(16), + "Z": int64(17), +} + +func TestEncode(t *testing.T) { + // Top is defined in decoder_test.go + var in = Top{ + Level0: 1, + Embed0: Embed0{ + Level1b: 2, + Level1c: 3, + }, + Embed0a: &Embed0a{ + Level1a: 5, + Level1b: 6, + }, + Embed0b: &Embed0b{ + Level1a: 8, + Level1b: 9, + Level1c: 10, + Level1d: 11, + Level1e: 12, + }, + Loop: Loop{ + Loop1: 13, + Loop2: 14, + }, + Embed0p: Embed0p{ + Point: image.Point{X: 15, Y: 16}, + }, + Embed0q: Embed0q{ + Point: Point{Z: 17}, + }, + } + + got, err := Encode(&in) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(got, encodeExpected) { + t.Errorf(" got: %v\nwant: %v\n", got, encodeExpected) + } +} + +type Optionals struct { + Sr string `gorethink:"sr"` + So string `gorethink:"so,omitempty"` + Sw string `gorethink:"-"` + + Ir int `gorethink:"omitempty"` // actually named omitempty, not an option + Io int `gorethink:"io,omitempty"` + + Tr time.Time `gorethink:"tr"` + To time.Time `gorethink:"to,omitempty"` + + Slr []string `gorethink:"slr"` + Slo []string `gorethink:"slo,omitempty"` + + Mr map[string]interface{} `gorethink:"mr"` + Mo map[string]interface{} `gorethink:",omitempty"` +} + +var optionalsExpected = map[string]interface{}{ + "sr": "", + "omitempty": int64(0), + "tr": map[string]interface{}{"$reql_type$": "TIME", "epoch_time": 0, "timezone": "+00:00"}, + "slr": []interface{}{}, + "mr": map[string]interface{}{}, +} + +func TestOmitEmpty(t *testing.T) { + var o Optionals + o.Sw = "something" + o.Tr = time.Unix(0, 0) + o.Mr = map[string]interface{}{} + o.Mo = map[string]interface{}{} + + got, err := Encode(&o) + if err != nil { + t.Fatal(err) + } + if !jsonEqual(got, optionalsExpected) { + t.Errorf("\ngot: %#v\nwant: %#v\n", got, optionalsExpected) + } +} + +type IntType int + +type MyStruct struct { + IntType +} + +func TestAnonymousNonstruct(t *testing.T) { + var i IntType = 11 + a := MyStruct{i} + var want = map[string]interface{}{"IntType": int64(11)} + + got, err := Encode(a) + if err != nil { + t.Fatalf("Encode: %v", err) + } + if !reflect.DeepEqual(got, want) { + t.Errorf("got %v, want %v", got, want) + } +} + +func TestEncodePointer(t *testing.T) { + v := Pointer{PPoint: &Point{Z: 1}, Point: Point{Z: 2}} + var want = map[string]interface{}{ + "PPoint": map[string]interface{}{"Z": int64(1)}, + "Point": map[string]interface{}{"Z": int64(2)}, + } + + got, err := Encode(v) + if err != nil { + t.Fatalf("Encode: %v", err) + } + if !reflect.DeepEqual(got, want) { + t.Errorf("got %v, want %v", got, want) + } +} + +func TestEncodeNilPointer(t *testing.T) { + v := Pointer{PPoint: nil, Point: Point{Z: 2}} + var want = map[string]interface{}{ + "PPoint": nil, + "Point": map[string]interface{}{"Z": int64(2)}, + } + + got, err := Encode(v) + if err != nil { + t.Fatalf("Encode: %v", err) + } + if !reflect.DeepEqual(got, want) { + t.Errorf("got %v, want %v", got, want) + } +} + +type BugA struct { + S string +} + +type BugB struct { + BugA + S string +} + +type BugC struct { + S string +} + +// Legal Go: We never use the repeated embedded field (S). +type BugX struct { + A int + BugA + BugB +} + +// Issue 5245. +func TestEmbeddedBug(t *testing.T) { + v := BugB{ + BugA{"A"}, + "B", + } + got, err := Encode(v) + if err != nil { + t.Fatal("Encode:", err) + } + want := map[string]interface{}{"S": "B"} + if !reflect.DeepEqual(got, want) { + t.Fatalf("Encode: got %v want %v", got, want) + } + // Now check that the duplicate field, S, does not appear. + x := BugX{ + A: 23, + } + got, err = Encode(x) + if err != nil { + t.Fatal("Encode:", err) + } + want = map[string]interface{}{"A": int64(23)} + if !reflect.DeepEqual(got, want) { + t.Fatalf("Encode: got %v want %v", got, want) + } +} + +type BugD struct { // Same as BugA after tagging. + XXX string `gorethink:"S"` +} + +// BugD's tagged S field should dominate BugA's. +type BugY struct { + BugA + BugD +} + +// Test that a field with a tag dominates untagged fields. +func TestTaggedFieldDominates(t *testing.T) { + v := BugY{ + BugA{"BugA"}, + BugD{"BugD"}, + } + got, err := Encode(v) + if err != nil { + t.Fatal("Encode:", err) + } + want := map[string]interface{}{"S": "BugD"} + if !reflect.DeepEqual(got, want) { + t.Fatalf("Encode: got %v want %v", got, want) + } +} + +// There are no tags here, so S should not appear. +type BugZ struct { + BugA + BugC + BugY // Contains a tagged S field through BugD; should not dominate. +} + +func TestDuplicatedFieldDisappears(t *testing.T) { + v := BugZ{ + BugA{"BugA"}, + BugC{"BugC"}, + BugY{ + BugA{"nested BugA"}, + BugD{"nested BugD"}, + }, + } + got, err := Encode(v) + if err != nil { + t.Fatal("Encode:", err) + } + want := map[string]interface{}{} + if !reflect.DeepEqual(got, want) { + t.Fatalf("Encode: got %v want %v", got, want) + } +} diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/encoder_types.go b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/encoder_types.go new file mode 100644 index 0000000000000..de38a19057c43 --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/encoder_types.go @@ -0,0 +1,303 @@ +package encoding + +import ( + "encoding/base64" + "math" + "reflect" + "time" +) + +// newTypeEncoder constructs an encoderFunc for a type. +// The returned encoder only checks CanAddr when allowAddr is true. +func newTypeEncoder(t reflect.Type, allowAddr bool) encoderFunc { + if t.Implements(marshalerType) { + return marshalerEncoder + } + if t.Kind() != reflect.Ptr && allowAddr { + if reflect.PtrTo(t).Implements(marshalerType) { + return newCondAddrEncoder(addrMarshalerEncoder, newTypeEncoder(t, false)) + } + } + + // Check for psuedo-types first + switch t { + case timeType: + return timePseudoTypeEncoder + } + + switch t.Kind() { + case reflect.Bool: + return boolEncoder + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return intEncoder + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + return uintEncoder + case reflect.Float32, reflect.Float64: + return floatEncoder + case reflect.String: + return stringEncoder + case reflect.Interface: + return interfaceEncoder + case reflect.Struct: + return newStructEncoder(t) + case reflect.Map: + return newMapEncoder(t) + case reflect.Slice: + return newSliceEncoder(t) + case reflect.Array: + return newArrayEncoder(t) + case reflect.Ptr: + return newPtrEncoder(t) + default: + return unsupportedTypeEncoder + } +} + +func invalidValueEncoder(v reflect.Value) interface{} { + return nil +} + +func doNothingEncoder(v reflect.Value) interface{} { + return v.Interface() +} + +func marshalerEncoder(v reflect.Value) interface{} { + if v.Kind() == reflect.Ptr && v.IsNil() { + return nil + } + m := v.Interface().(Marshaler) + ev, err := m.MarshalRQL() + if err != nil { + panic(&MarshalerError{v.Type(), err}) + } + + return ev +} + +func addrMarshalerEncoder(v reflect.Value) interface{} { + va := v.Addr() + if va.IsNil() { + return nil + } + m := va.Interface().(Marshaler) + ev, err := m.MarshalRQL() + if err != nil { + panic(&MarshalerError{v.Type(), err}) + } + + return ev +} + +func boolEncoder(v reflect.Value) interface{} { + if v.Bool() { + return true + } else { + return false + } +} + +func intEncoder(v reflect.Value) interface{} { + return v.Int() +} + +func uintEncoder(v reflect.Value) interface{} { + return v.Uint() +} + +func floatEncoder(v reflect.Value) interface{} { + return v.Float() +} + +func stringEncoder(v reflect.Value) interface{} { + return v.String() +} + +func interfaceEncoder(v reflect.Value) interface{} { + if v.IsNil() { + return nil + } + return encode(v.Elem()) +} + +func unsupportedTypeEncoder(v reflect.Value) interface{} { + panic(&UnsupportedTypeError{v.Type()}) +} + +type structEncoder struct { + fields []field + fieldEncs []encoderFunc +} + +func (se *structEncoder) encode(v reflect.Value) interface{} { + m := make(map[string]interface{}) + + for i, f := range se.fields { + fv := fieldByIndex(v, f.index) + if !fv.IsValid() || f.omitEmpty && se.isEmptyValue(fv) { + continue + } + + m[f.name] = se.fieldEncs[i](fv) + } + + return m +} + +func (se *structEncoder) isEmptyValue(v reflect.Value) bool { + if v.Type() == timeType { + return v.Interface().(time.Time) == time.Time{} + } + + return isEmptyValue(v) +} + +func newStructEncoder(t reflect.Type) encoderFunc { + fields := cachedTypeFields(t) + se := &structEncoder{ + fields: fields, + fieldEncs: make([]encoderFunc, len(fields)), + } + for i, f := range fields { + se.fieldEncs[i] = typeEncoder(typeByIndex(t, f.index)) + } + return se.encode +} + +type mapEncoder struct { + elemEnc encoderFunc +} + +func (me *mapEncoder) encode(v reflect.Value) interface{} { + if v.IsNil() { + return nil + } + + m := make(map[string]interface{}) + + for _, k := range v.MapKeys() { + m[k.String()] = me.elemEnc(v.MapIndex(k)) + } + + return m +} + +func newMapEncoder(t reflect.Type) encoderFunc { + if t.Key().Kind() != reflect.String { + return unsupportedTypeEncoder + } + me := &mapEncoder{typeEncoder(t.Elem())} + return me.encode +} + +// sliceEncoder just wraps an arrayEncoder, checking to make sure the value isn't nil. +type sliceEncoder struct { + arrayEnc encoderFunc +} + +func (se *sliceEncoder) encode(v reflect.Value) interface{} { + if v.IsNil() { + return []interface{}{} + } + return se.arrayEnc(v) +} + +func newSliceEncoder(t reflect.Type) encoderFunc { + // Byte slices get special treatment; arrays don't. + if t.Elem().Kind() == reflect.Uint8 { + return encodeByteSlice + } + enc := &sliceEncoder{newArrayEncoder(t)} + return enc.encode +} + +type arrayEncoder struct { + elemEnc encoderFunc +} + +func (ae *arrayEncoder) encode(v reflect.Value) interface{} { + n := v.Len() + + a := make([]interface{}, n) + for i := 0; i < n; i++ { + a[i] = ae.elemEnc(v.Index(i)) + } + + return a +} + +func newArrayEncoder(t reflect.Type) encoderFunc { + enc := &arrayEncoder{typeEncoder(t.Elem())} + return enc.encode +} + +type ptrEncoder struct { + elemEnc encoderFunc +} + +func (pe *ptrEncoder) encode(v reflect.Value) interface{} { + if v.IsNil() { + return nil + } + return pe.elemEnc(v.Elem()) +} + +func newPtrEncoder(t reflect.Type) encoderFunc { + enc := &ptrEncoder{typeEncoder(t.Elem())} + return enc.encode +} + +type condAddrEncoder struct { + canAddrEnc, elseEnc encoderFunc +} + +func (ce *condAddrEncoder) encode(v reflect.Value) interface{} { + if v.CanAddr() { + return ce.canAddrEnc(v) + } else { + return ce.elseEnc(v) + } +} + +// newCondAddrEncoder returns an encoder that checks whether its value +// CanAddr and delegates to canAddrEnc if so, else to elseEnc. +func newCondAddrEncoder(canAddrEnc, elseEnc encoderFunc) encoderFunc { + enc := &condAddrEncoder{canAddrEnc: canAddrEnc, elseEnc: elseEnc} + return enc.encode +} + +// Pseudo-type encoders + +// Encode a time.Time value to the TIME RQL type +func timePseudoTypeEncoder(v reflect.Value) interface{} { + t := v.Interface().(time.Time) + + timeVal := float64(t.UnixNano()) / float64(time.Second) + + // use seconds-since-epoch precision if time.Time `t` + // is before the oldest nanosecond time + if t.Before(time.Unix(0, math.MinInt64)) { + timeVal = float64(t.Unix()) + } + + return map[string]interface{}{ + "$reql_type$": "TIME", + "epoch_time": timeVal, + "timezone": "+00:00", + } +} + +// Encode a byte slice to the BINARY RQL type +func encodeByteSlice(v reflect.Value) interface{} { + var b []byte + if !v.IsNil() { + b = v.Bytes() + } + + dst := make([]byte, base64.StdEncoding.EncodedLen(len(b))) + base64.StdEncoding.Encode(dst, b) + + return map[string]interface{}{ + "$reql_type$": "BINARY", + "data": string(dst), + } +} diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/encoding.go b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/encoding.go new file mode 100644 index 0000000000000..0169e14481033 --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/encoding.go @@ -0,0 +1,32 @@ +package encoding + +import ( + "reflect" + "time" +) + +var ( + // type constants + stringType = reflect.TypeOf("") + timeType = reflect.TypeOf(new(time.Time)).Elem() + + marshalerType = reflect.TypeOf(new(Marshaler)).Elem() + unmarshalerType = reflect.TypeOf(new(Unmarshaler)).Elem() +) + +// Marshaler is the interface implemented by objects that +// can marshal themselves into a valid RQL psuedo-type. +type Marshaler interface { + MarshalRQL() (interface{}, error) +} + +// Unmarshaler is the interface implemented by objects +// that can unmarshal a psuedo-type object of themselves. +type Unmarshaler interface { + UnmarshalRQL(interface{}) error +} + +func init() { + encoderCache.m = make(map[reflect.Type]encoderFunc) + decoderCache.m = make(map[decoderCacheKey]decoderFunc) +} diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/errors.go b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/errors.go new file mode 100644 index 0000000000000..8b9ac2c524202 --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/errors.go @@ -0,0 +1,102 @@ +package encoding + +import ( + "fmt" + "reflect" + "strings" +) + +type MarshalerError struct { + Type reflect.Type + Err error +} + +func (e *MarshalerError) Error() string { + return "gorethink: error calling MarshalRQL for type " + e.Type.String() + ": " + e.Err.Error() +} + +type InvalidUnmarshalError struct { + Type reflect.Type +} + +func (e *InvalidUnmarshalError) Error() string { + if e.Type == nil { + return "gorethink: UnmarshalRQL(nil)" + } + + if e.Type.Kind() != reflect.Ptr { + return "gorethink: UnmarshalRQL(non-pointer " + e.Type.String() + ")" + } + return "gorethink: UnmarshalRQL(nil " + e.Type.String() + ")" +} + +// An InvalidTypeError describes a value that was +// not appropriate for a value of a specific Go type. +type DecodeTypeError struct { + DestType, SrcType reflect.Type + Reason string +} + +func (e *DecodeTypeError) Error() string { + if e.Reason != "" { + return "gorethink: could not decode type " + e.SrcType.String() + " into Go value of type " + e.DestType.String() + ": " + e.Reason + } else { + return "gorethink: could not decode type " + e.SrcType.String() + " into Go value of type " + e.DestType.String() + + } +} + +// An UnsupportedTypeError is returned by Marshal when attempting +// to encode an unsupported value type. +type UnsupportedTypeError struct { + Type reflect.Type +} + +func (e *UnsupportedTypeError) Error() string { + return "gorethink: unsupported type: " + e.Type.String() +} + +// An UnsupportedTypeError is returned by Marshal when attempting +// to encode an unexpected value type. +type UnexpectedTypeError struct { + DestType, SrcType reflect.Type +} + +func (e *UnexpectedTypeError) Error() string { + return "gorethink: expected type: " + e.DestType.String() + ", got " + e.SrcType.String() +} + +type UnsupportedValueError struct { + Value reflect.Value + Str string +} + +func (e *UnsupportedValueError) Error() string { + return "gorethink: unsupported value: " + e.Str +} + +// Error implements the error interface and can represents multiple +// errors that occur in the course of a single decode. +type Error struct { + Errors []string +} + +func (e *Error) Error() string { + points := make([]string, len(e.Errors)) + for i, err := range e.Errors { + points[i] = fmt.Sprintf("* %s", err) + } + + return fmt.Sprintf( + "%d error(s) decoding:\n\n%s", + len(e.Errors), strings.Join(points, "\n")) +} + +func appendErrors(errors []string, err error) []string { + switch e := err.(type) { + case *Error: + return append(errors, e.Errors...) + default: + return append(errors, e.Error()) + } +} diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/fold.go b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/fold.go new file mode 100644 index 0000000000000..21c9e68e499f3 --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/fold.go @@ -0,0 +1,139 @@ +package encoding + +import ( + "bytes" + "unicode/utf8" +) + +const ( + caseMask = ^byte(0x20) // Mask to ignore case in ASCII. + kelvin = '\u212a' + smallLongEss = '\u017f' +) + +// foldFunc returns one of four different case folding equivalence +// functions, from most general (and slow) to fastest: +// +// 1) bytes.EqualFold, if the key s contains any non-ASCII UTF-8 +// 2) equalFoldRight, if s contains special folding ASCII ('k', 'K', 's', 'S') +// 3) asciiEqualFold, no special, but includes non-letters (including _) +// 4) simpleLetterEqualFold, no specials, no non-letters. +// +// The letters S and K are special because they map to 3 runes, not just 2: +// * S maps to s and to U+017F 'ſ' Latin small letter long s +// * k maps to K and to U+212A 'K' Kelvin sign +// See http://play.golang.org/p/tTxjOc0OGo +// +// The returned function is specialized for matching against s and +// should only be given s. It's not curried for performance reasons. +func foldFunc(s []byte) func(s, t []byte) bool { + nonLetter := false + special := false // special letter + for _, b := range s { + if b >= utf8.RuneSelf { + return bytes.EqualFold + } + upper := b & caseMask + if upper < 'A' || upper > 'Z' { + nonLetter = true + } else if upper == 'K' || upper == 'S' { + // See above for why these letters are special. + special = true + } + } + if special { + return equalFoldRight + } + if nonLetter { + return asciiEqualFold + } + return simpleLetterEqualFold +} + +// equalFoldRight is a specialization of bytes.EqualFold when s is +// known to be all ASCII (including punctuation), but contains an 's', +// 'S', 'k', or 'K', requiring a Unicode fold on the bytes in t. +// See comments on foldFunc. +func equalFoldRight(s, t []byte) bool { + for _, sb := range s { + if len(t) == 0 { + return false + } + tb := t[0] + if tb < utf8.RuneSelf { + if sb != tb { + sbUpper := sb & caseMask + if 'A' <= sbUpper && sbUpper <= 'Z' { + if sbUpper != tb&caseMask { + return false + } + } else { + return false + } + } + t = t[1:] + continue + } + // sb is ASCII and t is not. t must be either kelvin + // sign or long s; sb must be s, S, k, or K. + tr, size := utf8.DecodeRune(t) + switch sb { + case 's', 'S': + if tr != smallLongEss { + return false + } + case 'k', 'K': + if tr != kelvin { + return false + } + default: + return false + } + t = t[size:] + + } + if len(t) > 0 { + return false + } + return true +} + +// asciiEqualFold is a specialization of bytes.EqualFold for use when +// s is all ASCII (but may contain non-letters) and contains no +// special-folding letters. +// See comments on foldFunc. +func asciiEqualFold(s, t []byte) bool { + if len(s) != len(t) { + return false + } + for i, sb := range s { + tb := t[i] + if sb == tb { + continue + } + if ('a' <= sb && sb <= 'z') || ('A' <= sb && sb <= 'Z') { + if sb&caseMask != tb&caseMask { + return false + } + } else { + return false + } + } + return true +} + +// simpleLetterEqualFold is a specialization of bytes.EqualFold for +// use when s is all ASCII letters (no underscores, etc) and also +// doesn't contain 'k', 'K', 's', or 'S'. +// See comments on foldFunc. +func simpleLetterEqualFold(s, t []byte) bool { + if len(s) != len(t) { + return false + } + for i, b := range s { + if b&caseMask != t[i]&caseMask { + return false + } + } + return true +} diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/tags.go b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/tags.go new file mode 100644 index 0000000000000..cea3edaf19fa6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/tags.go @@ -0,0 +1,69 @@ +// This code is based on encoding/json and gorilla/schema + +package encoding + +import ( + "reflect" + "strings" + "unicode" +) + +const TagName = "gorethink" + +// tagOptions is the string following a comma in a struct field's +// tag, or the empty string. It does not include the leading comma. +type tagOptions string + +func getTag(sf reflect.StructField) string { + return sf.Tag.Get(TagName) +} + +// parseTag splits a struct field's tag into its name and +// comma-separated options. +func parseTag(tag string) (string, tagOptions) { + if idx := strings.Index(tag, ","); idx != -1 { + return tag[:idx], tagOptions(tag[idx+1:]) + } + return tag, tagOptions("") +} + +func isValidTag(s string) bool { + if s == "" { + return false + } + for _, c := range s { + switch { + case strings.ContainsRune("!#$%&()*+-./:<=>?@[]^_{|}~ ", c): + // Backslash and quote chars are reserved, but + // otherwise any punctuation chars are allowed + // in a tag name. + default: + if !unicode.IsLetter(c) && !unicode.IsDigit(c) { + return false + } + } + } + return true +} + +// Contains returns whether checks that a comma-separated list of options +// contains a particular substr flag. substr must be surrounded by a +// string boundary or commas. +func (o tagOptions) Contains(optionName string) bool { + if len(o) == 0 { + return false + } + s := string(o) + for s != "" { + var next string + i := strings.Index(s, ",") + if i >= 0 { + s, next = s[:i], s[i+1:] + } + if s == optionName { + return true + } + s = next + } + return false +} diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/utils.go b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/utils.go new file mode 100644 index 0000000000000..0ca2c7734c765 --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/encoding/utils.go @@ -0,0 +1,72 @@ +package encoding + +import "reflect" + +func getTypeKind(t reflect.Type) reflect.Kind { + kind := t.Kind() + + switch { + case kind >= reflect.Int && kind <= reflect.Int64: + return reflect.Int + case kind >= reflect.Uint && kind <= reflect.Uint64: + return reflect.Uint + case kind >= reflect.Float32 && kind <= reflect.Float64: + return reflect.Float32 + default: + return kind + } +} + +func isEmptyValue(v reflect.Value) bool { + switch v.Kind() { + case reflect.Array, reflect.Map, reflect.Slice, reflect.String: + return v.Len() == 0 + case reflect.Bool: + return !v.Bool() + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return v.Int() == 0 + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + return v.Uint() == 0 + case reflect.Float32, reflect.Float64: + return v.Float() == 0 + case reflect.Interface, reflect.Ptr: + return v.IsNil() + } + return false +} + +func fieldByIndex(v reflect.Value, index []int) reflect.Value { + for _, i := range index { + if v.Kind() == reflect.Ptr { + if v.IsNil() { + v.Set(reflect.New(v.Type().Elem())) + } + v = v.Elem() + } + v = v.Field(i) + } + + return v +} + +func typeByIndex(t reflect.Type, index []int) reflect.Type { + for _, i := range index { + if t.Kind() == reflect.Ptr { + t = t.Elem() + } + t = t.Field(i).Type + } + return t +} + +// valueByString sorts reflect.Value by the string value, this is useful for +// sorting the result of MapKeys +type valueByString []reflect.Value + +func (x valueByString) Len() int { return len(x) } + +func (x valueByString) Swap(i, j int) { x[i], x[j] = x[j], x[i] } + +func (x valueByString) Less(i, j int) bool { + return x[i].String() < x[j].String() +} diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/ql2/ql2.pb.go b/Godeps/_workspace/src/github.com/dancannon/gorethink/ql2/ql2.pb.go new file mode 100644 index 0000000000000..54ac15198be04 --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/ql2/ql2.pb.go @@ -0,0 +1,1243 @@ +// Code generated by protoc-gen-go. +// source: ql2.proto +// DO NOT EDIT! + +package ql2 + +import proto "github.com/golang/protobuf/proto" +import json "encoding/json" +import math "math" + +// Reference proto, json, and math imports to suppress error if they are not otherwise used. +var _ = proto.Marshal +var _ = &json.SyntaxError{} +var _ = math.Inf + +type VersionDummy_Version int32 + +const ( + VersionDummy_V0_1 VersionDummy_Version = 1063369270 + VersionDummy_V0_2 VersionDummy_Version = 1915781601 + VersionDummy_V0_3 VersionDummy_Version = 1601562686 + VersionDummy_V0_4 VersionDummy_Version = 1074539808 +) + +var VersionDummy_Version_name = map[int32]string{ + 1063369270: "V0_1", + 1915781601: "V0_2", + 1601562686: "V0_3", + 1074539808: "V0_4", +} +var VersionDummy_Version_value = map[string]int32{ + "V0_1": 1063369270, + "V0_2": 1915781601, + "V0_3": 1601562686, + "V0_4": 1074539808, +} + +func (x VersionDummy_Version) Enum() *VersionDummy_Version { + p := new(VersionDummy_Version) + *p = x + return p +} +func (x VersionDummy_Version) String() string { + return proto.EnumName(VersionDummy_Version_name, int32(x)) +} +func (x VersionDummy_Version) MarshalJSON() ([]byte, error) { + return json.Marshal(x.String()) +} +func (x *VersionDummy_Version) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(VersionDummy_Version_value, data, "VersionDummy_Version") + if err != nil { + return err + } + *x = VersionDummy_Version(value) + return nil +} + +type VersionDummy_Protocol int32 + +const ( + VersionDummy_PROTOBUF VersionDummy_Protocol = 656407617 + VersionDummy_JSON VersionDummy_Protocol = 2120839367 +) + +var VersionDummy_Protocol_name = map[int32]string{ + 656407617: "PROTOBUF", + 2120839367: "JSON", +} +var VersionDummy_Protocol_value = map[string]int32{ + "PROTOBUF": 656407617, + "JSON": 2120839367, +} + +func (x VersionDummy_Protocol) Enum() *VersionDummy_Protocol { + p := new(VersionDummy_Protocol) + *p = x + return p +} +func (x VersionDummy_Protocol) String() string { + return proto.EnumName(VersionDummy_Protocol_name, int32(x)) +} +func (x VersionDummy_Protocol) MarshalJSON() ([]byte, error) { + return json.Marshal(x.String()) +} +func (x *VersionDummy_Protocol) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(VersionDummy_Protocol_value, data, "VersionDummy_Protocol") + if err != nil { + return err + } + *x = VersionDummy_Protocol(value) + return nil +} + +type Query_QueryType int32 + +const ( + Query_START Query_QueryType = 1 + Query_CONTINUE Query_QueryType = 2 + Query_STOP Query_QueryType = 3 + Query_NOREPLY_WAIT Query_QueryType = 4 +) + +var Query_QueryType_name = map[int32]string{ + 1: "START", + 2: "CONTINUE", + 3: "STOP", + 4: "NOREPLY_WAIT", +} +var Query_QueryType_value = map[string]int32{ + "START": 1, + "CONTINUE": 2, + "STOP": 3, + "NOREPLY_WAIT": 4, +} + +func (x Query_QueryType) Enum() *Query_QueryType { + p := new(Query_QueryType) + *p = x + return p +} +func (x Query_QueryType) String() string { + return proto.EnumName(Query_QueryType_name, int32(x)) +} +func (x Query_QueryType) MarshalJSON() ([]byte, error) { + return json.Marshal(x.String()) +} +func (x *Query_QueryType) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(Query_QueryType_value, data, "Query_QueryType") + if err != nil { + return err + } + *x = Query_QueryType(value) + return nil +} + +type Frame_FrameType int32 + +const ( + Frame_POS Frame_FrameType = 1 + Frame_OPT Frame_FrameType = 2 +) + +var Frame_FrameType_name = map[int32]string{ + 1: "POS", + 2: "OPT", +} +var Frame_FrameType_value = map[string]int32{ + "POS": 1, + "OPT": 2, +} + +func (x Frame_FrameType) Enum() *Frame_FrameType { + p := new(Frame_FrameType) + *p = x + return p +} +func (x Frame_FrameType) String() string { + return proto.EnumName(Frame_FrameType_name, int32(x)) +} +func (x Frame_FrameType) MarshalJSON() ([]byte, error) { + return json.Marshal(x.String()) +} +func (x *Frame_FrameType) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(Frame_FrameType_value, data, "Frame_FrameType") + if err != nil { + return err + } + *x = Frame_FrameType(value) + return nil +} + +type Response_ResponseType int32 + +const ( + Response_SUCCESS_ATOM Response_ResponseType = 1 + Response_SUCCESS_SEQUENCE Response_ResponseType = 2 + Response_SUCCESS_PARTIAL Response_ResponseType = 3 + Response_WAIT_COMPLETE Response_ResponseType = 4 + Response_CLIENT_ERROR Response_ResponseType = 16 + Response_COMPILE_ERROR Response_ResponseType = 17 + Response_RUNTIME_ERROR Response_ResponseType = 18 +) + +var Response_ResponseType_name = map[int32]string{ + 1: "SUCCESS_ATOM", + 2: "SUCCESS_SEQUENCE", + 3: "SUCCESS_PARTIAL", + 4: "WAIT_COMPLETE", + 16: "CLIENT_ERROR", + 17: "COMPILE_ERROR", + 18: "RUNTIME_ERROR", +} +var Response_ResponseType_value = map[string]int32{ + "SUCCESS_ATOM": 1, + "SUCCESS_SEQUENCE": 2, + "SUCCESS_PARTIAL": 3, + "WAIT_COMPLETE": 4, + "CLIENT_ERROR": 16, + "COMPILE_ERROR": 17, + "RUNTIME_ERROR": 18, +} + +func (x Response_ResponseType) Enum() *Response_ResponseType { + p := new(Response_ResponseType) + *p = x + return p +} +func (x Response_ResponseType) String() string { + return proto.EnumName(Response_ResponseType_name, int32(x)) +} +func (x Response_ResponseType) MarshalJSON() ([]byte, error) { + return json.Marshal(x.String()) +} +func (x *Response_ResponseType) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(Response_ResponseType_value, data, "Response_ResponseType") + if err != nil { + return err + } + *x = Response_ResponseType(value) + return nil +} + +type Response_ResponseNote int32 + +const ( + Response_SEQUENCE_FEED Response_ResponseNote = 1 + Response_ATOM_FEED Response_ResponseNote = 2 + Response_ORDER_BY_LIMIT_FEED Response_ResponseNote = 3 + Response_UNIONED_FEED Response_ResponseNote = 4 + Response_INCLUDES_STATES Response_ResponseNote = 5 +) + +var Response_ResponseNote_name = map[int32]string{ + 1: "SEQUENCE_FEED", + 2: "ATOM_FEED", + 3: "ORDER_BY_LIMIT_FEED", + 4: "UNIONED_FEED", + 5: "INCLUDES_STATES", +} +var Response_ResponseNote_value = map[string]int32{ + "SEQUENCE_FEED": 1, + "ATOM_FEED": 2, + "ORDER_BY_LIMIT_FEED": 3, + "UNIONED_FEED": 4, + "INCLUDES_STATES": 5, +} + +func (x Response_ResponseNote) Enum() *Response_ResponseNote { + p := new(Response_ResponseNote) + *p = x + return p +} +func (x Response_ResponseNote) String() string { + return proto.EnumName(Response_ResponseNote_name, int32(x)) +} +func (x Response_ResponseNote) MarshalJSON() ([]byte, error) { + return json.Marshal(x.String()) +} +func (x *Response_ResponseNote) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(Response_ResponseNote_value, data, "Response_ResponseNote") + if err != nil { + return err + } + *x = Response_ResponseNote(value) + return nil +} + +type Datum_DatumType int32 + +const ( + Datum_R_NULL Datum_DatumType = 1 + Datum_R_BOOL Datum_DatumType = 2 + Datum_R_NUM Datum_DatumType = 3 + Datum_R_STR Datum_DatumType = 4 + Datum_R_ARRAY Datum_DatumType = 5 + Datum_R_OBJECT Datum_DatumType = 6 + Datum_R_JSON Datum_DatumType = 7 +) + +var Datum_DatumType_name = map[int32]string{ + 1: "R_NULL", + 2: "R_BOOL", + 3: "R_NUM", + 4: "R_STR", + 5: "R_ARRAY", + 6: "R_OBJECT", + 7: "R_JSON", +} +var Datum_DatumType_value = map[string]int32{ + "R_NULL": 1, + "R_BOOL": 2, + "R_NUM": 3, + "R_STR": 4, + "R_ARRAY": 5, + "R_OBJECT": 6, + "R_JSON": 7, +} + +func (x Datum_DatumType) Enum() *Datum_DatumType { + p := new(Datum_DatumType) + *p = x + return p +} +func (x Datum_DatumType) String() string { + return proto.EnumName(Datum_DatumType_name, int32(x)) +} +func (x Datum_DatumType) MarshalJSON() ([]byte, error) { + return json.Marshal(x.String()) +} +func (x *Datum_DatumType) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(Datum_DatumType_value, data, "Datum_DatumType") + if err != nil { + return err + } + *x = Datum_DatumType(value) + return nil +} + +type Term_TermType int32 + +const ( + Term_DATUM Term_TermType = 1 + Term_MAKE_ARRAY Term_TermType = 2 + Term_MAKE_OBJ Term_TermType = 3 + Term_VAR Term_TermType = 10 + Term_JAVASCRIPT Term_TermType = 11 + Term_UUID Term_TermType = 169 + Term_HTTP Term_TermType = 153 + Term_ERROR Term_TermType = 12 + Term_IMPLICIT_VAR Term_TermType = 13 + Term_DB Term_TermType = 14 + Term_TABLE Term_TermType = 15 + Term_GET Term_TermType = 16 + Term_GET_ALL Term_TermType = 78 + Term_EQ Term_TermType = 17 + Term_NE Term_TermType = 18 + Term_LT Term_TermType = 19 + Term_LE Term_TermType = 20 + Term_GT Term_TermType = 21 + Term_GE Term_TermType = 22 + Term_NOT Term_TermType = 23 + Term_ADD Term_TermType = 24 + Term_SUB Term_TermType = 25 + Term_MUL Term_TermType = 26 + Term_DIV Term_TermType = 27 + Term_MOD Term_TermType = 28 + Term_APPEND Term_TermType = 29 + Term_PREPEND Term_TermType = 80 + Term_DIFFERENCE Term_TermType = 95 + Term_SET_INSERT Term_TermType = 88 + Term_SET_INTERSECTION Term_TermType = 89 + Term_SET_UNION Term_TermType = 90 + Term_SET_DIFFERENCE Term_TermType = 91 + Term_SLICE Term_TermType = 30 + Term_SKIP Term_TermType = 70 + Term_LIMIT Term_TermType = 71 + Term_OFFSETS_OF Term_TermType = 87 + Term_CONTAINS Term_TermType = 93 + Term_GET_FIELD Term_TermType = 31 + Term_KEYS Term_TermType = 94 + Term_OBJECT Term_TermType = 143 + Term_HAS_FIELDS Term_TermType = 32 + Term_WITH_FIELDS Term_TermType = 96 + Term_PLUCK Term_TermType = 33 + Term_WITHOUT Term_TermType = 34 + Term_MERGE Term_TermType = 35 + Term_BETWEEN_DEPRECATED Term_TermType = 36 + Term_BETWEEN Term_TermType = 182 + Term_REDUCE Term_TermType = 37 + Term_MAP Term_TermType = 38 + Term_FILTER Term_TermType = 39 + Term_CONCAT_MAP Term_TermType = 40 + Term_ORDER_BY Term_TermType = 41 + Term_DISTINCT Term_TermType = 42 + Term_COUNT Term_TermType = 43 + Term_IS_EMPTY Term_TermType = 86 + Term_UNION Term_TermType = 44 + Term_NTH Term_TermType = 45 + Term_BRACKET Term_TermType = 170 + Term_INNER_JOIN Term_TermType = 48 + Term_OUTER_JOIN Term_TermType = 49 + Term_EQ_JOIN Term_TermType = 50 + Term_ZIP Term_TermType = 72 + Term_RANGE Term_TermType = 173 + Term_INSERT_AT Term_TermType = 82 + Term_DELETE_AT Term_TermType = 83 + Term_CHANGE_AT Term_TermType = 84 + Term_SPLICE_AT Term_TermType = 85 + Term_COERCE_TO Term_TermType = 51 + Term_TYPE_OF Term_TermType = 52 + Term_UPDATE Term_TermType = 53 + Term_DELETE Term_TermType = 54 + Term_REPLACE Term_TermType = 55 + Term_INSERT Term_TermType = 56 + Term_DB_CREATE Term_TermType = 57 + Term_DB_DROP Term_TermType = 58 + Term_DB_LIST Term_TermType = 59 + Term_TABLE_CREATE Term_TermType = 60 + Term_TABLE_DROP Term_TermType = 61 + Term_TABLE_LIST Term_TermType = 62 + Term_CONFIG Term_TermType = 174 + Term_STATUS Term_TermType = 175 + Term_WAIT Term_TermType = 177 + Term_RECONFIGURE Term_TermType = 176 + Term_REBALANCE Term_TermType = 179 + Term_SYNC Term_TermType = 138 + Term_INDEX_CREATE Term_TermType = 75 + Term_INDEX_DROP Term_TermType = 76 + Term_INDEX_LIST Term_TermType = 77 + Term_INDEX_STATUS Term_TermType = 139 + Term_INDEX_WAIT Term_TermType = 140 + Term_INDEX_RENAME Term_TermType = 156 + Term_FUNCALL Term_TermType = 64 + Term_BRANCH Term_TermType = 65 + Term_OR Term_TermType = 66 + Term_AND Term_TermType = 67 + Term_FOR_EACH Term_TermType = 68 + Term_FUNC Term_TermType = 69 + Term_ASC Term_TermType = 73 + Term_DESC Term_TermType = 74 + Term_INFO Term_TermType = 79 + Term_MATCH Term_TermType = 97 + Term_UPCASE Term_TermType = 141 + Term_DOWNCASE Term_TermType = 142 + Term_SAMPLE Term_TermType = 81 + Term_DEFAULT Term_TermType = 92 + Term_JSON Term_TermType = 98 + Term_TO_JSON_STRING Term_TermType = 172 + Term_ISO8601 Term_TermType = 99 + Term_TO_ISO8601 Term_TermType = 100 + Term_EPOCH_TIME Term_TermType = 101 + Term_TO_EPOCH_TIME Term_TermType = 102 + Term_NOW Term_TermType = 103 + Term_IN_TIMEZONE Term_TermType = 104 + Term_DURING Term_TermType = 105 + Term_DATE Term_TermType = 106 + Term_TIME_OF_DAY Term_TermType = 126 + Term_TIMEZONE Term_TermType = 127 + Term_YEAR Term_TermType = 128 + Term_MONTH Term_TermType = 129 + Term_DAY Term_TermType = 130 + Term_DAY_OF_WEEK Term_TermType = 131 + Term_DAY_OF_YEAR Term_TermType = 132 + Term_HOURS Term_TermType = 133 + Term_MINUTES Term_TermType = 134 + Term_SECONDS Term_TermType = 135 + Term_TIME Term_TermType = 136 + Term_MONDAY Term_TermType = 107 + Term_TUESDAY Term_TermType = 108 + Term_WEDNESDAY Term_TermType = 109 + Term_THURSDAY Term_TermType = 110 + Term_FRIDAY Term_TermType = 111 + Term_SATURDAY Term_TermType = 112 + Term_SUNDAY Term_TermType = 113 + Term_JANUARY Term_TermType = 114 + Term_FEBRUARY Term_TermType = 115 + Term_MARCH Term_TermType = 116 + Term_APRIL Term_TermType = 117 + Term_MAY Term_TermType = 118 + Term_JUNE Term_TermType = 119 + Term_JULY Term_TermType = 120 + Term_AUGUST Term_TermType = 121 + Term_SEPTEMBER Term_TermType = 122 + Term_OCTOBER Term_TermType = 123 + Term_NOVEMBER Term_TermType = 124 + Term_DECEMBER Term_TermType = 125 + Term_LITERAL Term_TermType = 137 + Term_GROUP Term_TermType = 144 + Term_SUM Term_TermType = 145 + Term_AVG Term_TermType = 146 + Term_MIN Term_TermType = 147 + Term_MAX Term_TermType = 148 + Term_SPLIT Term_TermType = 149 + Term_UNGROUP Term_TermType = 150 + Term_RANDOM Term_TermType = 151 + Term_CHANGES Term_TermType = 152 + Term_ARGS Term_TermType = 154 + Term_BINARY Term_TermType = 155 + Term_GEOJSON Term_TermType = 157 + Term_TO_GEOJSON Term_TermType = 158 + Term_POINT Term_TermType = 159 + Term_LINE Term_TermType = 160 + Term_POLYGON Term_TermType = 161 + Term_DISTANCE Term_TermType = 162 + Term_INTERSECTS Term_TermType = 163 + Term_INCLUDES Term_TermType = 164 + Term_CIRCLE Term_TermType = 165 + Term_GET_INTERSECTING Term_TermType = 166 + Term_FILL Term_TermType = 167 + Term_GET_NEAREST Term_TermType = 168 + Term_POLYGON_SUB Term_TermType = 171 + Term_MINVAL Term_TermType = 180 + Term_MAXVAL Term_TermType = 181 +) + +var Term_TermType_name = map[int32]string{ + 1: "DATUM", + 2: "MAKE_ARRAY", + 3: "MAKE_OBJ", + 10: "VAR", + 11: "JAVASCRIPT", + 169: "UUID", + 153: "HTTP", + 12: "ERROR", + 13: "IMPLICIT_VAR", + 14: "DB", + 15: "TABLE", + 16: "GET", + 78: "GET_ALL", + 17: "EQ", + 18: "NE", + 19: "LT", + 20: "LE", + 21: "GT", + 22: "GE", + 23: "NOT", + 24: "ADD", + 25: "SUB", + 26: "MUL", + 27: "DIV", + 28: "MOD", + 29: "APPEND", + 80: "PREPEND", + 95: "DIFFERENCE", + 88: "SET_INSERT", + 89: "SET_INTERSECTION", + 90: "SET_UNION", + 91: "SET_DIFFERENCE", + 30: "SLICE", + 70: "SKIP", + 71: "LIMIT", + 87: "OFFSETS_OF", + 93: "CONTAINS", + 31: "GET_FIELD", + 94: "KEYS", + 143: "OBJECT", + 32: "HAS_FIELDS", + 96: "WITH_FIELDS", + 33: "PLUCK", + 34: "WITHOUT", + 35: "MERGE", + 36: "BETWEEN_DEPRECATED", + 182: "BETWEEN", + 37: "REDUCE", + 38: "MAP", + 39: "FILTER", + 40: "CONCAT_MAP", + 41: "ORDER_BY", + 42: "DISTINCT", + 43: "COUNT", + 86: "IS_EMPTY", + 44: "UNION", + 45: "NTH", + 170: "BRACKET", + 48: "INNER_JOIN", + 49: "OUTER_JOIN", + 50: "EQ_JOIN", + 72: "ZIP", + 173: "RANGE", + 82: "INSERT_AT", + 83: "DELETE_AT", + 84: "CHANGE_AT", + 85: "SPLICE_AT", + 51: "COERCE_TO", + 52: "TYPE_OF", + 53: "UPDATE", + 54: "DELETE", + 55: "REPLACE", + 56: "INSERT", + 57: "DB_CREATE", + 58: "DB_DROP", + 59: "DB_LIST", + 60: "TABLE_CREATE", + 61: "TABLE_DROP", + 62: "TABLE_LIST", + 174: "CONFIG", + 175: "STATUS", + 177: "WAIT", + 176: "RECONFIGURE", + 179: "REBALANCE", + 138: "SYNC", + 75: "INDEX_CREATE", + 76: "INDEX_DROP", + 77: "INDEX_LIST", + 139: "INDEX_STATUS", + 140: "INDEX_WAIT", + 156: "INDEX_RENAME", + 64: "FUNCALL", + 65: "BRANCH", + 66: "OR", + 67: "AND", + 68: "FOR_EACH", + 69: "FUNC", + 73: "ASC", + 74: "DESC", + 79: "INFO", + 97: "MATCH", + 141: "UPCASE", + 142: "DOWNCASE", + 81: "SAMPLE", + 92: "DEFAULT", + 98: "JSON", + 172: "TO_JSON_STRING", + 99: "ISO8601", + 100: "TO_ISO8601", + 101: "EPOCH_TIME", + 102: "TO_EPOCH_TIME", + 103: "NOW", + 104: "IN_TIMEZONE", + 105: "DURING", + 106: "DATE", + 126: "TIME_OF_DAY", + 127: "TIMEZONE", + 128: "YEAR", + 129: "MONTH", + 130: "DAY", + 131: "DAY_OF_WEEK", + 132: "DAY_OF_YEAR", + 133: "HOURS", + 134: "MINUTES", + 135: "SECONDS", + 136: "TIME", + 107: "MONDAY", + 108: "TUESDAY", + 109: "WEDNESDAY", + 110: "THURSDAY", + 111: "FRIDAY", + 112: "SATURDAY", + 113: "SUNDAY", + 114: "JANUARY", + 115: "FEBRUARY", + 116: "MARCH", + 117: "APRIL", + 118: "MAY", + 119: "JUNE", + 120: "JULY", + 121: "AUGUST", + 122: "SEPTEMBER", + 123: "OCTOBER", + 124: "NOVEMBER", + 125: "DECEMBER", + 137: "LITERAL", + 144: "GROUP", + 145: "SUM", + 146: "AVG", + 147: "MIN", + 148: "MAX", + 149: "SPLIT", + 150: "UNGROUP", + 151: "RANDOM", + 152: "CHANGES", + 154: "ARGS", + 155: "BINARY", + 157: "GEOJSON", + 158: "TO_GEOJSON", + 159: "POINT", + 160: "LINE", + 161: "POLYGON", + 162: "DISTANCE", + 163: "INTERSECTS", + 164: "INCLUDES", + 165: "CIRCLE", + 166: "GET_INTERSECTING", + 167: "FILL", + 168: "GET_NEAREST", + 171: "POLYGON_SUB", + 180: "MINVAL", + 181: "MAXVAL", +} +var Term_TermType_value = map[string]int32{ + "DATUM": 1, + "MAKE_ARRAY": 2, + "MAKE_OBJ": 3, + "VAR": 10, + "JAVASCRIPT": 11, + "UUID": 169, + "HTTP": 153, + "ERROR": 12, + "IMPLICIT_VAR": 13, + "DB": 14, + "TABLE": 15, + "GET": 16, + "GET_ALL": 78, + "EQ": 17, + "NE": 18, + "LT": 19, + "LE": 20, + "GT": 21, + "GE": 22, + "NOT": 23, + "ADD": 24, + "SUB": 25, + "MUL": 26, + "DIV": 27, + "MOD": 28, + "APPEND": 29, + "PREPEND": 80, + "DIFFERENCE": 95, + "SET_INSERT": 88, + "SET_INTERSECTION": 89, + "SET_UNION": 90, + "SET_DIFFERENCE": 91, + "SLICE": 30, + "SKIP": 70, + "LIMIT": 71, + "OFFSETS_OF": 87, + "CONTAINS": 93, + "GET_FIELD": 31, + "KEYS": 94, + "OBJECT": 143, + "HAS_FIELDS": 32, + "WITH_FIELDS": 96, + "PLUCK": 33, + "WITHOUT": 34, + "MERGE": 35, + "BETWEEN_DEPRECATED": 36, + "BETWEEN": 182, + "REDUCE": 37, + "MAP": 38, + "FILTER": 39, + "CONCAT_MAP": 40, + "ORDER_BY": 41, + "DISTINCT": 42, + "COUNT": 43, + "IS_EMPTY": 86, + "UNION": 44, + "NTH": 45, + "BRACKET": 170, + "INNER_JOIN": 48, + "OUTER_JOIN": 49, + "EQ_JOIN": 50, + "ZIP": 72, + "RANGE": 173, + "INSERT_AT": 82, + "DELETE_AT": 83, + "CHANGE_AT": 84, + "SPLICE_AT": 85, + "COERCE_TO": 51, + "TYPE_OF": 52, + "UPDATE": 53, + "DELETE": 54, + "REPLACE": 55, + "INSERT": 56, + "DB_CREATE": 57, + "DB_DROP": 58, + "DB_LIST": 59, + "TABLE_CREATE": 60, + "TABLE_DROP": 61, + "TABLE_LIST": 62, + "CONFIG": 174, + "STATUS": 175, + "WAIT": 177, + "RECONFIGURE": 176, + "REBALANCE": 179, + "SYNC": 138, + "INDEX_CREATE": 75, + "INDEX_DROP": 76, + "INDEX_LIST": 77, + "INDEX_STATUS": 139, + "INDEX_WAIT": 140, + "INDEX_RENAME": 156, + "FUNCALL": 64, + "BRANCH": 65, + "OR": 66, + "AND": 67, + "FOR_EACH": 68, + "FUNC": 69, + "ASC": 73, + "DESC": 74, + "INFO": 79, + "MATCH": 97, + "UPCASE": 141, + "DOWNCASE": 142, + "SAMPLE": 81, + "DEFAULT": 92, + "JSON": 98, + "TO_JSON_STRING": 172, + "ISO8601": 99, + "TO_ISO8601": 100, + "EPOCH_TIME": 101, + "TO_EPOCH_TIME": 102, + "NOW": 103, + "IN_TIMEZONE": 104, + "DURING": 105, + "DATE": 106, + "TIME_OF_DAY": 126, + "TIMEZONE": 127, + "YEAR": 128, + "MONTH": 129, + "DAY": 130, + "DAY_OF_WEEK": 131, + "DAY_OF_YEAR": 132, + "HOURS": 133, + "MINUTES": 134, + "SECONDS": 135, + "TIME": 136, + "MONDAY": 107, + "TUESDAY": 108, + "WEDNESDAY": 109, + "THURSDAY": 110, + "FRIDAY": 111, + "SATURDAY": 112, + "SUNDAY": 113, + "JANUARY": 114, + "FEBRUARY": 115, + "MARCH": 116, + "APRIL": 117, + "MAY": 118, + "JUNE": 119, + "JULY": 120, + "AUGUST": 121, + "SEPTEMBER": 122, + "OCTOBER": 123, + "NOVEMBER": 124, + "DECEMBER": 125, + "LITERAL": 137, + "GROUP": 144, + "SUM": 145, + "AVG": 146, + "MIN": 147, + "MAX": 148, + "SPLIT": 149, + "UNGROUP": 150, + "RANDOM": 151, + "CHANGES": 152, + "ARGS": 154, + "BINARY": 155, + "GEOJSON": 157, + "TO_GEOJSON": 158, + "POINT": 159, + "LINE": 160, + "POLYGON": 161, + "DISTANCE": 162, + "INTERSECTS": 163, + "INCLUDES": 164, + "CIRCLE": 165, + "GET_INTERSECTING": 166, + "FILL": 167, + "GET_NEAREST": 168, + "POLYGON_SUB": 171, + "MINVAL": 180, + "MAXVAL": 181, +} + +func (x Term_TermType) Enum() *Term_TermType { + p := new(Term_TermType) + *p = x + return p +} +func (x Term_TermType) String() string { + return proto.EnumName(Term_TermType_name, int32(x)) +} +func (x Term_TermType) MarshalJSON() ([]byte, error) { + return json.Marshal(x.String()) +} +func (x *Term_TermType) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(Term_TermType_value, data, "Term_TermType") + if err != nil { + return err + } + *x = Term_TermType(value) + return nil +} + +type VersionDummy struct { + XXX_unrecognized []byte `json:"-"` +} + +func (m *VersionDummy) Reset() { *m = VersionDummy{} } +func (m *VersionDummy) String() string { return proto.CompactTextString(m) } +func (*VersionDummy) ProtoMessage() {} + +type Query struct { + Type *Query_QueryType `protobuf:"varint,1,opt,name=type,enum=Query_QueryType" json:"type,omitempty"` + Query *Term `protobuf:"bytes,2,opt,name=query" json:"query,omitempty"` + Token *int64 `protobuf:"varint,3,opt,name=token" json:"token,omitempty"` + OBSOLETENoreply *bool `protobuf:"varint,4,opt,name=OBSOLETE_noreply,def=0" json:"OBSOLETE_noreply,omitempty"` + AcceptsRJson *bool `protobuf:"varint,5,opt,name=accepts_r_json,def=0" json:"accepts_r_json,omitempty"` + GlobalOptargs []*Query_AssocPair `protobuf:"bytes,6,rep,name=global_optargs" json:"global_optargs,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Query) Reset() { *m = Query{} } +func (m *Query) String() string { return proto.CompactTextString(m) } +func (*Query) ProtoMessage() {} + +const Default_Query_OBSOLETENoreply bool = false +const Default_Query_AcceptsRJson bool = false + +func (m *Query) GetType() Query_QueryType { + if m != nil && m.Type != nil { + return *m.Type + } + return 0 +} + +func (m *Query) GetQuery() *Term { + if m != nil { + return m.Query + } + return nil +} + +func (m *Query) GetToken() int64 { + if m != nil && m.Token != nil { + return *m.Token + } + return 0 +} + +func (m *Query) GetOBSOLETENoreply() bool { + if m != nil && m.OBSOLETENoreply != nil { + return *m.OBSOLETENoreply + } + return Default_Query_OBSOLETENoreply +} + +func (m *Query) GetAcceptsRJson() bool { + if m != nil && m.AcceptsRJson != nil { + return *m.AcceptsRJson + } + return Default_Query_AcceptsRJson +} + +func (m *Query) GetGlobalOptargs() []*Query_AssocPair { + if m != nil { + return m.GlobalOptargs + } + return nil +} + +type Query_AssocPair struct { + Key *string `protobuf:"bytes,1,opt,name=key" json:"key,omitempty"` + Val *Term `protobuf:"bytes,2,opt,name=val" json:"val,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Query_AssocPair) Reset() { *m = Query_AssocPair{} } +func (m *Query_AssocPair) String() string { return proto.CompactTextString(m) } +func (*Query_AssocPair) ProtoMessage() {} + +func (m *Query_AssocPair) GetKey() string { + if m != nil && m.Key != nil { + return *m.Key + } + return "" +} + +func (m *Query_AssocPair) GetVal() *Term { + if m != nil { + return m.Val + } + return nil +} + +type Frame struct { + Type *Frame_FrameType `protobuf:"varint,1,opt,name=type,enum=Frame_FrameType" json:"type,omitempty"` + Pos *int64 `protobuf:"varint,2,opt,name=pos" json:"pos,omitempty"` + Opt *string `protobuf:"bytes,3,opt,name=opt" json:"opt,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Frame) Reset() { *m = Frame{} } +func (m *Frame) String() string { return proto.CompactTextString(m) } +func (*Frame) ProtoMessage() {} + +func (m *Frame) GetType() Frame_FrameType { + if m != nil && m.Type != nil { + return *m.Type + } + return 0 +} + +func (m *Frame) GetPos() int64 { + if m != nil && m.Pos != nil { + return *m.Pos + } + return 0 +} + +func (m *Frame) GetOpt() string { + if m != nil && m.Opt != nil { + return *m.Opt + } + return "" +} + +type Backtrace struct { + Frames []*Frame `protobuf:"bytes,1,rep,name=frames" json:"frames,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Backtrace) Reset() { *m = Backtrace{} } +func (m *Backtrace) String() string { return proto.CompactTextString(m) } +func (*Backtrace) ProtoMessage() {} + +func (m *Backtrace) GetFrames() []*Frame { + if m != nil { + return m.Frames + } + return nil +} + +type Response struct { + Type *Response_ResponseType `protobuf:"varint,1,opt,name=type,enum=Response_ResponseType" json:"type,omitempty"` + Notes []Response_ResponseNote `protobuf:"varint,6,rep,name=notes,enum=Response_ResponseNote" json:"notes,omitempty"` + Token *int64 `protobuf:"varint,2,opt,name=token" json:"token,omitempty"` + Response []*Datum `protobuf:"bytes,3,rep,name=response" json:"response,omitempty"` + Backtrace *Backtrace `protobuf:"bytes,4,opt,name=backtrace" json:"backtrace,omitempty"` + Profile *Datum `protobuf:"bytes,5,opt,name=profile" json:"profile,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Response) Reset() { *m = Response{} } +func (m *Response) String() string { return proto.CompactTextString(m) } +func (*Response) ProtoMessage() {} + +func (m *Response) GetType() Response_ResponseType { + if m != nil && m.Type != nil { + return *m.Type + } + return 0 +} + +func (m *Response) GetNotes() []Response_ResponseNote { + if m != nil { + return m.Notes + } + return nil +} + +func (m *Response) GetToken() int64 { + if m != nil && m.Token != nil { + return *m.Token + } + return 0 +} + +func (m *Response) GetResponse() []*Datum { + if m != nil { + return m.Response + } + return nil +} + +func (m *Response) GetBacktrace() *Backtrace { + if m != nil { + return m.Backtrace + } + return nil +} + +func (m *Response) GetProfile() *Datum { + if m != nil { + return m.Profile + } + return nil +} + +type Datum struct { + Type *Datum_DatumType `protobuf:"varint,1,opt,name=type,enum=Datum_DatumType" json:"type,omitempty"` + RBool *bool `protobuf:"varint,2,opt,name=r_bool" json:"r_bool,omitempty"` + RNum *float64 `protobuf:"fixed64,3,opt,name=r_num" json:"r_num,omitempty"` + RStr *string `protobuf:"bytes,4,opt,name=r_str" json:"r_str,omitempty"` + RArray []*Datum `protobuf:"bytes,5,rep,name=r_array" json:"r_array,omitempty"` + RObject []*Datum_AssocPair `protobuf:"bytes,6,rep,name=r_object" json:"r_object,omitempty"` + XXX_extensions map[int32]proto.Extension `json:"-"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Datum) Reset() { *m = Datum{} } +func (m *Datum) String() string { return proto.CompactTextString(m) } +func (*Datum) ProtoMessage() {} + +var extRange_Datum = []proto.ExtensionRange{ + {10000, 20000}, +} + +func (*Datum) ExtensionRangeArray() []proto.ExtensionRange { + return extRange_Datum +} +func (m *Datum) ExtensionMap() map[int32]proto.Extension { + if m.XXX_extensions == nil { + m.XXX_extensions = make(map[int32]proto.Extension) + } + return m.XXX_extensions +} + +func (m *Datum) GetType() Datum_DatumType { + if m != nil && m.Type != nil { + return *m.Type + } + return 0 +} + +func (m *Datum) GetRBool() bool { + if m != nil && m.RBool != nil { + return *m.RBool + } + return false +} + +func (m *Datum) GetRNum() float64 { + if m != nil && m.RNum != nil { + return *m.RNum + } + return 0 +} + +func (m *Datum) GetRStr() string { + if m != nil && m.RStr != nil { + return *m.RStr + } + return "" +} + +func (m *Datum) GetRArray() []*Datum { + if m != nil { + return m.RArray + } + return nil +} + +func (m *Datum) GetRObject() []*Datum_AssocPair { + if m != nil { + return m.RObject + } + return nil +} + +type Datum_AssocPair struct { + Key *string `protobuf:"bytes,1,opt,name=key" json:"key,omitempty"` + Val *Datum `protobuf:"bytes,2,opt,name=val" json:"val,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Datum_AssocPair) Reset() { *m = Datum_AssocPair{} } +func (m *Datum_AssocPair) String() string { return proto.CompactTextString(m) } +func (*Datum_AssocPair) ProtoMessage() {} + +func (m *Datum_AssocPair) GetKey() string { + if m != nil && m.Key != nil { + return *m.Key + } + return "" +} + +func (m *Datum_AssocPair) GetVal() *Datum { + if m != nil { + return m.Val + } + return nil +} + +type Term struct { + Type *Term_TermType `protobuf:"varint,1,opt,name=type,enum=Term_TermType" json:"type,omitempty"` + Datum *Datum `protobuf:"bytes,2,opt,name=datum" json:"datum,omitempty"` + Args []*Term `protobuf:"bytes,3,rep,name=args" json:"args,omitempty"` + Optargs []*Term_AssocPair `protobuf:"bytes,4,rep,name=optargs" json:"optargs,omitempty"` + XXX_extensions map[int32]proto.Extension `json:"-"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Term) Reset() { *m = Term{} } +func (m *Term) String() string { return proto.CompactTextString(m) } +func (*Term) ProtoMessage() {} + +var extRange_Term = []proto.ExtensionRange{ + {10000, 20000}, +} + +func (*Term) ExtensionRangeArray() []proto.ExtensionRange { + return extRange_Term +} +func (m *Term) ExtensionMap() map[int32]proto.Extension { + if m.XXX_extensions == nil { + m.XXX_extensions = make(map[int32]proto.Extension) + } + return m.XXX_extensions +} + +func (m *Term) GetType() Term_TermType { + if m != nil && m.Type != nil { + return *m.Type + } + return 0 +} + +func (m *Term) GetDatum() *Datum { + if m != nil { + return m.Datum + } + return nil +} + +func (m *Term) GetArgs() []*Term { + if m != nil { + return m.Args + } + return nil +} + +func (m *Term) GetOptargs() []*Term_AssocPair { + if m != nil { + return m.Optargs + } + return nil +} + +type Term_AssocPair struct { + Key *string `protobuf:"bytes,1,opt,name=key" json:"key,omitempty"` + Val *Term `protobuf:"bytes,2,opt,name=val" json:"val,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Term_AssocPair) Reset() { *m = Term_AssocPair{} } +func (m *Term_AssocPair) String() string { return proto.CompactTextString(m) } +func (*Term_AssocPair) ProtoMessage() {} + +func (m *Term_AssocPair) GetKey() string { + if m != nil && m.Key != nil { + return *m.Key + } + return "" +} + +func (m *Term_AssocPair) GetVal() *Term { + if m != nil { + return m.Val + } + return nil +} + +func init() { + proto.RegisterEnum("VersionDummy_Version", VersionDummy_Version_name, VersionDummy_Version_value) + proto.RegisterEnum("VersionDummy_Protocol", VersionDummy_Protocol_name, VersionDummy_Protocol_value) + proto.RegisterEnum("Query_QueryType", Query_QueryType_name, Query_QueryType_value) + proto.RegisterEnum("Frame_FrameType", Frame_FrameType_name, Frame_FrameType_value) + proto.RegisterEnum("Response_ResponseType", Response_ResponseType_name, Response_ResponseType_value) + proto.RegisterEnum("Response_ResponseNote", Response_ResponseNote_name, Response_ResponseNote_value) + proto.RegisterEnum("Datum_DatumType", Datum_DatumType_name, Datum_DatumType_value) + proto.RegisterEnum("Term_TermType", Term_TermType_name, Term_TermType_value) +} diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/ql2/ql2.proto b/Godeps/_workspace/src/github.com/dancannon/gorethink/ql2/ql2.proto new file mode 100644 index 0000000000000..3ad50a2dc0308 --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/ql2/ql2.proto @@ -0,0 +1,805 @@ +//////////////////////////////////////////////////////////////////////////////// +// THE HIGH-LEVEL VIEW // +//////////////////////////////////////////////////////////////////////////////// + +// Process: When you first open a connection, send the magic number +// for the version of the protobuf you're targeting (in the [Version] +// enum). This should **NOT** be sent as a protobuf; just send the +// little-endian 32-bit integer over the wire raw. This number should +// only be sent once per connection. + +// The magic number shall be followed by an authorization key. The +// first 4 bytes are the length of the key to be sent as a little-endian +// 32-bit integer, followed by the key string. Even if there is no key, +// an empty string should be sent (length 0 and no data). + +// Following the authorization key, the client shall send a magic number +// for the communication protocol they want to use (in the [Protocol] +// enum). This shall be a little-endian 32-bit integer. + +// The server will then respond with a NULL-terminated string response. +// "SUCCESS" indicates that the connection has been accepted. Any other +// response indicates an error, and the response string should describe +// the error. + +// Next, for each query you want to send, construct a [Query] protobuf +// and serialize it to a binary blob. Send the blob's size to the +// server encoded as a little-endian 32-bit integer, followed by the +// blob itself. You will recieve a [Response] protobuf back preceded +// by its own size, once again encoded as a little-endian 32-bit +// integer. You can see an example exchange below in **EXAMPLE**. + +// A query consists of a [Term] to evaluate and a unique-per-connection +// [token]. + +// Tokens are used for two things: +// * Keeping track of which responses correspond to which queries. +// * Batched queries. Some queries return lots of results, so we send back +// batches of <1000, and you need to send a [CONTINUE] query with the same +// token to get more results from the original query. +//////////////////////////////////////////////////////////////////////////////// + +message VersionDummy { // We need to wrap it like this for some + // non-conforming protobuf libraries + // This enum contains the magic numbers for your version. See **THE HIGH-LEVEL + // VIEW** for what to do with it. + enum Version { + V0_1 = 0x3f61ba36; + V0_2 = 0x723081e1; // Authorization key during handshake + V0_3 = 0x5f75e83e; // Authorization key and protocol during handshake + V0_4 = 0x400c2d20; // Queries execute in parallel + } + + // The protocol to use after the handshake, specified in V0_3 + enum Protocol { + PROTOBUF = 0x271ffc41; + JSON = 0x7e6970c7; + } +} + +// You send one of: +// * A [START] query with a [Term] to evaluate and a unique-per-connection token. +// * A [CONTINUE] query with the same token as a [START] query that returned +// [SUCCESS_PARTIAL] in its [Response]. +// * A [STOP] query with the same token as a [START] query that you want to stop. +// * A [NOREPLY_WAIT] query with a unique per-connection token. The server answers +// with a [WAIT_COMPLETE] [Response]. +message Query { + enum QueryType { + START = 1; // Start a new query. + CONTINUE = 2; // Continue a query that returned [SUCCESS_PARTIAL] + // (see [Response]). + STOP = 3; // Stop a query partway through executing. + NOREPLY_WAIT = 4; + // Wait for noreply operations to finish. + } + optional QueryType type = 1; + // A [Term] is how we represent the operations we want a query to perform. + optional Term query = 2; // only present when [type] = [START] + optional int64 token = 3; + // This flag is ignored on the server. `noreply` should be added + // to `global_optargs` instead (the key "noreply" should map to + // either true or false). + optional bool OBSOLETE_noreply = 4 [default = false]; + + // If this is set to [true], then [Datum] values will sometimes be + // of [DatumType] [R_JSON] (see below). This can provide enormous + // speedups in languages with poor protobuf libraries. + optional bool accepts_r_json = 5 [default = false]; + + message AssocPair { + optional string key = 1; + optional Term val = 2; + } + repeated AssocPair global_optargs = 6; +} + +// A backtrace frame (see `backtrace` in Response below) +message Frame { + enum FrameType { + POS = 1; // Error occured in a positional argument. + OPT = 2; // Error occured in an optional argument. + } + optional FrameType type = 1; + optional int64 pos = 2; // The index of the positional argument. + optional string opt = 3; // The name of the optional argument. +} +message Backtrace { + repeated Frame frames = 1; +} + +// You get back a response with the same [token] as your query. +message Response { + enum ResponseType { + // These response types indicate success. + SUCCESS_ATOM = 1; // Query returned a single RQL datatype. + SUCCESS_SEQUENCE = 2; // Query returned a sequence of RQL datatypes. + SUCCESS_PARTIAL = 3; // Query returned a partial sequence of RQL + // datatypes. If you send a [CONTINUE] query with + // the same token as this response, you will get + // more of the sequence. Keep sending [CONTINUE] + // queries until you get back [SUCCESS_SEQUENCE]. + WAIT_COMPLETE = 4; // A [NOREPLY_WAIT] query completed. + + // These response types indicate failure. + CLIENT_ERROR = 16; // Means the client is buggy. An example is if the + // client sends a malformed protobuf, or tries to + // send [CONTINUE] for an unknown token. + COMPILE_ERROR = 17; // Means the query failed during parsing or type + // checking. For example, if you pass too many + // arguments to a function. + RUNTIME_ERROR = 18; // Means the query failed at runtime. An example is + // if you add together two values from a table, but + // they turn out at runtime to be booleans rather + // than numbers. + } + optional ResponseType type = 1; + + // ResponseNotes are used to provide information about the query + // response that may be useful for people writing drivers or ORMs. + // Currently all the notes we send indicate that a stream has certain + // special properties. + enum ResponseNote { + // The stream is a changefeed stream (e.g. `r.table('test').changes()`). + SEQUENCE_FEED = 1; + // The stream is a point changefeed stream + // (e.g. `r.table('test').get(0).changes()`). + ATOM_FEED = 2; + // The stream is an order_by_limit changefeed stream + // (e.g. `r.table('test').order_by(index: 'id').limit(5).changes()`). + ORDER_BY_LIMIT_FEED = 3; + // The stream is a union of multiple changefeed types that can't be + // collapsed to a single type + // (e.g. `r.table('test').changes().union(r.table('test').get(0).changes())`). + UNIONED_FEED = 4; + // The stream is a changefeed stream and includes notes on what state + // the changefeed stream is in (e.g. objects of the form `{state: + // 'initializing'}`). + INCLUDES_STATES = 5; + } + repeated ResponseNote notes = 6; + + optional int64 token = 2; // Indicates what [Query] this response corresponds to. + + // [response] contains 1 RQL datum if [type] is [SUCCESS_ATOM], or many RQL + // data if [type] is [SUCCESS_SEQUENCE] or [SUCCESS_PARTIAL]. It contains 1 + // error message (of type [R_STR]) in all other cases. + repeated Datum response = 3; + + // If [type] is [CLIENT_ERROR], [TYPE_ERROR], or [RUNTIME_ERROR], then a + // backtrace will be provided. The backtrace says where in the query the + // error occured. Ideally this information will be presented to the user as + // a pretty-printed version of their query with the erroneous section + // underlined. A backtrace is a series of 0 or more [Frame]s, each of which + // specifies either the index of a positional argument or the name of an + // optional argument. (Those words will make more sense if you look at the + // [Term] message below.) + optional Backtrace backtrace = 4; // Contains n [Frame]s when you get back an error. + + // If the [global_optargs] in the [Query] that this [Response] is a + // response to contains a key "profile" which maps to a static value of + // true then [profile] will contain a [Datum] which provides profiling + // information about the execution of the query. This field should be + // returned to the user along with the result that would normally be + // returned (a datum or a cursor). In official drivers this is accomplished + // by putting them inside of an object with "value" mapping to the return + // value and "profile" mapping to the profile object. + optional Datum profile = 5; +} + +// A [Datum] is a chunk of data that can be serialized to disk or returned to +// the user in a Response. Currently we only support JSON types, but we may +// support other types in the future (e.g., a date type or an integer type). +message Datum { + enum DatumType { + R_NULL = 1; + R_BOOL = 2; + R_NUM = 3; // a double + R_STR = 4; + R_ARRAY = 5; + R_OBJECT = 6; + // This [DatumType] will only be used if [accepts_r_json] is + // set to [true] in [Query]. [r_str] will be filled with a + // JSON encoding of the [Datum]. + R_JSON = 7; // uses r_str + } + optional DatumType type = 1; + optional bool r_bool = 2; + optional double r_num = 3; + optional string r_str = 4; + + repeated Datum r_array = 5; + message AssocPair { + optional string key = 1; + optional Datum val = 2; + } + repeated AssocPair r_object = 6; + + extensions 10000 to 20000; +} + +// A [Term] is either a piece of data (see **Datum** above), or an operator and +// its operands. If you have a [Datum], it's stored in the member [datum]. If +// you have an operator, its positional arguments are stored in [args] and its +// optional arguments are stored in [optargs]. +// +// A note about type signatures: +// We use the following notation to denote types: +// arg1_type, arg2_type, argrest_type... -> result_type +// So, for example, if we have a function `avg` that takes any number of +// arguments and averages them, we might write: +// NUMBER... -> NUMBER +// Or if we had a function that took one number modulo another: +// NUMBER, NUMBER -> NUMBER +// Or a function that takes a table and a primary key of any Datum type, then +// retrieves the entry with that primary key: +// Table, DATUM -> OBJECT +// Some arguments must be provided as literal values (and not the results of sub +// terms). These are marked with a `!`. +// Optional arguments are specified within curly braces as argname `:` value +// type (e.x `{use_outdated:BOOL}`) +// Many RQL operations are polymorphic. For these, alterantive type signatures +// are separated by `|`. +// +// The RQL type hierarchy is as follows: +// Top +// DATUM +// NULL +// BOOL +// NUMBER +// STRING +// OBJECT +// SingleSelection +// ARRAY +// Sequence +// ARRAY +// Stream +// StreamSelection +// Table +// Database +// Function +// Ordering - used only by ORDER_BY +// Pathspec -- an object, string, or array that specifies a path +// Error +message Term { + enum TermType { + // A RQL datum, stored in `datum` below. + DATUM = 1; + + MAKE_ARRAY = 2; // DATUM... -> ARRAY + // Evaluate the terms in [optargs] and make an object + MAKE_OBJ = 3; // {...} -> OBJECT + + // * Compound types + + // Takes an integer representing a variable and returns the value stored + // in that variable. It's the responsibility of the client to translate + // from their local representation of a variable to a unique _non-negative_ + // integer for that variable. (We do it this way instead of letting + // clients provide variable names as strings to discourage + // variable-capturing client libraries, and because it's more efficient + // on the wire.) + VAR = 10; // !NUMBER -> DATUM + // Takes some javascript code and executes it. + JAVASCRIPT = 11; // STRING {timeout: !NUMBER} -> DATUM | + // STRING {timeout: !NUMBER} -> Function(*) + UUID = 169; // () -> DATUM + + // Takes an HTTP URL and gets it. If the get succeeds and + // returns valid JSON, it is converted into a DATUM + HTTP = 153; // STRING {data: OBJECT | STRING, + // timeout: !NUMBER, + // method: STRING, + // params: OBJECT, + // header: OBJECT | ARRAY, + // attempts: NUMBER, + // redirects: NUMBER, + // verify: BOOL, + // page: FUNC | STRING, + // page_limit: NUMBER, + // auth: OBJECT, + // result_format: STRING, + // } -> STRING | STREAM + + // Takes a string and throws an error with that message. + // Inside of a `default` block, you can omit the first + // argument to rethrow whatever error you catch (this is most + // useful as an argument to the `default` filter optarg). + ERROR = 12; // STRING -> Error | -> Error + // Takes nothing and returns a reference to the implicit variable. + IMPLICIT_VAR = 13; // -> DATUM + + // * Data Operators + // Returns a reference to a database. + DB = 14; // STRING -> Database + // Returns a reference to a table. + TABLE = 15; // Database, STRING, {use_outdated:BOOL, identifier_format:STRING} -> Table + // STRING, {use_outdated:BOOL, identifier_format:STRING} -> Table + // Gets a single element from a table by its primary or a secondary key. + GET = 16; // Table, STRING -> SingleSelection | Table, NUMBER -> SingleSelection | + // Table, STRING -> NULL | Table, NUMBER -> NULL | + GET_ALL = 78; // Table, DATUM..., {index:!STRING} => ARRAY + + // Simple DATUM Ops + EQ = 17; // DATUM... -> BOOL + NE = 18; // DATUM... -> BOOL + LT = 19; // DATUM... -> BOOL + LE = 20; // DATUM... -> BOOL + GT = 21; // DATUM... -> BOOL + GE = 22; // DATUM... -> BOOL + NOT = 23; // BOOL -> BOOL + // ADD can either add two numbers or concatenate two arrays. + ADD = 24; // NUMBER... -> NUMBER | STRING... -> STRING + SUB = 25; // NUMBER... -> NUMBER + MUL = 26; // NUMBER... -> NUMBER + DIV = 27; // NUMBER... -> NUMBER + MOD = 28; // NUMBER, NUMBER -> NUMBER + + // DATUM Array Ops + // Append a single element to the end of an array (like `snoc`). + APPEND = 29; // ARRAY, DATUM -> ARRAY + // Prepend a single element to the end of an array (like `cons`). + PREPEND = 80; // ARRAY, DATUM -> ARRAY + //Remove the elements of one array from another array. + DIFFERENCE = 95; // ARRAY, ARRAY -> ARRAY + + // DATUM Set Ops + // Set ops work on arrays. They don't use actual sets and thus have + // performance characteristics you would expect from arrays rather than + // from sets. All set operations have the post condition that they + // array they return contains no duplicate values. + SET_INSERT = 88; // ARRAY, DATUM -> ARRAY + SET_INTERSECTION = 89; // ARRAY, ARRAY -> ARRAY + SET_UNION = 90; // ARRAY, ARRAY -> ARRAY + SET_DIFFERENCE = 91; // ARRAY, ARRAY -> ARRAY + + SLICE = 30; // Sequence, NUMBER, NUMBER -> Sequence + SKIP = 70; // Sequence, NUMBER -> Sequence + LIMIT = 71; // Sequence, NUMBER -> Sequence + OFFSETS_OF = 87; // Sequence, DATUM -> Sequence | Sequence, Function(1) -> Sequence + CONTAINS = 93; // Sequence, DATUM -> BOOL | Sequence, Function(1) -> BOOL + + // Stream/Object Ops + // Get a particular field from an object, or map that over a + // sequence. + GET_FIELD = 31; // OBJECT, STRING -> DATUM + // | Sequence, STRING -> Sequence + // Return an array containing the keys of the object. + KEYS = 94; // OBJECT -> ARRAY + // Creates an object + OBJECT = 143; // STRING, DATUM, ... -> OBJECT + // Check whether an object contains all the specified fields, + // or filters a sequence so that all objects inside of it + // contain all the specified fields. + HAS_FIELDS = 32; // OBJECT, Pathspec... -> BOOL + // x.with_fields(...) <=> x.has_fields(...).pluck(...) + WITH_FIELDS = 96; // Sequence, Pathspec... -> Sequence + // Get a subset of an object by selecting some attributes to preserve, + // or map that over a sequence. (Both pick and pluck, polymorphic.) + PLUCK = 33; // Sequence, Pathspec... -> Sequence | OBJECT, Pathspec... -> OBJECT + // Get a subset of an object by selecting some attributes to discard, or + // map that over a sequence. (Both unpick and without, polymorphic.) + WITHOUT = 34; // Sequence, Pathspec... -> Sequence | OBJECT, Pathspec... -> OBJECT + // Merge objects (right-preferential) + MERGE = 35; // OBJECT... -> OBJECT | Sequence -> Sequence + + // Sequence Ops + // Get all elements of a sequence between two values. + // Half-open by default, but the openness of either side can be + // changed by passing 'closed' or 'open for `right_bound` or + // `left_bound`. + BETWEEN_DEPRECATED = 36; // Deprecated version of between, which allows `null` to specify unboundedness + // With the newer version, clients should use `r.minval` and `r.maxval` for unboundedness + BETWEEN = 182; // StreamSelection, DATUM, DATUM, {index:!STRING, right_bound:STRING, left_bound:STRING} -> StreamSelection + REDUCE = 37; // Sequence, Function(2) -> DATUM + MAP = 38; // Sequence, Function(1) -> Sequence + // The arity of the function should be + // Sequence..., Function(sizeof...(Sequence)) -> Sequence + + // Filter a sequence with either a function or a shortcut + // object (see API docs for details). The body of FILTER is + // wrapped in an implicit `.default(false)`, and you can + // change the default value by specifying the `default` + // optarg. If you make the default `r.error`, all errors + // caught by `default` will be rethrown as if the `default` + // did not exist. + FILTER = 39; // Sequence, Function(1), {default:DATUM} -> Sequence | + // Sequence, OBJECT, {default:DATUM} -> Sequence + // Map a function over a sequence and then concatenate the results together. + CONCAT_MAP = 40; // Sequence, Function(1) -> Sequence + // Order a sequence based on one or more attributes. + ORDER_BY = 41; // Sequence, (!STRING | Ordering)... -> Sequence + // Get all distinct elements of a sequence (like `uniq`). + DISTINCT = 42; // Sequence -> Sequence + // Count the number of elements in a sequence, or only the elements that match + // a given filter. + COUNT = 43; // Sequence -> NUMBER | Sequence, DATUM -> NUMBER | Sequence, Function(1) -> NUMBER + IS_EMPTY = 86; // Sequence -> BOOL + // Take the union of multiple sequences (preserves duplicate elements! (use distinct)). + UNION = 44; // Sequence... -> Sequence + // Get the Nth element of a sequence. + NTH = 45; // Sequence, NUMBER -> DATUM + // do NTH or GET_FIELD depending on target object + BRACKET = 170; // Sequence | OBJECT, NUMBER | STRING -> DATUM + // OBSOLETE_GROUPED_MAPREDUCE = 46; + // OBSOLETE_GROUPBY = 47; + + INNER_JOIN = 48; // Sequence, Sequence, Function(2) -> Sequence + OUTER_JOIN = 49; // Sequence, Sequence, Function(2) -> Sequence + // An inner-join that does an equality comparison on two attributes. + EQ_JOIN = 50; // Sequence, !STRING, Sequence, {index:!STRING} -> Sequence + ZIP = 72; // Sequence -> Sequence + RANGE = 173; // -> Sequence [0, +inf) + // NUMBER -> Sequence [0, a) + // NUMBER, NUMBER -> Sequence [a, b) + + // Array Ops + // Insert an element in to an array at a given index. + INSERT_AT = 82; // ARRAY, NUMBER, DATUM -> ARRAY + // Remove an element at a given index from an array. + DELETE_AT = 83; // ARRAY, NUMBER -> ARRAY | + // ARRAY, NUMBER, NUMBER -> ARRAY + // Change the element at a given index of an array. + CHANGE_AT = 84; // ARRAY, NUMBER, DATUM -> ARRAY + // Splice one array in to another array. + SPLICE_AT = 85; // ARRAY, NUMBER, ARRAY -> ARRAY + + // * Type Ops + // Coerces a datum to a named type (e.g. "bool"). + // If you previously used `stream_to_array`, you should use this instead + // with the type "array". + COERCE_TO = 51; // Top, STRING -> Top + // Returns the named type of a datum (e.g. TYPE_OF(true) = "BOOL") + TYPE_OF = 52; // Top -> STRING + + // * Write Ops (the OBJECTs contain data about number of errors etc.) + // Updates all the rows in a selection. Calls its Function with the row + // to be updated, and then merges the result of that call. + UPDATE = 53; // StreamSelection, Function(1), {non_atomic:BOOL, durability:STRING, return_changes:BOOL} -> OBJECT | + // SingleSelection, Function(1), {non_atomic:BOOL, durability:STRING, return_changes:BOOL} -> OBJECT | + // StreamSelection, OBJECT, {non_atomic:BOOL, durability:STRING, return_changes:BOOL} -> OBJECT | + // SingleSelection, OBJECT, {non_atomic:BOOL, durability:STRING, return_changes:BOOL} -> OBJECT + // Deletes all the rows in a selection. + DELETE = 54; // StreamSelection, {durability:STRING, return_changes:BOOL} -> OBJECT | SingleSelection -> OBJECT + // Replaces all the rows in a selection. Calls its Function with the row + // to be replaced, and then discards it and stores the result of that + // call. + REPLACE = 55; // StreamSelection, Function(1), {non_atomic:BOOL, durability:STRING, return_changes:BOOL} -> OBJECT | SingleSelection, Function(1), {non_atomic:BOOL, durability:STRING, return_changes:BOOL} -> OBJECT + // Inserts into a table. If `conflict` is replace, overwrites + // entries with the same primary key. If `conflict` is + // update, does an update on the entry. If `conflict` is + // error, or is omitted, conflicts will trigger an error. + INSERT = 56; // Table, OBJECT, {conflict:STRING, durability:STRING, return_changes:BOOL} -> OBJECT | Table, Sequence, {conflict:STRING, durability:STRING, return_changes:BOOL} -> OBJECT + + // * Administrative OPs + // Creates a database with a particular name. + DB_CREATE = 57; // STRING -> OBJECT + // Drops a database with a particular name. + DB_DROP = 58; // STRING -> OBJECT + // Lists all the databases by name. (Takes no arguments) + DB_LIST = 59; // -> ARRAY + // Creates a table with a particular name in a particular + // database. (You may omit the first argument to use the + // default database.) + TABLE_CREATE = 60; // Database, STRING, {primary_key:STRING, shards:NUMBER, replicas:NUMBER, primary_replica_tag:STRING} -> OBJECT + // Database, STRING, {primary_key:STRING, shards:NUMBER, replicas:OBJECT, primary_replica_tag:STRING} -> OBJECT + // STRING, {primary_key:STRING, shards:NUMBER, replicas:NUMBER, primary_replica_tag:STRING} -> OBJECT + // STRING, {primary_key:STRING, shards:NUMBER, replicas:OBJECT, primary_replica_tag:STRING} -> OBJECT + // Drops a table with a particular name from a particular + // database. (You may omit the first argument to use the + // default database.) + TABLE_DROP = 61; // Database, STRING -> OBJECT + // STRING -> OBJECT + // Lists all the tables in a particular database. (You may + // omit the first argument to use the default database.) + TABLE_LIST = 62; // Database -> ARRAY + // -> ARRAY + // Returns the row in the `rethinkdb.table_config` or `rethinkdb.db_config` table + // that corresponds to the given database or table. + CONFIG = 174; // Database -> SingleSelection + // Table -> SingleSelection + // Returns the row in the `rethinkdb.table_status` table that corresponds to the + // given table. + STATUS = 175; // Table -> SingleSelection + // Called on a table, waits for that table to be ready for read/write operations. + // Called on a database, waits for all of the tables in the database to be ready. + // Returns the corresponding row or rows from the `rethinkdb.table_status` table. + WAIT = 177; // Table -> OBJECT + // Database -> OBJECT + // Generates a new config for the given table, or all tables in the given database + // The `shards` and `replicas` arguments are required + RECONFIGURE = 176; // Database, {shards:NUMBER, replicas:NUMBER[, primary_replica_tag:STRING, dry_run:BOOLEAN]} -> OBJECT + // Database, {shards:NUMBER, replicas:OBJECT[, primary_replica_tag:STRING, dry_run:BOOLEAN]} -> OBJECT + // Table, {shards:NUMBER, replicas:NUMBER[, primary_replica_tag:STRING, dry_run:BOOLEAN]} -> OBJECT + // Table, {shards:NUMBER, replicas:OBJECT[, primary_replica_tag:STRING, dry_run:BOOLEAN]} -> OBJECT + // Balances the table's shards but leaves everything else the same. Can also be + // applied to an entire database at once. + REBALANCE = 179; // Table -> OBJECT + // Database -> OBJECT + + // Ensures that previously issued soft-durability writes are complete and + // written to disk. + SYNC = 138; // Table -> OBJECT + + // * Secondary indexes OPs + // Creates a new secondary index with a particular name and definition. + INDEX_CREATE = 75; // Table, STRING, Function(1), {multi:BOOL} -> OBJECT + // Drops a secondary index with a particular name from the specified table. + INDEX_DROP = 76; // Table, STRING -> OBJECT + // Lists all secondary indexes on a particular table. + INDEX_LIST = 77; // Table -> ARRAY + // Gets information about whether or not a set of indexes are ready to + // be accessed. Returns a list of objects that look like this: + // {index:STRING, ready:BOOL[, blocks_processed:NUMBER, blocks_total:NUMBER]} + INDEX_STATUS = 139; // Table, STRING... -> ARRAY + // Blocks until a set of indexes are ready to be accessed. Returns the + // same values INDEX_STATUS. + INDEX_WAIT = 140; // Table, STRING... -> ARRAY + // Renames the given index to a new name + INDEX_RENAME = 156; // Table, STRING, STRING, {overwrite:BOOL} -> OBJECT + + // * Control Operators + // Calls a function on data + FUNCALL = 64; // Function(*), DATUM... -> DATUM + // Executes its first argument, and returns its second argument if it + // got [true] or its third argument if it got [false] (like an `if` + // statement). + BRANCH = 65; // BOOL, Top, Top -> Top + // Returns true if any of its arguments returns true (short-circuits). + OR = 66; // BOOL... -> BOOL + // Returns true if all of its arguments return true (short-circuits). + AND = 67; // BOOL... -> BOOL + // Calls its Function with each entry in the sequence + // and executes the array of terms that Function returns. + FOR_EACH = 68; // Sequence, Function(1) -> OBJECT + +//////////////////////////////////////////////////////////////////////////////// +////////// Special Terms +//////////////////////////////////////////////////////////////////////////////// + + // An anonymous function. Takes an array of numbers representing + // variables (see [VAR] above), and a [Term] to execute with those in + // scope. Returns a function that may be passed an array of arguments, + // then executes the Term with those bound to the variable names. The + // user will never construct this directly. We use it internally for + // things like `map` which take a function. The "arity" of a [Function] is + // the number of arguments it takes. + // For example, here's what `_X_.map{|x| x+2}` turns into: + // Term { + // type = MAP; + // args = [_X_, + // Term { + // type = Function; + // args = [Term { + // type = DATUM; + // datum = Datum { + // type = R_ARRAY; + // r_array = [Datum { type = R_NUM; r_num = 1; }]; + // }; + // }, + // Term { + // type = ADD; + // args = [Term { + // type = VAR; + // args = [Term { + // type = DATUM; + // datum = Datum { type = R_NUM; + // r_num = 1}; + // }]; + // }, + // Term { + // type = DATUM; + // datum = Datum { type = R_NUM; r_num = 2; }; + // }]; + // }]; + // }]; + FUNC = 69; // ARRAY, Top -> ARRAY -> Top + + // Indicates to ORDER_BY that this attribute is to be sorted in ascending order. + ASC = 73; // !STRING -> Ordering + // Indicates to ORDER_BY that this attribute is to be sorted in descending order. + DESC = 74; // !STRING -> Ordering + + // Gets info about anything. INFO is most commonly called on tables. + INFO = 79; // Top -> OBJECT + + // `a.match(b)` returns a match object if the string `a` + // matches the regular expression `b`. + MATCH = 97; // STRING, STRING -> DATUM + + // Change the case of a string. + UPCASE = 141; // STRING -> STRING + DOWNCASE = 142; // STRING -> STRING + + // Select a number of elements from sequence with uniform distribution. + SAMPLE = 81; // Sequence, NUMBER -> Sequence + + // Evaluates its first argument. If that argument returns + // NULL or throws an error related to the absence of an + // expected value (for instance, accessing a non-existent + // field or adding NULL to an integer), DEFAULT will either + // return its second argument or execute it if it's a + // function. If the second argument is a function, it will be + // passed either the text of the error or NULL as its + // argument. + DEFAULT = 92; // Top, Top -> Top + + // Parses its first argument as a json string and returns it as a + // datum. + JSON = 98; // STRING -> DATUM + // Returns the datum as a JSON string. + // N.B.: we would really prefer this be named TO_JSON and that exists as + // an alias in Python and JavaScript drivers; however it conflicts with the + // standard `to_json` method defined by Ruby's standard json library. + TO_JSON_STRING = 172; // DATUM -> STRING + + // Parses its first arguments as an ISO 8601 time and returns it as a + // datum. + ISO8601 = 99; // STRING -> PSEUDOTYPE(TIME) + // Prints a time as an ISO 8601 time. + TO_ISO8601 = 100; // PSEUDOTYPE(TIME) -> STRING + + // Returns a time given seconds since epoch in UTC. + EPOCH_TIME = 101; // NUMBER -> PSEUDOTYPE(TIME) + // Returns seconds since epoch in UTC given a time. + TO_EPOCH_TIME = 102; // PSEUDOTYPE(TIME) -> NUMBER + + // The time the query was received by the server. + NOW = 103; // -> PSEUDOTYPE(TIME) + // Puts a time into an ISO 8601 timezone. + IN_TIMEZONE = 104; // PSEUDOTYPE(TIME), STRING -> PSEUDOTYPE(TIME) + // a.during(b, c) returns whether a is in the range [b, c) + DURING = 105; // PSEUDOTYPE(TIME), PSEUDOTYPE(TIME), PSEUDOTYPE(TIME) -> BOOL + // Retrieves the date portion of a time. + DATE = 106; // PSEUDOTYPE(TIME) -> PSEUDOTYPE(TIME) + // x.time_of_day == x.date - x + TIME_OF_DAY = 126; // PSEUDOTYPE(TIME) -> NUMBER + // Returns the timezone of a time. + TIMEZONE = 127; // PSEUDOTYPE(TIME) -> STRING + + // These access the various components of a time. + YEAR = 128; // PSEUDOTYPE(TIME) -> NUMBER + MONTH = 129; // PSEUDOTYPE(TIME) -> NUMBER + DAY = 130; // PSEUDOTYPE(TIME) -> NUMBER + DAY_OF_WEEK = 131; // PSEUDOTYPE(TIME) -> NUMBER + DAY_OF_YEAR = 132; // PSEUDOTYPE(TIME) -> NUMBER + HOURS = 133; // PSEUDOTYPE(TIME) -> NUMBER + MINUTES = 134; // PSEUDOTYPE(TIME) -> NUMBER + SECONDS = 135; // PSEUDOTYPE(TIME) -> NUMBER + + // Construct a time from a date and optional timezone or a + // date+time and optional timezone. + TIME = 136; // NUMBER, NUMBER, NUMBER, STRING -> PSEUDOTYPE(TIME) | + // NUMBER, NUMBER, NUMBER, NUMBER, NUMBER, NUMBER, STRING -> PSEUDOTYPE(TIME) | + + // Constants for ISO 8601 days of the week. + MONDAY = 107; // -> 1 + TUESDAY = 108; // -> 2 + WEDNESDAY = 109; // -> 3 + THURSDAY = 110; // -> 4 + FRIDAY = 111; // -> 5 + SATURDAY = 112; // -> 6 + SUNDAY = 113; // -> 7 + + // Constants for ISO 8601 months. + JANUARY = 114; // -> 1 + FEBRUARY = 115; // -> 2 + MARCH = 116; // -> 3 + APRIL = 117; // -> 4 + MAY = 118; // -> 5 + JUNE = 119; // -> 6 + JULY = 120; // -> 7 + AUGUST = 121; // -> 8 + SEPTEMBER = 122; // -> 9 + OCTOBER = 123; // -> 10 + NOVEMBER = 124; // -> 11 + DECEMBER = 125; // -> 12 + + // Indicates to MERGE to replace the other object rather than merge it. + LITERAL = 137; // JSON -> Merging + + // SEQUENCE, STRING -> GROUPED_SEQUENCE | SEQUENCE, FUNCTION -> GROUPED_SEQUENCE + GROUP = 144; + SUM = 145; + AVG = 146; + MIN = 147; + MAX = 148; + + // `str.split()` splits on whitespace + // `str.split(" ")` splits on spaces only + // `str.split(" ", 5)` splits on spaces with at most 5 results + // `str.split(nil, 5)` splits on whitespace with at most 5 results + SPLIT = 149; // STRING -> ARRAY | STRING, STRING -> ARRAY | STRING, STRING, NUMBER -> ARRAY | STRING, NULL, NUMBER -> ARRAY + + UNGROUP = 150; // GROUPED_DATA -> ARRAY + + // Takes a range of numbers and returns a random number within the range + RANDOM = 151; // NUMBER, NUMBER {float:BOOL} -> DATUM + + CHANGES = 152; // TABLE -> STREAM + ARGS = 154; // ARRAY -> SPECIAL (used to splice arguments) + + // BINARY is client-only at the moment, it is not supported on the server + BINARY = 155; // STRING -> PSEUDOTYPE(BINARY) + + GEOJSON = 157; // OBJECT -> PSEUDOTYPE(GEOMETRY) + TO_GEOJSON = 158; // PSEUDOTYPE(GEOMETRY) -> OBJECT + POINT = 159; // NUMBER, NUMBER -> PSEUDOTYPE(GEOMETRY) + LINE = 160; // (ARRAY | PSEUDOTYPE(GEOMETRY))... -> PSEUDOTYPE(GEOMETRY) + POLYGON = 161; // (ARRAY | PSEUDOTYPE(GEOMETRY))... -> PSEUDOTYPE(GEOMETRY) + DISTANCE = 162; // PSEUDOTYPE(GEOMETRY), PSEUDOTYPE(GEOMETRY) {geo_system:STRING, unit:STRING} -> NUMBER + INTERSECTS = 163; // PSEUDOTYPE(GEOMETRY), PSEUDOTYPE(GEOMETRY) -> BOOL + INCLUDES = 164; // PSEUDOTYPE(GEOMETRY), PSEUDOTYPE(GEOMETRY) -> BOOL + CIRCLE = 165; // PSEUDOTYPE(GEOMETRY), NUMBER {num_vertices:NUMBER, geo_system:STRING, unit:STRING, fill:BOOL} -> PSEUDOTYPE(GEOMETRY) + GET_INTERSECTING = 166; // TABLE, PSEUDOTYPE(GEOMETRY) {index:!STRING} -> StreamSelection + FILL = 167; // PSEUDOTYPE(GEOMETRY) -> PSEUDOTYPE(GEOMETRY) + GET_NEAREST = 168; // TABLE, PSEUDOTYPE(GEOMETRY) {index:!STRING, max_results:NUM, max_dist:NUM, geo_system:STRING, unit:STRING} -> ARRAY + POLYGON_SUB = 171; // PSEUDOTYPE(GEOMETRY), PSEUDOTYPE(GEOMETRY) -> PSEUDOTYPE(GEOMETRY) + + // Constants for specifying key ranges + MINVAL = 180; + MAXVAL = 181; + } + optional TermType type = 1; + + // This is only used when type is DATUM. + optional Datum datum = 2; + + repeated Term args = 3; // Holds the positional arguments of the query. + message AssocPair { + optional string key = 1; + optional Term val = 2; + } + repeated AssocPair optargs = 4; // Holds the optional arguments of the query. + // (Note that the order of the optional arguments doesn't matter; think of a + // Hash.) + + extensions 10000 to 20000; +} + +//////////////////////////////////////////////////////////////////////////////// +// EXAMPLE // +//////////////////////////////////////////////////////////////////////////////// +// ```ruby +// r.table('tbl', {:use_outdated => true}).insert([{:id => 0}, {:id => 1}]) +// ``` +// Would turn into: +// Term { +// type = INSERT; +// args = [Term { +// type = TABLE; +// args = [Term { +// type = DATUM; +// datum = Datum { type = R_STR; r_str = "tbl"; }; +// }]; +// optargs = [["use_outdated", +// Term { +// type = DATUM; +// datum = Datum { type = R_BOOL; r_bool = true; }; +// }]]; +// }, +// Term { +// type = MAKE_ARRAY; +// args = [Term { +// type = DATUM; +// datum = Datum { type = R_OBJECT; r_object = [["id", 0]]; }; +// }, +// Term { +// type = DATUM; +// datum = Datum { type = R_OBJECT; r_object = [["id", 1]]; }; +// }]; +// }] +// } +// And the server would reply: +// Response { +// type = SUCCESS_ATOM; +// token = 1; +// response = [Datum { type = R_OBJECT; r_object = [["inserted", 2]]; }]; +// } +// Or, if there were an error: +// Response { +// type = RUNTIME_ERROR; +// token = 1; +// response = [Datum { type = R_STR; r_str = "The table `tbl` doesn't exist!"; }]; +// backtrace = [Frame { type = POS; pos = 0; }, Frame { type = POS; pos = 0; }]; +// } diff --git a/Godeps/_workspace/src/github.com/dancannon/gorethink/types/geometry.go b/Godeps/_workspace/src/github.com/dancannon/gorethink/types/geometry.go new file mode 100644 index 0000000000000..00ff80f0d5a07 --- /dev/null +++ b/Godeps/_workspace/src/github.com/dancannon/gorethink/types/geometry.go @@ -0,0 +1,225 @@ +package types + +import ( + "fmt" +) + +type Geometry struct { + Type string + Point Point + Line Line + Lines Lines +} + +func (g Geometry) MarshalRQL() (interface{}, error) { + switch g.Type { + case "Point": + return g.Point.MarshalRQL() + case "LineString": + return g.Line.MarshalRQL() + case "Polygon": + return g.Lines.MarshalRQL() + default: + return nil, fmt.Errorf("pseudo-type GEOMETRY object field 'type' %s is not valid", g.Type) + } +} + +func (g *Geometry) UnmarshalRQL(data interface{}) error { + if data, ok := data.(Geometry); ok { + g.Type = data.Type + g.Point = data.Point + g.Line = data.Line + g.Lines = data.Lines + + return nil + } + + m, ok := data.(map[string]interface{}) + if !ok { + return fmt.Errorf("pseudo-type GEOMETRY object is not valid") + } + + typ, ok := m["type"] + if !ok { + return fmt.Errorf("pseudo-type GEOMETRY object is not valid, expects 'type' field") + } + coords, ok := m["coordinates"] + if !ok { + return fmt.Errorf("pseudo-type GEOMETRY object is not valid, expects 'coordinates' field") + } + + var err error + switch typ { + case "Point": + g.Type = "Point" + g.Point, err = UnmarshalPoint(coords) + case "LineString": + g.Type = "LineString" + g.Line, err = UnmarshalLineString(coords) + case "Polygon": + g.Type = "Polygon" + g.Lines, err = UnmarshalPolygon(coords) + default: + return fmt.Errorf("pseudo-type GEOMETRY object has invalid type") + } + + if err != nil { + return err + } + + return nil +} + +type Point struct { + Lon float64 + Lat float64 +} +type Line []Point +type Lines []Line + +func (p Point) Coords() interface{} { + return []interface{}{p.Lon, p.Lat} +} + +func (p Point) MarshalRQL() (interface{}, error) { + return map[string]interface{}{ + "$reql_type$": "GEOMETRY", + "coordinates": p.Coords(), + "type": "Point", + }, nil +} + +func (p *Point) UnmarshalRQL(data interface{}) error { + g := &Geometry{} + err := g.UnmarshalRQL(data) + if err != nil { + return err + } + if g.Type != "Point" { + return fmt.Errorf("pseudo-type GEOMETRY object has type %s, expected type %s", g.Type, "Point") + } + + p.Lat = g.Point.Lat + p.Lon = g.Point.Lon + + return nil +} + +func (l Line) Coords() interface{} { + coords := make([]interface{}, len(l)) + for i, point := range l { + coords[i] = point.Coords() + } + return coords +} + +func (l Line) MarshalRQL() (interface{}, error) { + return map[string]interface{}{ + "$reql_type$": "GEOMETRY", + "coordinates": l.Coords(), + "type": "LineString", + }, nil +} + +func (l *Line) UnmarshalRQL(data interface{}) error { + g := &Geometry{} + err := g.UnmarshalRQL(data) + if err != nil { + return err + } + if g.Type != "LineString" { + return fmt.Errorf("pseudo-type GEOMETRY object has type %s, expected type %s", g.Type, "LineString") + } + + *l = g.Line + + return nil +} + +func (l Lines) Coords() interface{} { + coords := make([]interface{}, len(l)) + for i, line := range l { + coords[i] = line.Coords() + } + return coords +} + +func (l Lines) MarshalRQL() (interface{}, error) { + return map[string]interface{}{ + "$reql_type$": "GEOMETRY", + "coordinates": l.Coords(), + "type": "Polygon", + }, nil +} + +func (l *Lines) UnmarshalRQL(data interface{}) error { + g := &Geometry{} + err := g.UnmarshalRQL(data) + if err != nil { + return err + } + if g.Type != "Polygon" { + return fmt.Errorf("pseudo-type GEOMETRY object has type %s, expected type %s", g.Type, "Polygon") + } + + *l = g.Lines + + return nil +} + +func UnmarshalPoint(v interface{}) (Point, error) { + coords, ok := v.([]interface{}) + if !ok { + return Point{}, fmt.Errorf("pseudo-type GEOMETRY object field 'coordinates' is not valid") + } + if len(coords) != 2 { + return Point{}, fmt.Errorf("pseudo-type GEOMETRY object field 'coordinates' is not valid") + } + lon, ok := coords[0].(float64) + if !ok { + return Point{}, fmt.Errorf("pseudo-type GEOMETRY object field 'coordinates' is not valid") + } + lat, ok := coords[1].(float64) + if !ok { + return Point{}, fmt.Errorf("pseudo-type GEOMETRY object field 'coordinates' is not valid") + } + + return Point{ + Lon: lon, + Lat: lat, + }, nil +} + +func UnmarshalLineString(v interface{}) (Line, error) { + points, ok := v.([]interface{}) + if !ok { + return Line{}, fmt.Errorf("pseudo-type GEOMETRY object field 'coordinates' is not valid") + } + + var err error + line := make(Line, len(points)) + for i, coords := range points { + line[i], err = UnmarshalPoint(coords) + if err != nil { + return Line{}, err + } + } + return line, nil +} + +func UnmarshalPolygon(v interface{}) (Lines, error) { + lines, ok := v.([]interface{}) + if !ok { + return Lines{}, fmt.Errorf("pseudo-type GEOMETRY object field 'coordinates' is not valid") + } + + var err error + polygon := make(Lines, len(lines)) + for i, line := range lines { + polygon[i], err = UnmarshalLineString(line) + if err != nil { + return Lines{}, err + } + } + return polygon, nil +} diff --git a/Godeps/_workspace/src/github.com/eapache/go-resiliency/breaker/README.md b/Godeps/_workspace/src/github.com/eapache/go-resiliency/breaker/README.md new file mode 100644 index 0000000000000..2d1b3d93225d8 --- /dev/null +++ b/Godeps/_workspace/src/github.com/eapache/go-resiliency/breaker/README.md @@ -0,0 +1,34 @@ +circuit-breaker +=============== + +[![Build Status](https://travis-ci.org/eapache/go-resiliency.svg?branch=master)](https://travis-ci.org/eapache/go-resiliency) +[![GoDoc](https://godoc.org/github.com/eapache/go-resiliency/breaker?status.svg)](https://godoc.org/github.com/eapache/go-resiliency/breaker) +[![Code of Conduct](https://img.shields.io/badge/code%20of%20conduct-active-blue.svg)](https://eapache.github.io/conduct.html) + +The circuit-breaker resiliency pattern for golang. + +Creating a breaker takes three parameters: +- error threshold (for opening the breaker) +- success threshold (for closing the breaker) +- timeout (how long to keep the breaker open) + +```go +b := breaker.New(3, 1, 5*time.Second) + +for { + result := b.Run(func() error { + // communicate with some external service and + // return an error if the communication failed + return nil + }) + + switch result { + case nil: + // success! + case breaker.ErrBreakerOpen: + // our function wasn't run because the breaker was open + default: + // some other error + } +} +``` diff --git a/Godeps/_workspace/src/github.com/eapache/go-resiliency/breaker/breaker.go b/Godeps/_workspace/src/github.com/eapache/go-resiliency/breaker/breaker.go new file mode 100644 index 0000000000000..f88ca7248b0fd --- /dev/null +++ b/Godeps/_workspace/src/github.com/eapache/go-resiliency/breaker/breaker.go @@ -0,0 +1,161 @@ +// Package breaker implements the circuit-breaker resiliency pattern for Go. +package breaker + +import ( + "errors" + "sync" + "sync/atomic" + "time" +) + +// ErrBreakerOpen is the error returned from Run() when the function is not executed +// because the breaker is currently open. +var ErrBreakerOpen = errors.New("circuit breaker is open") + +const ( + closed uint32 = iota + open + halfOpen +) + +// Breaker implements the circuit-breaker resiliency pattern +type Breaker struct { + errorThreshold, successThreshold int + timeout time.Duration + + lock sync.Mutex + state uint32 + errors, successes int + lastError time.Time +} + +// New constructs a new circuit-breaker that starts closed. +// From closed, the breaker opens if "errorThreshold" errors are seen +// without an error-free period of at least "timeout". From open, the +// breaker half-closes after "timeout". From half-open, the breaker closes +// after "successThreshold" consecutive successes, or opens on a single error. +func New(errorThreshold, successThreshold int, timeout time.Duration) *Breaker { + return &Breaker{ + errorThreshold: errorThreshold, + successThreshold: successThreshold, + timeout: timeout, + } +} + +// Run will either return ErrBreakerOpen immediately if the circuit-breaker is +// already open, or it will run the given function and pass along its return +// value. It is safe to call Run concurrently on the same Breaker. +func (b *Breaker) Run(work func() error) error { + state := atomic.LoadUint32(&b.state) + + if state == open { + return ErrBreakerOpen + } + + return b.doWork(state, work) +} + +// Go will either return ErrBreakerOpen immediately if the circuit-breaker is +// already open, or it will run the given function in a separate goroutine. +// If the function is run, Go will return nil immediately, and will *not* return +// the return value of the function. It is safe to call Go concurrently on the +// same Breaker. +func (b *Breaker) Go(work func() error) error { + state := atomic.LoadUint32(&b.state) + + if state == open { + return ErrBreakerOpen + } + + // errcheck complains about ignoring the error return value, but + // that's on purpose; if you want an error from a goroutine you have to + // get it over a channel or something + go b.doWork(state, work) + + return nil +} + +func (b *Breaker) doWork(state uint32, work func() error) error { + var panicValue interface{} + + result := func() error { + defer func() { + panicValue = recover() + }() + return work() + }() + + if result == nil && panicValue == nil && state == closed { + // short-circuit the normal, success path without contending + // on the lock + return nil + } + + // oh well, I guess we have to contend on the lock + b.processResult(result, panicValue) + + if panicValue != nil { + // as close as Go lets us come to a "rethrow" although unfortunately + // we lose the original panicing location + panic(panicValue) + } + + return result +} + +func (b *Breaker) processResult(result error, panicValue interface{}) { + b.lock.Lock() + defer b.lock.Unlock() + + if result == nil && panicValue == nil { + if b.state == halfOpen { + b.successes++ + if b.successes == b.successThreshold { + b.closeBreaker() + } + } + } else { + if b.errors > 0 { + expiry := b.lastError.Add(b.timeout) + if time.Now().After(expiry) { + b.errors = 0 + } + } + + switch b.state { + case closed: + b.errors++ + if b.errors == b.errorThreshold { + b.openBreaker() + } else { + b.lastError = time.Now() + } + case halfOpen: + b.openBreaker() + } + } +} + +func (b *Breaker) openBreaker() { + b.changeState(open) + go b.timer() +} + +func (b *Breaker) closeBreaker() { + b.changeState(closed) +} + +func (b *Breaker) timer() { + time.Sleep(b.timeout) + + b.lock.Lock() + defer b.lock.Unlock() + + b.changeState(halfOpen) +} + +func (b *Breaker) changeState(newState uint32) { + b.errors = 0 + b.successes = 0 + atomic.StoreUint32(&b.state, newState) +} diff --git a/Godeps/_workspace/src/github.com/eapache/go-resiliency/breaker/breaker_test.go b/Godeps/_workspace/src/github.com/eapache/go-resiliency/breaker/breaker_test.go new file mode 100644 index 0000000000000..b41308db60b48 --- /dev/null +++ b/Godeps/_workspace/src/github.com/eapache/go-resiliency/breaker/breaker_test.go @@ -0,0 +1,196 @@ +package breaker + +import ( + "errors" + "testing" + "time" +) + +var errSomeError = errors.New("errSomeError") + +func alwaysPanics() error { + panic("foo") +} + +func returnsError() error { + return errSomeError +} + +func returnsSuccess() error { + return nil +} + +func TestBreakerErrorExpiry(t *testing.T) { + breaker := New(2, 1, 1*time.Second) + + for i := 0; i < 3; i++ { + if err := breaker.Run(returnsError); err != errSomeError { + t.Error(err) + } + time.Sleep(1 * time.Second) + } + + for i := 0; i < 3; i++ { + if err := breaker.Go(returnsError); err != nil { + t.Error(err) + } + time.Sleep(1 * time.Second) + } +} + +func TestBreakerPanicsCountAsErrors(t *testing.T) { + breaker := New(3, 2, 1*time.Second) + + // three errors opens the breaker + for i := 0; i < 3; i++ { + func() { + defer func() { + val := recover() + if val.(string) != "foo" { + t.Error("incorrect panic") + } + }() + if err := breaker.Run(alwaysPanics); err != nil { + t.Error(err) + } + t.Error("shouldn't get here") + }() + } + + // breaker is open + for i := 0; i < 5; i++ { + if err := breaker.Run(returnsError); err != ErrBreakerOpen { + t.Error(err) + } + } +} + +func TestBreakerStateTransitions(t *testing.T) { + breaker := New(3, 2, 1*time.Second) + + // three errors opens the breaker + for i := 0; i < 3; i++ { + if err := breaker.Run(returnsError); err != errSomeError { + t.Error(err) + } + } + + // breaker is open + for i := 0; i < 5; i++ { + if err := breaker.Run(returnsError); err != ErrBreakerOpen { + t.Error(err) + } + } + + // wait for it to half-close + time.Sleep(2 * time.Second) + // one success works, but is not enough to fully close + if err := breaker.Run(returnsSuccess); err != nil { + t.Error(err) + } + // error works, but re-opens immediately + if err := breaker.Run(returnsError); err != errSomeError { + t.Error(err) + } + // breaker is open + if err := breaker.Run(returnsError); err != ErrBreakerOpen { + t.Error(err) + } + + // wait for it to half-close + time.Sleep(2 * time.Second) + // two successes is enough to close it for good + for i := 0; i < 2; i++ { + if err := breaker.Run(returnsSuccess); err != nil { + t.Error(err) + } + } + // error works + if err := breaker.Run(returnsError); err != errSomeError { + t.Error(err) + } + // breaker is still closed + if err := breaker.Run(returnsSuccess); err != nil { + t.Error(err) + } +} + +func TestBreakerAsyncStateTransitions(t *testing.T) { + breaker := New(3, 2, 1*time.Second) + + // three errors opens the breaker + for i := 0; i < 3; i++ { + if err := breaker.Go(returnsError); err != nil { + t.Error(err) + } + } + + // just enough to yield the scheduler and let the goroutines work off + time.Sleep(1 * time.Millisecond) + + // breaker is open + for i := 0; i < 5; i++ { + if err := breaker.Go(returnsError); err != ErrBreakerOpen { + t.Error(err) + } + } + + // wait for it to half-close + time.Sleep(2 * time.Second) + // one success works, but is not enough to fully close + if err := breaker.Go(returnsSuccess); err != nil { + t.Error(err) + } + // error works, but re-opens immediately + if err := breaker.Go(returnsError); err != nil { + t.Error(err) + } + // just enough to yield the scheduler and let the goroutines work off + time.Sleep(1 * time.Millisecond) + // breaker is open + if err := breaker.Go(returnsError); err != ErrBreakerOpen { + t.Error(err) + } + + // wait for it to half-close + time.Sleep(2 * time.Second) + // two successes is enough to close it for good + for i := 0; i < 2; i++ { + if err := breaker.Go(returnsSuccess); err != nil { + t.Error(err) + } + } + // just enough to yield the scheduler and let the goroutines work off + time.Sleep(1 * time.Millisecond) + // error works + if err := breaker.Go(returnsError); err != nil { + t.Error(err) + } + // just enough to yield the scheduler and let the goroutines work off + time.Sleep(1 * time.Millisecond) + // breaker is still closed + if err := breaker.Go(returnsSuccess); err != nil { + t.Error(err) + } +} + +func ExampleBreaker() { + breaker := New(3, 1, 5*time.Second) + + for { + result := breaker.Run(func() error { + // communicate with some external service and + // return an error if the communication failed + return nil + }) + + switch result { + case nil: + // success! + case ErrBreakerOpen: + // our function wasn't run because the breaker was open + default: + // some other error + } + } +} diff --git a/Godeps/_workspace/src/github.com/eapache/queue/.gitignore b/Godeps/_workspace/src/github.com/eapache/queue/.gitignore new file mode 100644 index 0000000000000..836562412fe8a --- /dev/null +++ b/Godeps/_workspace/src/github.com/eapache/queue/.gitignore @@ -0,0 +1,23 @@ +# Compiled Object files, Static and Dynamic libs (Shared Objects) +*.o +*.a +*.so + +# Folders +_obj +_test + +# Architecture specific extensions/prefixes +*.[568vq] +[568vq].out + +*.cgo1.go +*.cgo2.c +_cgo_defun.c +_cgo_gotypes.go +_cgo_export.* + +_testmain.go + +*.exe +*.test diff --git a/Godeps/_workspace/src/github.com/eapache/queue/.travis.yml b/Godeps/_workspace/src/github.com/eapache/queue/.travis.yml new file mode 100644 index 0000000000000..235a40a493fff --- /dev/null +++ b/Godeps/_workspace/src/github.com/eapache/queue/.travis.yml @@ -0,0 +1,7 @@ +language: go +sudo: false + +go: + - 1.2 + - 1.3 + - 1.4 diff --git a/Godeps/_workspace/src/github.com/eapache/queue/LICENSE b/Godeps/_workspace/src/github.com/eapache/queue/LICENSE new file mode 100644 index 0000000000000..d5f36dbcaaf61 --- /dev/null +++ b/Godeps/_workspace/src/github.com/eapache/queue/LICENSE @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2014 Evan Huus + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. \ No newline at end of file diff --git a/Godeps/_workspace/src/github.com/eapache/queue/README.md b/Godeps/_workspace/src/github.com/eapache/queue/README.md new file mode 100644 index 0000000000000..8e782335cd773 --- /dev/null +++ b/Godeps/_workspace/src/github.com/eapache/queue/README.md @@ -0,0 +1,16 @@ +Queue +===== + +[![Build Status](https://travis-ci.org/eapache/queue.svg)](https://travis-ci.org/eapache/queue) +[![GoDoc](https://godoc.org/github.com/eapache/queue?status.png)](https://godoc.org/github.com/eapache/queue) +[![Code of Conduct](https://img.shields.io/badge/code%20of%20conduct-active-blue.svg)](https://eapache.github.io/conduct.html) + +A fast Golang queue using a ring-buffer, based on the version suggested by Dariusz Górecki. +Using this instead of other, simpler, queue implementations (slice+append or linked list) provides +substantial memory and time benefits, and fewer GC pauses. + +The queue implemented here is as fast as it is in part because it is *not* thread-safe. + +Follows semantic versioning using https://gopkg.in/ - import from +[`gopkg.in/eapache/queue.v1`](https://gopkg.in/eapache/queue.v1) +for guaranteed API stability. diff --git a/Godeps/_workspace/src/github.com/eapache/queue/queue.go b/Godeps/_workspace/src/github.com/eapache/queue/queue.go new file mode 100644 index 0000000000000..2dc8d9395660f --- /dev/null +++ b/Godeps/_workspace/src/github.com/eapache/queue/queue.go @@ -0,0 +1,88 @@ +/* +Package queue provides a fast, ring-buffer queue based on the version suggested by Dariusz Górecki. +Using this instead of other, simpler, queue implementations (slice+append or linked list) provides +substantial memory and time benefits, and fewer GC pauses. + +The queue implemented here is as fast as it is for an additional reason: it is *not* thread-safe. +*/ +package queue + +const minQueueLen = 16 + +// Queue represents a single instance of the queue data structure. +type Queue struct { + buf []interface{} + head, tail, count int +} + +// New constructs and returns a new Queue. +func New() *Queue { + return &Queue{ + buf: make([]interface{}, minQueueLen), + } +} + +// Length returns the number of elements currently stored in the queue. +func (q *Queue) Length() int { + return q.count +} + +// resizes the queue to fit exactly twice its current contents +// this can result in shrinking if the queue is less than half-full +func (q *Queue) resize() { + newBuf := make([]interface{}, q.count*2) + + if q.tail > q.head { + copy(newBuf, q.buf[q.head:q.tail]) + } else { + n := copy(newBuf, q.buf[q.head:]) + copy(newBuf[n:], q.buf[:q.tail]) + } + + q.head = 0 + q.tail = q.count + q.buf = newBuf +} + +// Add puts an element on the end of the queue. +func (q *Queue) Add(elem interface{}) { + if q.count == len(q.buf) { + q.resize() + } + + q.buf[q.tail] = elem + q.tail = (q.tail + 1) % len(q.buf) + q.count++ +} + +// Peek returns the element at the head of the queue. This call panics +// if the queue is empty. +func (q *Queue) Peek() interface{} { + if q.count <= 0 { + panic("queue: Peek() called on empty queue") + } + return q.buf[q.head] +} + +// Get returns the element at index i in the queue. If the index is +// invalid, the call will panic. +func (q *Queue) Get(i int) interface{} { + if i < 0 || i >= q.count { + panic("queue: Get() called with index out of range") + } + return q.buf[(q.head+i)%len(q.buf)] +} + +// Remove removes the element from the front of the queue. If you actually +// want the element, call Peek first. This call panics if the queue is empty. +func (q *Queue) Remove() { + if q.count <= 0 { + panic("queue: Remove() called on empty queue") + } + q.buf[q.head] = nil + q.head = (q.head + 1) % len(q.buf) + q.count-- + if len(q.buf) > minQueueLen && q.count*4 == len(q.buf) { + q.resize() + } +} diff --git a/Godeps/_workspace/src/github.com/eapache/queue/queue_test.go b/Godeps/_workspace/src/github.com/eapache/queue/queue_test.go new file mode 100644 index 0000000000000..f2765c14d66b4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/eapache/queue/queue_test.go @@ -0,0 +1,162 @@ +package queue + +import "testing" + +func TestQueueSimple(t *testing.T) { + q := New() + + for i := 0; i < minQueueLen; i++ { + q.Add(i) + } + for i := 0; i < minQueueLen; i++ { + if q.Peek().(int) != i { + t.Error("peek", i, "had value", q.Peek()) + } + q.Remove() + } +} + +func TestQueueWrapping(t *testing.T) { + q := New() + + for i := 0; i < minQueueLen; i++ { + q.Add(i) + } + for i := 0; i < 3; i++ { + q.Remove() + q.Add(minQueueLen + i) + } + + for i := 0; i < minQueueLen; i++ { + if q.Peek().(int) != i+3 { + t.Error("peek", i, "had value", q.Peek()) + } + q.Remove() + } +} + +func TestQueueLength(t *testing.T) { + q := New() + + if q.Length() != 0 { + t.Error("empty queue length not 0") + } + + for i := 0; i < 1000; i++ { + q.Add(i) + if q.Length() != i+1 { + t.Error("adding: queue with", i, "elements has length", q.Length()) + } + } + for i := 0; i < 1000; i++ { + q.Remove() + if q.Length() != 1000-i-1 { + t.Error("removing: queue with", 1000-i-i, "elements has length", q.Length()) + } + } +} + +func TestQueueGet(t *testing.T) { + q := New() + + for i := 0; i < 1000; i++ { + q.Add(i) + for j := 0; j < q.Length(); j++ { + if q.Get(j).(int) != j { + t.Errorf("index %d doesn't contain %d", j, j) + } + } + } +} + +func TestQueueGetOutOfRangePanics(t *testing.T) { + q := New() + + q.Add(1) + q.Add(2) + q.Add(3) + + assertPanics(t, "should panic when negative index", func() { + q.Get(-1) + }) + + assertPanics(t, "should panic when index greater than length", func() { + q.Get(4) + }) +} + +func TestQueuePeekOutOfRangePanics(t *testing.T) { + q := New() + + assertPanics(t, "should panic when peeking empty queue", func() { + q.Peek() + }) + + q.Add(1) + q.Remove() + + assertPanics(t, "should panic when peeking emptied queue", func() { + q.Peek() + }) +} + +func TestQueueRemoveOutOfRangePanics(t *testing.T) { + q := New() + + assertPanics(t, "should panic when removing empty queue", func() { + q.Remove() + }) + + q.Add(1) + q.Remove() + + assertPanics(t, "should panic when removing emptied queue", func() { + q.Remove() + }) +} + +func assertPanics(t *testing.T, name string, f func()) { + defer func() { + if r := recover(); r == nil { + t.Errorf("%s: didn't panic as expected", name) + } + }() + + f() +} + +// General warning: Go's benchmark utility (go test -bench .) increases the number of +// iterations until the benchmarks take a reasonable amount of time to run; memory usage +// is *NOT* considered. On my machine, these benchmarks hit around ~1GB before they've had +// enough, but if you have less than that available and start swapping, then all bets are off. + +func BenchmarkQueueSerial(b *testing.B) { + q := New() + for i := 0; i < b.N; i++ { + q.Add(nil) + } + for i := 0; i < b.N; i++ { + q.Peek() + q.Remove() + } +} + +func BenchmarkQueueGet(b *testing.B) { + q := New() + for i := 0; i < b.N; i++ { + q.Add(i) + } + b.ResetTimer() + for i := 0; i < b.N; i++ { + q.Get(i) + } +} + +func BenchmarkQueueTickTock(b *testing.B) { + q := New() + for i := 0; i < b.N; i++ { + q.Add(nil) + q.Peek() + q.Remove() + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/.travis.yml b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/.travis.yml new file mode 100644 index 0000000000000..3926838aceead --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/.travis.yml @@ -0,0 +1,11 @@ +language: go +sudo: false +go: + - 1.3.1 + - 1.4 + - tip +env: + - GOARCH=amd64 + - GOARCH=386 +script: + - make test diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/AUTHORS b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/AUTHORS new file mode 100644 index 0000000000000..2febb1f039cb2 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/AUTHORS @@ -0,0 +1,91 @@ +# This is the official list of go-dockerclient authors for copyright purposes. + +Adam Bell-Hanssen +Aldrin Leal +Andreas Jaekle +Andrews Medina +Artem Sidorenko +Andy Goldstein +Ben Marini +Ben McCann +Brendan Fosberry +Brian Lalor +Brian Palmer +Burke Libbey +Carlos Diaz-Padron +Cezar Sa Espinola +Cheah Chu Yeow +cheneydeng +CMGS +Craig Jellick +Dan Williams +Daniel, Dao Quang Minh +Daniel Garcia +Darren Shepherd +Dave Choi +David Huie +Dawn Chen +Dinesh Subhraveti +Ed +Eric Anderson +Ewout Prangsma +Fabio Rehm +Fatih Arslan +Flavia Missi +Francisco Souza +Guillermo Álvarez Fernández +He Simei +Ivan Mikushin +James Bardin +Jari Kolehmainen +Jason Wilder +Jawher Moussa +Jean-Baptiste Dalido +Jeff Mitchell +Jeffrey Hulten +Johan Euphrosine +Kamil Domanski +Karan Misra +Kim, Hirokuni +Kyle Allan +Liron Levin +Liu Peng +Lucas Clemente +Lucas Weiblen +Mantas Matelis +Martin Sweeney +Máximo Cuadros Ortiz +Michal Fojtik +Mike Dillon +Mrunal Patel +Nick Ethier +Omeid Matten +Orivej Desh +Paul Bellamy +Paul Morie +Paul Weil +Peter Edge +Peter Jihoon Kim +Philippe Lafoucrière +Rafe Colton +Rob Miller +Robert Williamson +Salvador Gironès +Sam Rijs +Simon Eskildsen +Simon Menke +Skolos +Soulou +Sridhar Ratnakumar +Summer Mousa +Sunjin Lee +Tarsis Azevedo +Tim Schindler +Tobi Knaup +Tonic +ttyh061 +Victor Marmol +Vincenzo Prignano +Wiliam Souza +Ye Yin +Yuriy Bogdanov diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/DOCKER-LICENSE b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/DOCKER-LICENSE new file mode 100644 index 0000000000000..706634474870e --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/DOCKER-LICENSE @@ -0,0 +1,6 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + +You can find the Docker license at the following link: +https://raw.githubusercontent.com/docker/docker/master/LICENSE diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/LICENSE b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/LICENSE new file mode 100644 index 0000000000000..4e11de1007ae2 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/LICENSE @@ -0,0 +1,22 @@ +Copyright (c) 2015, go-dockerclient authors +All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + + * Redistributions of source code must retain the above copyright notice, +this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above copyright notice, +this list of conditions and the following disclaimer in the documentation +and/or other materials provided with the distribution. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE +FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL +DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR +SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, +OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/Makefile b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/Makefile new file mode 100644 index 0000000000000..b8c2c99b30d1d --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/Makefile @@ -0,0 +1,47 @@ +.PHONY: \ + all \ + vendor \ + lint \ + vet \ + fmt \ + fmtcheck \ + pretest \ + test \ + cov \ + clean + +SRCS = $(shell git ls-files '*.go' | grep -v '^external/') +PKGS = ./. ./testing + +all: test + +vendor: + @ go get -v github.com/mjibson/party + party -d external -c -u + +lint: + @ go get -v github.com/golang/lint/golint + $(foreach file,$(SRCS),golint $(file) || exit;) + +vet: + @-go get -v golang.org/x/tools/cmd/vet + $(foreach pkg,$(PKGS),go vet $(pkg);) + +fmt: + gofmt -w $(SRCS) + +fmtcheck: + $(foreach file,$(SRCS),gofmt $(file) | diff -u $(file) - || exit;) + +pretest: lint vet fmtcheck + +test: pretest + $(foreach pkg,$(PKGS),go test $(pkg) || exit;) + +cov: + @ go get -v github.com/axw/gocov/gocov + @ go get golang.org/x/tools/cmd/cover + gocov test | gocov report + +clean: + $(foreach pkg,$(PKGS),go clean $(pkg) || exit;) diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/README.markdown b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/README.markdown new file mode 100644 index 0000000000000..a124d0b45e0ff --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/README.markdown @@ -0,0 +1,106 @@ +# go-dockerclient + +[![Drone](https://drone.io/github.com/fsouza/go-dockerclient/status.png)](https://drone.io/github.com/fsouza/go-dockerclient/latest) +[![Travis](https://img.shields.io/travis/fsouza/go-dockerclient.svg?style=flat-square)](https://travis-ci.org/fsouza/go-dockerclient) +[![GoDoc](https://img.shields.io/badge/api-Godoc-blue.svg?style=flat-square)](https://godoc.org/github.com/fsouza/go-dockerclient) + +This package presents a client for the Docker remote API. It also provides +support for the extensions in the [Swarm API](https://docs.docker.com/swarm/API/). + +This package also provides support for docker's network API, which is a simple +passthrough to the libnetwork remote API. Note that docker's network API is +only available in docker 1.8 and above, and only enabled in docker if +DOCKER_EXPERIMENTAL is defined during the docker build process. + +For more details, check the [remote API documentation](http://docs.docker.com/en/latest/reference/api/docker_remote_api/). + +## Vendoring + +If you are having issues with Go 1.5 and have `GO15VENDOREXPERIMENT` set with an application that has go-dockerclient vendored, +please update your vendoring of go-dockerclient :) We recently moved the `vendor` directory to `external` so that go-dockerclient +is compatible with this configuration. See [338](https://github.com/fsouza/go-dockerclient/issues/338) and [339](https://github.com/fsouza/go-dockerclient/pull/339) +for details. + +## Example + +```go +package main + +import ( + "fmt" + + "github.com/fsouza/go-dockerclient" +) + +func main() { + endpoint := "unix:///var/run/docker.sock" + client, _ := docker.NewClient(endpoint) + imgs, _ := client.ListImages(docker.ListImagesOptions{All: false}) + for _, img := range imgs { + fmt.Println("ID: ", img.ID) + fmt.Println("RepoTags: ", img.RepoTags) + fmt.Println("Created: ", img.Created) + fmt.Println("Size: ", img.Size) + fmt.Println("VirtualSize: ", img.VirtualSize) + fmt.Println("ParentId: ", img.ParentID) + } +} +``` + +## Using with TLS + +In order to instantiate the client for a TLS-enabled daemon, you should use NewTLSClient, passing the endpoint and path for key and certificates as parameters. + +```go +package main + +import ( + "fmt" + + "github.com/fsouza/go-dockerclient" +) + +func main() { + endpoint := "tcp://[ip]:[port]" + path := os.Getenv("DOCKER_CERT_PATH") + ca := fmt.Sprintf("%s/ca.pem", path) + cert := fmt.Sprintf("%s/cert.pem", path) + key := fmt.Sprintf("%s/key.pem", path) + client, _ := docker.NewTLSClient(endpoint, cert, key, ca) + // use client +} +``` + +If using [docker-machine](https://docs.docker.com/machine/), or another application that exports environment variables +`DOCKER_HOST, DOCKER_TLS_VERIFY, DOCKER_CERT_PATH`, you can use NewClientFromEnv. + + +```go +package main + +import ( + "fmt" + + "github.com/fsouza/go-dockerclient" +) + +func main() { + client, _ := docker.NewClientFromEnv() + // use client +} +``` + +See the documentation for more details. + +## Developing + +All development commands can be seen in the [Makefile](Makefile). + +Commited code must pass: + +* [golint](https://github.com/golang/lint) +* [go vet](https://godoc.org/golang.org/x/tools/cmd/vet) +* [gofmt](https://golang.org/cmd/gofmt) +* [go test](https://golang.org/cmd/go/#hdr-Test_packages) + +Running `make test` will check all of these. If your editor does not automatically call gofmt, `make fmt` will format all go files in this repository. diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go new file mode 100644 index 0000000000000..fccd55740190e --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth.go @@ -0,0 +1,133 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "bytes" + "encoding/base64" + "encoding/json" + "errors" + "fmt" + "io" + "os" + "path" + "strings" +) + +var AuthParseError error = errors.New("Failed to read authentication from dockercfg") + +// AuthConfiguration represents authentication options to use in the PushImage +// method. It represents the authentication in the Docker index server. +type AuthConfiguration struct { + Username string `json:"username,omitempty"` + Password string `json:"password,omitempty"` + Email string `json:"email,omitempty"` + ServerAddress string `json:"serveraddress,omitempty"` +} + +// AuthConfigurations represents authentication options to use for the +// PushImage method accommodating the new X-Registry-Config header +type AuthConfigurations struct { + Configs map[string]AuthConfiguration `json:"configs"` +} + +// dockerConfig represents a registry authentation configuration from the +// .dockercfg file. +type dockerConfig struct { + Auth string `json:"auth"` + Email string `json:"email"` +} + +// NewAuthConfigurationsFromDockerCfg returns AuthConfigurations from the +// ~/.dockercfg file. +func NewAuthConfigurationsFromDockerCfg() (*AuthConfigurations, error) { + var r io.Reader + var err error + p := path.Join(os.Getenv("HOME"), ".docker", "config.json") + r, err = os.Open(p) + if err != nil { + p := path.Join(os.Getenv("HOME"), ".dockercfg") + r, err = os.Open(p) + if err != nil { + return nil, err + } + } + return NewAuthConfigurations(r) +} + +// NewAuthConfigurations returns AuthConfigurations from a JSON encoded string in the +// same format as the .dockercfg file. +func NewAuthConfigurations(r io.Reader) (*AuthConfigurations, error) { + var auth *AuthConfigurations + confs, err := parseDockerConfig(r) + if err != nil { + return nil, err + } + auth, err = authConfigs(confs) + if err != nil { + return nil, err + } + return auth, nil +} + +func parseDockerConfig(r io.Reader) (map[string]dockerConfig, error) { + buf := new(bytes.Buffer) + buf.ReadFrom(r) + byteData := buf.Bytes() + + var confsWrapper map[string]map[string]dockerConfig + if err := json.Unmarshal(byteData, &confsWrapper); err == nil { + if confs, ok := confsWrapper["auths"]; ok { + return confs, nil + } + } + + var confs map[string]dockerConfig + if err := json.Unmarshal(byteData, &confs); err != nil { + return nil, err + } + return confs, nil +} + +// authConfigs converts a dockerConfigs map to a AuthConfigurations object. +func authConfigs(confs map[string]dockerConfig) (*AuthConfigurations, error) { + c := &AuthConfigurations{ + Configs: make(map[string]AuthConfiguration), + } + for reg, conf := range confs { + data, err := base64.StdEncoding.DecodeString(conf.Auth) + if err != nil { + return nil, err + } + userpass := strings.Split(string(data), ":") + if len(userpass) != 2 { + return nil, AuthParseError + } + c.Configs[reg] = AuthConfiguration{ + Email: conf.Email, + Username: userpass[0], + Password: userpass[1], + ServerAddress: reg, + } + } + return c, nil +} + +// AuthCheck validates the given credentials. It returns nil if successful. +// +// See https://goo.gl/vPoEfJ for more details. +func (c *Client) AuthCheck(conf *AuthConfiguration) error { + if conf == nil { + return fmt.Errorf("conf is nil") + } + body, statusCode, err := c.do("POST", "/auth", doOptions{data: conf}) + if err != nil { + return err + } + if statusCode > 400 { + return fmt.Errorf("auth error (%d): %s", statusCode, body) + } + return nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth_test.go new file mode 100644 index 0000000000000..fc0ffab84a3f1 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/auth_test.go @@ -0,0 +1,91 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "encoding/base64" + "fmt" + "net/http" + "strings" + "testing" +) + +func TestAuthLegacyConfig(t *testing.T) { + auth := base64.StdEncoding.EncodeToString([]byte("user:pass")) + read := strings.NewReader(fmt.Sprintf(`{"docker.io":{"auth":"%s","email":"user@example.com"}}`, auth)) + ac, err := NewAuthConfigurations(read) + if err != nil { + t.Error(err) + } + c, ok := ac.Configs["docker.io"] + if !ok { + t.Error("NewAuthConfigurations: Expected Configs to contain docker.io") + } + if got, want := c.Email, "user@example.com"; got != want { + t.Errorf(`AuthConfigurations.Configs["docker.io"].Email: wrong result. Want %q. Got %q`, want, got) + } + if got, want := c.Username, "user"; got != want { + t.Errorf(`AuthConfigurations.Configs["docker.io"].Username: wrong result. Want %q. Got %q`, want, got) + } + if got, want := c.Password, "pass"; got != want { + t.Errorf(`AuthConfigurations.Configs["docker.io"].Password: wrong result. Want %q. Got %q`, want, got) + } + if got, want := c.ServerAddress, "docker.io"; got != want { + t.Errorf(`AuthConfigurations.Configs["docker.io"].ServerAddress: wrong result. Want %q. Got %q`, want, got) + } +} + +func TestAuthBadConfig(t *testing.T) { + auth := base64.StdEncoding.EncodeToString([]byte("userpass")) + read := strings.NewReader(fmt.Sprintf(`{"docker.io":{"auth":"%s","email":"user@example.com"}}`, auth)) + ac, err := NewAuthConfigurations(read) + if err != AuthParseError { + t.Errorf("Incorrect error returned %v\n", err) + } + if ac != nil { + t.Errorf("Invalid auth configuration returned, should be nil %v\n", ac) + } +} + +func TestAuthConfig(t *testing.T) { + auth := base64.StdEncoding.EncodeToString([]byte("user:pass")) + read := strings.NewReader(fmt.Sprintf(`{"auths":{"docker.io":{"auth":"%s","email":"user@example.com"}}}`, auth)) + ac, err := NewAuthConfigurations(read) + if err != nil { + t.Error(err) + } + c, ok := ac.Configs["docker.io"] + if !ok { + t.Error("NewAuthConfigurations: Expected Configs to contain docker.io") + } + if got, want := c.Email, "user@example.com"; got != want { + t.Errorf(`AuthConfigurations.Configs["docker.io"].Email: wrong result. Want %q. Got %q`, want, got) + } + if got, want := c.Username, "user"; got != want { + t.Errorf(`AuthConfigurations.Configs["docker.io"].Username: wrong result. Want %q. Got %q`, want, got) + } + if got, want := c.Password, "pass"; got != want { + t.Errorf(`AuthConfigurations.Configs["docker.io"].Password: wrong result. Want %q. Got %q`, want, got) + } + if got, want := c.ServerAddress, "docker.io"; got != want { + t.Errorf(`AuthConfigurations.Configs["docker.io"].ServerAddress: wrong result. Want %q. Got %q`, want, got) + } +} + +func TestAuthCheck(t *testing.T) { + fakeRT := &FakeRoundTripper{status: http.StatusOK} + client := newTestClient(fakeRT) + if err := client.AuthCheck(nil); err == nil { + t.Fatalf("expected error on nil auth config") + } + // test good auth + if err := client.AuthCheck(&AuthConfiguration{}); err != nil { + t.Fatal(err) + } + *fakeRT = FakeRoundTripper{status: http.StatusUnauthorized} + if err := client.AuthCheck(&AuthConfiguration{}); err == nil { + t.Fatal("expected failure from unauthorized auth") + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/build_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/build_test.go new file mode 100644 index 0000000000000..a4864db837037 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/build_test.go @@ -0,0 +1,144 @@ +package docker + +import ( + "bytes" + "io" + "io/ioutil" + "net/http" + "os" + "reflect" + "testing" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive" +) + +func TestBuildImageMultipleContextsError(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + opts := BuildImageOptions{ + Name: "testImage", + NoCache: true, + SuppressOutput: true, + RmTmpContainer: true, + ForceRmTmpContainer: true, + InputStream: &buf, + OutputStream: &buf, + ContextDir: "testing/data", + } + err := client.BuildImage(opts) + if err != ErrMultipleContexts { + t.Errorf("BuildImage: providing both InputStream and ContextDir should produce an error") + } +} + +func TestBuildImageContextDirDockerignoreParsing(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + opts := BuildImageOptions{ + Name: "testImage", + NoCache: true, + SuppressOutput: true, + RmTmpContainer: true, + ForceRmTmpContainer: true, + OutputStream: &buf, + ContextDir: "testing/data", + } + err := client.BuildImage(opts) + if err != nil { + t.Fatal(err) + } + reqBody := fakeRT.requests[0].Body + tmpdir, err := unpackBodyTarball(reqBody) + if err != nil { + t.Fatal(err) + } + + defer func() { + if err := os.RemoveAll(tmpdir); err != nil { + t.Fatal(err) + } + }() + + files, err := ioutil.ReadDir(tmpdir) + if err != nil { + t.Fatal(err) + } + + foundFiles := []string{} + for _, file := range files { + foundFiles = append(foundFiles, file.Name()) + } + + expectedFiles := []string{ + ".dockerignore", + "Dockerfile", + "barfile", + "ca.pem", + "cert.pem", + "key.pem", + "server.pem", + "serverkey.pem", + "symlink", + } + + if !reflect.DeepEqual(expectedFiles, foundFiles) { + t.Errorf( + "BuildImage: incorrect files sent in tarball to docker server\nexpected %+v, found %+v", + expectedFiles, foundFiles, + ) + } +} + +func TestBuildImageSendXRegistryConfig(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + opts := BuildImageOptions{ + Name: "testImage", + NoCache: true, + SuppressOutput: true, + RmTmpContainer: true, + ForceRmTmpContainer: true, + OutputStream: &buf, + ContextDir: "testing/data", + AuthConfigs: AuthConfigurations{ + Configs: map[string]AuthConfiguration{ + "quay.io": { + Username: "foo", + Password: "bar", + Email: "baz", + ServerAddress: "quay.io", + }, + }, + }, + } + + encodedConfig := "eyJjb25maWdzIjp7InF1YXkuaW8iOnsidXNlcm5hbWUiOiJmb28iLCJwYXNzd29yZCI6ImJhciIsImVtYWlsIjoiYmF6Iiwic2VydmVyYWRkcmVzcyI6InF1YXkuaW8ifX19Cg==" + + if err := client.BuildImage(opts); err != nil { + t.Fatal(err) + } + + xRegistryConfig := fakeRT.requests[0].Header["X-Registry-Config"][0] + if xRegistryConfig != encodedConfig { + t.Errorf( + "BuildImage: X-Registry-Config not set currectly: expected %q, got %q", + encodedConfig, + xRegistryConfig, + ) + } +} + +func unpackBodyTarball(req io.ReadCloser) (tmpdir string, err error) { + tmpdir, err = ioutil.TempDir("", "go-dockerclient-test") + if err != nil { + return + } + err = archive.Untar(req, tmpdir, &archive.TarOptions{ + Compression: archive.Uncompressed, + NoLchown: true, + }) + return +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/change.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/change.go new file mode 100644 index 0000000000000..e7b056c3f88af --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/change.go @@ -0,0 +1,43 @@ +// Copyright 2014 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import "fmt" + +// ChangeType is a type for constants indicating the type of change +// in a container +type ChangeType int + +const ( + // ChangeModify is the ChangeType for container modifications + ChangeModify ChangeType = iota + + // ChangeAdd is the ChangeType for additions to a container + ChangeAdd + + // ChangeDelete is the ChangeType for deletions from a container + ChangeDelete +) + +// Change represents a change in a container. +// +// See http://goo.gl/QkW9sH for more details. +type Change struct { + Path string + Kind ChangeType +} + +func (change *Change) String() string { + var kind string + switch change.Kind { + case ChangeModify: + kind = "C" + case ChangeAdd: + kind = "A" + case ChangeDelete: + kind = "D" + } + return fmt.Sprintf("%s %s", kind, change.Path) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/change_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/change_test.go new file mode 100644 index 0000000000000..7c2ec30f7b9b5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/change_test.go @@ -0,0 +1,26 @@ +// Copyright 2014 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "testing" +) + +func TestChangeString(t *testing.T) { + var tests = []struct { + change Change + expected string + }{ + {Change{"/etc/passwd", ChangeModify}, "C /etc/passwd"}, + {Change{"/etc/passwd", ChangeAdd}, "A /etc/passwd"}, + {Change{"/etc/passwd", ChangeDelete}, "D /etc/passwd"}, + {Change{"/etc/passwd", 33}, " /etc/passwd"}, + } + for _, tt := range tests { + if got := tt.change.String(); got != tt.expected { + t.Errorf("Change.String(): want %q. Got %q.", tt.expected, got) + } + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go new file mode 100644 index 0000000000000..986bbb3d28429 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client.go @@ -0,0 +1,835 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// Package docker provides a client for the Docker remote API. +// +// See http://goo.gl/G3plxW for more details on the remote API. +package docker + +import ( + "bufio" + "bytes" + "crypto/tls" + "crypto/x509" + "encoding/json" + "errors" + "fmt" + "io" + "io/ioutil" + "net" + "net/http" + "net/http/httputil" + "net/url" + "os" + "path/filepath" + "reflect" + "runtime" + "strconv" + "strings" + + "time" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts" + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir" + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy" +) + +const userAgent = "go-dockerclient" + +var ( + // ErrInvalidEndpoint is returned when the endpoint is not a valid HTTP URL. + ErrInvalidEndpoint = errors.New("invalid endpoint") + + // ErrConnectionRefused is returned when the client cannot connect to the given endpoint. + ErrConnectionRefused = errors.New("cannot connect to Docker endpoint") + + apiVersion112, _ = NewAPIVersion("1.12") +) + +// APIVersion is an internal representation of a version of the Remote API. +type APIVersion []int + +// NewAPIVersion returns an instance of APIVersion for the given string. +// +// The given string must be in the form .., where , +// and are integer numbers. +func NewAPIVersion(input string) (APIVersion, error) { + if !strings.Contains(input, ".") { + return nil, fmt.Errorf("Unable to parse version %q", input) + } + arr := strings.Split(input, ".") + ret := make(APIVersion, len(arr)) + var err error + for i, val := range arr { + ret[i], err = strconv.Atoi(val) + if err != nil { + return nil, fmt.Errorf("Unable to parse version %q: %q is not an integer", input, val) + } + } + return ret, nil +} + +func (version APIVersion) String() string { + var str string + for i, val := range version { + str += strconv.Itoa(val) + if i < len(version)-1 { + str += "." + } + } + return str +} + +// LessThan is a function for comparing APIVersion structs +func (version APIVersion) LessThan(other APIVersion) bool { + return version.compare(other) < 0 +} + +// LessThanOrEqualTo is a function for comparing APIVersion structs +func (version APIVersion) LessThanOrEqualTo(other APIVersion) bool { + return version.compare(other) <= 0 +} + +// GreaterThan is a function for comparing APIVersion structs +func (version APIVersion) GreaterThan(other APIVersion) bool { + return version.compare(other) > 0 +} + +// GreaterThanOrEqualTo is a function for comparing APIVersion structs +func (version APIVersion) GreaterThanOrEqualTo(other APIVersion) bool { + return version.compare(other) >= 0 +} + +func (version APIVersion) compare(other APIVersion) int { + for i, v := range version { + if i <= len(other)-1 { + otherVersion := other[i] + + if v < otherVersion { + return -1 + } else if v > otherVersion { + return 1 + } + } + } + if len(version) > len(other) { + return 1 + } + if len(version) < len(other) { + return -1 + } + return 0 +} + +// Client is the basic type of this package. It provides methods for +// interaction with the API. +type Client struct { + SkipServerVersionCheck bool + HTTPClient *http.Client + TLSConfig *tls.Config + + endpoint string + endpointURL *url.URL + eventMonitor *eventMonitoringState + requestedAPIVersion APIVersion + serverAPIVersion APIVersion + expectedAPIVersion APIVersion +} + +// NewClient returns a Client instance ready for communication with the given +// server endpoint. It will use the latest remote API version available in the +// server. +func NewClient(endpoint string) (*Client, error) { + client, err := NewVersionedClient(endpoint, "") + if err != nil { + return nil, err + } + client.SkipServerVersionCheck = true + return client, nil +} + +// NewTLSClient returns a Client instance ready for TLS communications with the givens +// server endpoint, key and certificates . It will use the latest remote API version +// available in the server. +func NewTLSClient(endpoint string, cert, key, ca string) (*Client, error) { + client, err := NewVersionedTLSClient(endpoint, cert, key, ca, "") + if err != nil { + return nil, err + } + client.SkipServerVersionCheck = true + return client, nil +} + +// NewTLSClientFromBytes returns a Client instance ready for TLS communications with the givens +// server endpoint, key and certificates (passed inline to the function as opposed to being +// read from a local file). It will use the latest remote API version available in the server. +func NewTLSClientFromBytes(endpoint string, certPEMBlock, keyPEMBlock, caPEMCert []byte) (*Client, error) { + client, err := NewVersionedTLSClientFromBytes(endpoint, certPEMBlock, keyPEMBlock, caPEMCert, "") + if err != nil { + return nil, err + } + client.SkipServerVersionCheck = true + return client, nil +} + +// NewVersionedClient returns a Client instance ready for communication with +// the given server endpoint, using a specific remote API version. +func NewVersionedClient(endpoint string, apiVersionString string) (*Client, error) { + u, err := parseEndpoint(endpoint, false) + if err != nil { + return nil, err + } + var requestedAPIVersion APIVersion + if strings.Contains(apiVersionString, ".") { + requestedAPIVersion, err = NewAPIVersion(apiVersionString) + if err != nil { + return nil, err + } + } + return &Client{ + HTTPClient: http.DefaultClient, + endpoint: endpoint, + endpointURL: u, + eventMonitor: new(eventMonitoringState), + requestedAPIVersion: requestedAPIVersion, + }, nil +} + +// NewVersionnedTLSClient has been DEPRECATED, please use NewVersionedTLSClient. +func NewVersionnedTLSClient(endpoint string, cert, key, ca, apiVersionString string) (*Client, error) { + return NewVersionedTLSClient(endpoint, cert, key, ca, apiVersionString) +} + +// NewVersionedTLSClient returns a Client instance ready for TLS communications with the givens +// server endpoint, key and certificates, using a specific remote API version. +func NewVersionedTLSClient(endpoint string, cert, key, ca, apiVersionString string) (*Client, error) { + certPEMBlock, err := ioutil.ReadFile(cert) + if err != nil { + return nil, err + } + keyPEMBlock, err := ioutil.ReadFile(key) + if err != nil { + return nil, err + } + caPEMCert, err := ioutil.ReadFile(ca) + if err != nil { + return nil, err + } + return NewVersionedTLSClientFromBytes(endpoint, certPEMBlock, keyPEMBlock, caPEMCert, apiVersionString) +} + +// NewClientFromEnv returns a Client instance ready for communication created from +// Docker's default logic for the environment variables DOCKER_HOST, DOCKER_TLS_VERIFY, and DOCKER_CERT_PATH. +// +// See https://github.com/docker/docker/blob/1f963af697e8df3a78217f6fdbf67b8123a7db94/docker/docker.go#L68. +// See https://github.com/docker/compose/blob/81707ef1ad94403789166d2fe042c8a718a4c748/compose/cli/docker_client.py#L7. +func NewClientFromEnv() (*Client, error) { + client, err := NewVersionedClientFromEnv("") + if err != nil { + return nil, err + } + client.SkipServerVersionCheck = true + return client, nil +} + +// NewVersionedClientFromEnv returns a Client instance ready for TLS communications created from +// Docker's default logic for the environment variables DOCKER_HOST, DOCKER_TLS_VERIFY, and DOCKER_CERT_PATH, +// and using a specific remote API version. +// +// See https://github.com/docker/docker/blob/1f963af697e8df3a78217f6fdbf67b8123a7db94/docker/docker.go#L68. +// See https://github.com/docker/compose/blob/81707ef1ad94403789166d2fe042c8a718a4c748/compose/cli/docker_client.py#L7. +func NewVersionedClientFromEnv(apiVersionString string) (*Client, error) { + dockerEnv, err := getDockerEnv() + if err != nil { + return nil, err + } + dockerHost := dockerEnv.dockerHost + if dockerEnv.dockerTLSVerify { + parts := strings.SplitN(dockerHost, "://", 2) + if len(parts) != 2 { + return nil, fmt.Errorf("could not split %s into two parts by ://", dockerHost) + } + dockerHost = fmt.Sprintf("https://%s", parts[1]) + cert := filepath.Join(dockerEnv.dockerCertPath, "cert.pem") + key := filepath.Join(dockerEnv.dockerCertPath, "key.pem") + ca := filepath.Join(dockerEnv.dockerCertPath, "ca.pem") + return NewVersionedTLSClient(dockerHost, cert, key, ca, apiVersionString) + } + return NewVersionedClient(dockerHost, apiVersionString) +} + +// NewVersionedTLSClientFromBytes returns a Client instance ready for TLS communications with the givens +// server endpoint, key and certificates (passed inline to the function as opposed to being +// read from a local file), using a specific remote API version. +func NewVersionedTLSClientFromBytes(endpoint string, certPEMBlock, keyPEMBlock, caPEMCert []byte, apiVersionString string) (*Client, error) { + u, err := parseEndpoint(endpoint, true) + if err != nil { + return nil, err + } + var requestedAPIVersion APIVersion + if strings.Contains(apiVersionString, ".") { + requestedAPIVersion, err = NewAPIVersion(apiVersionString) + if err != nil { + return nil, err + } + } + if certPEMBlock == nil || keyPEMBlock == nil { + return nil, errors.New("Both cert and key are required") + } + tlsCert, err := tls.X509KeyPair(certPEMBlock, keyPEMBlock) + if err != nil { + return nil, err + } + tlsConfig := &tls.Config{Certificates: []tls.Certificate{tlsCert}} + if caPEMCert == nil { + tlsConfig.InsecureSkipVerify = true + } else { + caPool := x509.NewCertPool() + if !caPool.AppendCertsFromPEM(caPEMCert) { + return nil, errors.New("Could not add RootCA pem") + } + tlsConfig.RootCAs = caPool + } + tr := &http.Transport{ + TLSClientConfig: tlsConfig, + } + if err != nil { + return nil, err + } + return &Client{ + HTTPClient: &http.Client{Transport: tr}, + TLSConfig: tlsConfig, + endpoint: endpoint, + endpointURL: u, + eventMonitor: new(eventMonitoringState), + requestedAPIVersion: requestedAPIVersion, + }, nil +} + +func (c *Client) checkAPIVersion() error { + serverAPIVersionString, err := c.getServerAPIVersionString() + if err != nil { + return err + } + c.serverAPIVersion, err = NewAPIVersion(serverAPIVersionString) + if err != nil { + return err + } + if c.requestedAPIVersion == nil { + c.expectedAPIVersion = c.serverAPIVersion + } else { + c.expectedAPIVersion = c.requestedAPIVersion + } + return nil +} + +// Ping pings the docker server +// +// See http://goo.gl/stJENm for more details. +func (c *Client) Ping() error { + path := "/_ping" + body, status, err := c.do("GET", path, doOptions{}) + if err != nil { + return err + } + if status != http.StatusOK { + return newError(status, body) + } + return nil +} + +func (c *Client) getServerAPIVersionString() (version string, err error) { + body, status, err := c.do("GET", "/version", doOptions{}) + if err != nil { + return "", err + } + if status != http.StatusOK { + return "", fmt.Errorf("Received unexpected status %d while trying to retrieve the server version", status) + } + var versionResponse map[string]interface{} + err = json.Unmarshal(body, &versionResponse) + if err != nil { + return "", err + } + if version, ok := (versionResponse["ApiVersion"]).(string); ok { + return version, nil + } + return "", nil +} + +type doOptions struct { + data interface{} + forceJSON bool +} + +func (c *Client) do(method, path string, doOptions doOptions) ([]byte, int, error) { + var params io.Reader + if doOptions.data != nil || doOptions.forceJSON { + buf, err := json.Marshal(doOptions.data) + if err != nil { + return nil, -1, err + } + params = bytes.NewBuffer(buf) + } + if path != "/version" && !c.SkipServerVersionCheck && c.expectedAPIVersion == nil { + err := c.checkAPIVersion() + if err != nil { + return nil, -1, err + } + } + req, err := http.NewRequest(method, c.getURL(path), params) + if err != nil { + return nil, -1, err + } + req.Header.Set("User-Agent", userAgent) + if doOptions.data != nil { + req.Header.Set("Content-Type", "application/json") + } else if method == "POST" { + req.Header.Set("Content-Type", "plain/text") + } + var resp *http.Response + protocol := c.endpointURL.Scheme + address := c.endpointURL.Path + if protocol == "unix" { + var dial net.Conn + dial, err = net.Dial(protocol, address) + if err != nil { + return nil, -1, err + } + defer dial.Close() + breader := bufio.NewReader(dial) + err = req.Write(dial) + if err != nil { + return nil, -1, err + } + resp, err = http.ReadResponse(breader, req) + } else { + resp, err = c.HTTPClient.Do(req) + } + if err != nil { + if strings.Contains(err.Error(), "connection refused") { + return nil, -1, ErrConnectionRefused + } + return nil, -1, err + } + defer resp.Body.Close() + body, err := ioutil.ReadAll(resp.Body) + if err != nil { + return nil, -1, err + } + if resp.StatusCode < 200 || resp.StatusCode >= 400 { + return nil, resp.StatusCode, newError(resp.StatusCode, body) + } + return body, resp.StatusCode, nil +} + +type streamOptions struct { + setRawTerminal bool + rawJSONStream bool + useJSONDecoder bool + headers map[string]string + in io.Reader + stdout io.Writer + stderr io.Writer + // timeout is the inital connection timeout + timeout time.Duration +} + +func (c *Client) stream(method, path string, streamOptions streamOptions) error { + if (method == "POST" || method == "PUT") && streamOptions.in == nil { + streamOptions.in = bytes.NewReader(nil) + } + if path != "/version" && !c.SkipServerVersionCheck && c.expectedAPIVersion == nil { + err := c.checkAPIVersion() + if err != nil { + return err + } + } + req, err := http.NewRequest(method, c.getURL(path), streamOptions.in) + if err != nil { + return err + } + req.Header.Set("User-Agent", userAgent) + if method == "POST" { + req.Header.Set("Content-Type", "plain/text") + } + for key, val := range streamOptions.headers { + req.Header.Set(key, val) + } + var resp *http.Response + protocol := c.endpointURL.Scheme + address := c.endpointURL.Path + if streamOptions.stdout == nil { + streamOptions.stdout = ioutil.Discard + } + if streamOptions.stderr == nil { + streamOptions.stderr = ioutil.Discard + } + if protocol == "unix" { + dial, err := net.Dial(protocol, address) + if err != nil { + return err + } + defer dial.Close() + breader := bufio.NewReader(dial) + err = req.Write(dial) + if err != nil { + return err + } + + // ReadResponse may hang if server does not replay + if streamOptions.timeout > 0 { + dial.SetDeadline(time.Now().Add(streamOptions.timeout)) + } + + if resp, err = http.ReadResponse(breader, req); err != nil { + // Cancel timeout for future I/O operations + if streamOptions.timeout > 0 { + dial.SetDeadline(time.Time{}) + } + if strings.Contains(err.Error(), "connection refused") { + return ErrConnectionRefused + } + return err + } + } else { + if resp, err = c.HTTPClient.Do(req); err != nil { + if strings.Contains(err.Error(), "connection refused") { + return ErrConnectionRefused + } + return err + } + } + defer resp.Body.Close() + if resp.StatusCode < 200 || resp.StatusCode >= 400 { + body, err := ioutil.ReadAll(resp.Body) + if err != nil { + return err + } + return newError(resp.StatusCode, body) + } + if streamOptions.useJSONDecoder || resp.Header.Get("Content-Type") == "application/json" { + // if we want to get raw json stream, just copy it back to output + // without decoding it + if streamOptions.rawJSONStream { + _, err = io.Copy(streamOptions.stdout, resp.Body) + return err + } + dec := json.NewDecoder(resp.Body) + for { + var m jsonMessage + if err := dec.Decode(&m); err == io.EOF { + break + } else if err != nil { + return err + } + if m.Stream != "" { + fmt.Fprint(streamOptions.stdout, m.Stream) + } else if m.Progress != "" { + fmt.Fprintf(streamOptions.stdout, "%s %s\r", m.Status, m.Progress) + } else if m.Error != "" { + return errors.New(m.Error) + } + if m.Status != "" { + fmt.Fprintln(streamOptions.stdout, m.Status) + } + } + } else { + if streamOptions.setRawTerminal { + _, err = io.Copy(streamOptions.stdout, resp.Body) + } else { + _, err = stdcopy.StdCopy(streamOptions.stdout, streamOptions.stderr, resp.Body) + } + return err + } + return nil +} + +type hijackOptions struct { + success chan struct{} + setRawTerminal bool + in io.Reader + stdout io.Writer + stderr io.Writer + data interface{} +} + +func (c *Client) hijack(method, path string, hijackOptions hijackOptions) error { + if path != "/version" && !c.SkipServerVersionCheck && c.expectedAPIVersion == nil { + err := c.checkAPIVersion() + if err != nil { + return err + } + } + + var params io.Reader + if hijackOptions.data != nil { + buf, err := json.Marshal(hijackOptions.data) + if err != nil { + return err + } + params = bytes.NewBuffer(buf) + } + + if hijackOptions.stdout == nil { + hijackOptions.stdout = ioutil.Discard + } + if hijackOptions.stderr == nil { + hijackOptions.stderr = ioutil.Discard + } + req, err := http.NewRequest(method, c.getURL(path), params) + if err != nil { + return err + } + req.Header.Set("Content-Type", "plain/text") + protocol := c.endpointURL.Scheme + address := c.endpointURL.Path + if protocol != "unix" { + protocol = "tcp" + address = c.endpointURL.Host + } + var dial net.Conn + if c.TLSConfig != nil && protocol != "unix" { + dial, err = tlsDial(protocol, address, c.TLSConfig) + if err != nil { + return err + } + } else { + dial, err = net.Dial(protocol, address) + if err != nil { + return err + } + } + clientconn := httputil.NewClientConn(dial, nil) + defer clientconn.Close() + clientconn.Do(req) + if hijackOptions.success != nil { + hijackOptions.success <- struct{}{} + <-hijackOptions.success + } + rwc, br := clientconn.Hijack() + defer rwc.Close() + errChanOut := make(chan error, 1) + errChanIn := make(chan error, 1) + exit := make(chan bool) + go func() { + defer close(exit) + defer close(errChanOut) + var err error + if hijackOptions.setRawTerminal { + // When TTY is ON, use regular copy + _, err = io.Copy(hijackOptions.stdout, br) + } else { + _, err = stdcopy.StdCopy(hijackOptions.stdout, hijackOptions.stderr, br) + } + errChanOut <- err + }() + go func() { + if hijackOptions.in != nil { + _, err := io.Copy(rwc, hijackOptions.in) + errChanIn <- err + } else { + errChanIn <- nil + } + rwc.(interface { + CloseWrite() error + }).CloseWrite() + }() + <-exit + errIn := <-errChanIn + errOut := <-errChanOut + if errIn != nil { + return errIn + } + return errOut +} + +func (c *Client) getURL(path string) string { + urlStr := strings.TrimRight(c.endpointURL.String(), "/") + if c.endpointURL.Scheme == "unix" { + urlStr = "" + } + + if c.requestedAPIVersion != nil { + return fmt.Sprintf("%s/v%s%s", urlStr, c.requestedAPIVersion, path) + } + return fmt.Sprintf("%s%s", urlStr, path) +} + +type jsonMessage struct { + Status string `json:"status,omitempty"` + Progress string `json:"progress,omitempty"` + Error string `json:"error,omitempty"` + Stream string `json:"stream,omitempty"` +} + +func queryString(opts interface{}) string { + if opts == nil { + return "" + } + value := reflect.ValueOf(opts) + if value.Kind() == reflect.Ptr { + value = value.Elem() + } + if value.Kind() != reflect.Struct { + return "" + } + items := url.Values(map[string][]string{}) + for i := 0; i < value.NumField(); i++ { + field := value.Type().Field(i) + if field.PkgPath != "" { + continue + } + key := field.Tag.Get("qs") + if key == "" { + key = strings.ToLower(field.Name) + } else if key == "-" { + continue + } + addQueryStringValue(items, key, value.Field(i)) + } + return items.Encode() +} + +func addQueryStringValue(items url.Values, key string, v reflect.Value) { + switch v.Kind() { + case reflect.Bool: + if v.Bool() { + items.Add(key, "1") + } + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + if v.Int() > 0 { + items.Add(key, strconv.FormatInt(v.Int(), 10)) + } + case reflect.Float32, reflect.Float64: + if v.Float() > 0 { + items.Add(key, strconv.FormatFloat(v.Float(), 'f', -1, 64)) + } + case reflect.String: + if v.String() != "" { + items.Add(key, v.String()) + } + case reflect.Ptr: + if !v.IsNil() { + if b, err := json.Marshal(v.Interface()); err == nil { + items.Add(key, string(b)) + } + } + case reflect.Map: + if len(v.MapKeys()) > 0 { + if b, err := json.Marshal(v.Interface()); err == nil { + items.Add(key, string(b)) + } + } + case reflect.Array, reflect.Slice: + vLen := v.Len() + if vLen > 0 { + for i := 0; i < vLen; i++ { + addQueryStringValue(items, key, v.Index(i)) + } + } + } +} + +// Error represents failures in the API. It represents a failure from the API. +type Error struct { + Status int + Message string +} + +func newError(status int, body []byte) *Error { + return &Error{Status: status, Message: string(body)} +} + +func (e *Error) Error() string { + return fmt.Sprintf("API error (%d): %s", e.Status, e.Message) +} + +func parseEndpoint(endpoint string, tls bool) (*url.URL, error) { + u, err := url.Parse(endpoint) + if err != nil { + return nil, ErrInvalidEndpoint + } + if tls { + u.Scheme = "https" + } + switch u.Scheme { + case "unix": + return u, nil + case "http", "https", "tcp": + _, port, err := net.SplitHostPort(u.Host) + if err != nil { + if e, ok := err.(*net.AddrError); ok { + if e.Err == "missing port in address" { + return u, nil + } + } + return nil, ErrInvalidEndpoint + } + number, err := strconv.ParseInt(port, 10, 64) + if err == nil && number > 0 && number < 65536 { + if u.Scheme == "tcp" { + if number == 2376 { + u.Scheme = "https" + } else { + u.Scheme = "http" + } + } + return u, nil + } + return nil, ErrInvalidEndpoint + default: + return nil, ErrInvalidEndpoint + } +} + +type dockerEnv struct { + dockerHost string + dockerTLSVerify bool + dockerCertPath string +} + +func getDockerEnv() (*dockerEnv, error) { + dockerHost := os.Getenv("DOCKER_HOST") + var err error + if dockerHost == "" { + dockerHost, err = getDefaultDockerHost() + if err != nil { + return nil, err + } + } + dockerTLSVerify := os.Getenv("DOCKER_TLS_VERIFY") != "" + var dockerCertPath string + if dockerTLSVerify { + dockerCertPath = os.Getenv("DOCKER_CERT_PATH") + if dockerCertPath == "" { + home := homedir.Get() + if home == "" { + return nil, errors.New("environment variable HOME must be set if DOCKER_CERT_PATH is not set") + } + dockerCertPath = filepath.Join(home, ".docker") + dockerCertPath, err = filepath.Abs(dockerCertPath) + if err != nil { + return nil, err + } + } + } + return &dockerEnv{ + dockerHost: dockerHost, + dockerTLSVerify: dockerTLSVerify, + dockerCertPath: dockerCertPath, + }, nil +} + +func getDefaultDockerHost() (string, error) { + var defaultHost string + if runtime.GOOS != "windows" { + // If we do not have a host, default to unix socket + defaultHost = fmt.Sprintf("unix://%s", opts.DefaultUnixSocket) + } else { + // If we do not have a host, default to TCP socket on Windows + defaultHost = fmt.Sprintf("tcp://%s:%d", opts.DefaultHTTPHost, opts.DefaultHTTPPort) + } + return opts.ValidateHost(defaultHost) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client_test.go new file mode 100644 index 0000000000000..c00c3d30c8e2b --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/client_test.go @@ -0,0 +1,422 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "fmt" + "io/ioutil" + "net" + "net/http" + "net/url" + "reflect" + "strconv" + "strings" + "testing" + "time" +) + +func TestNewAPIClient(t *testing.T) { + endpoint := "http://localhost:4243" + client, err := NewClient(endpoint) + if err != nil { + t.Fatal(err) + } + if client.endpoint != endpoint { + t.Errorf("Expected endpoint %s. Got %s.", endpoint, client.endpoint) + } + if client.HTTPClient != http.DefaultClient { + t.Errorf("Expected http.Client %#v. Got %#v.", http.DefaultClient, client.HTTPClient) + } + // test unix socket endpoints + endpoint = "unix:///var/run/docker.sock" + client, err = NewClient(endpoint) + if err != nil { + t.Fatal(err) + } + if client.endpoint != endpoint { + t.Errorf("Expected endpoint %s. Got %s.", endpoint, client.endpoint) + } + if !client.SkipServerVersionCheck { + t.Error("Expected SkipServerVersionCheck to be true, got false") + } + if client.requestedAPIVersion != nil { + t.Errorf("Expected requestedAPIVersion to be nil, got %#v.", client.requestedAPIVersion) + } +} + +func newTLSClient(endpoint string) (*Client, error) { + return NewTLSClient(endpoint, + "testing/data/cert.pem", + "testing/data/key.pem", + "testing/data/ca.pem") +} + +func TestNewTSLAPIClient(t *testing.T) { + endpoint := "https://localhost:4243" + client, err := newTLSClient(endpoint) + if err != nil { + t.Fatal(err) + } + if client.endpoint != endpoint { + t.Errorf("Expected endpoint %s. Got %s.", endpoint, client.endpoint) + } + if !client.SkipServerVersionCheck { + t.Error("Expected SkipServerVersionCheck to be true, got false") + } + if client.requestedAPIVersion != nil { + t.Errorf("Expected requestedAPIVersion to be nil, got %#v.", client.requestedAPIVersion) + } +} + +func TestNewVersionedClient(t *testing.T) { + endpoint := "http://localhost:4243" + client, err := NewVersionedClient(endpoint, "1.12") + if err != nil { + t.Fatal(err) + } + if client.endpoint != endpoint { + t.Errorf("Expected endpoint %s. Got %s.", endpoint, client.endpoint) + } + if client.HTTPClient != http.DefaultClient { + t.Errorf("Expected http.Client %#v. Got %#v.", http.DefaultClient, client.HTTPClient) + } + if reqVersion := client.requestedAPIVersion.String(); reqVersion != "1.12" { + t.Errorf("Wrong requestAPIVersion. Want %q. Got %q.", "1.12", reqVersion) + } + if client.SkipServerVersionCheck { + t.Error("Expected SkipServerVersionCheck to be false, got true") + } +} + +func TestNewTLSVersionedClient(t *testing.T) { + certPath := "testing/data/cert.pem" + keyPath := "testing/data/key.pem" + caPath := "testing/data/ca.pem" + endpoint := "https://localhost:4243" + client, err := NewVersionedTLSClient(endpoint, certPath, keyPath, caPath, "1.14") + if err != nil { + t.Fatal(err) + } + if client.endpoint != endpoint { + t.Errorf("Expected endpoint %s. Got %s.", endpoint, client.endpoint) + } + if reqVersion := client.requestedAPIVersion.String(); reqVersion != "1.14" { + t.Errorf("Wrong requestAPIVersion. Want %q. Got %q.", "1.14", reqVersion) + } + if client.SkipServerVersionCheck { + t.Error("Expected SkipServerVersionCheck to be false, got true") + } +} + +func TestNewTLSVersionedClientInvalidCA(t *testing.T) { + certPath := "testing/data/cert.pem" + keyPath := "testing/data/key.pem" + caPath := "testing/data/key.pem" + endpoint := "https://localhost:4243" + _, err := NewVersionedTLSClient(endpoint, certPath, keyPath, caPath, "1.14") + if err == nil { + t.Errorf("Expected invalid ca at %s", caPath) + } +} + +func TestNewClientInvalidEndpoint(t *testing.T) { + cases := []string{ + "htp://localhost:3243", "http://localhost:a", "localhost:8080", + "", "localhost", "http://localhost:8080:8383", "http://localhost:65536", + "https://localhost:-20", + } + for _, c := range cases { + client, err := NewClient(c) + if client != nil { + t.Errorf("Want client for invalid endpoint, got %#v.", client) + } + if !reflect.DeepEqual(err, ErrInvalidEndpoint) { + t.Errorf("NewClient(%q): Got invalid error for invalid endpoint. Want %#v. Got %#v.", c, ErrInvalidEndpoint, err) + } + } +} + +func TestNewTLSClient(t *testing.T) { + var tests = []struct { + endpoint string + expected string + }{ + {"tcp://localhost:2376", "https"}, + {"tcp://localhost:2375", "https"}, + {"tcp://localhost:4000", "https"}, + {"http://localhost:4000", "https"}, + } + + for _, tt := range tests { + client, err := newTLSClient(tt.endpoint) + if err != nil { + t.Error(err) + } + got := client.endpointURL.Scheme + if got != tt.expected { + t.Errorf("endpointURL.Scheme: Got %s. Want %s.", got, tt.expected) + } + } +} + +func TestGetURL(t *testing.T) { + var tests = []struct { + endpoint string + path string + expected string + }{ + {"http://localhost:4243/", "/", "http://localhost:4243/"}, + {"http://localhost:4243", "/", "http://localhost:4243/"}, + {"http://localhost:4243", "/containers/ps", "http://localhost:4243/containers/ps"}, + {"tcp://localhost:4243", "/containers/ps", "http://localhost:4243/containers/ps"}, + {"http://localhost:4243/////", "/", "http://localhost:4243/"}, + {"unix:///var/run/docker.socket", "/containers", "/containers"}, + } + for _, tt := range tests { + client, _ := NewClient(tt.endpoint) + client.endpoint = tt.endpoint + client.SkipServerVersionCheck = true + got := client.getURL(tt.path) + if got != tt.expected { + t.Errorf("getURL(%q): Got %s. Want %s.", tt.path, got, tt.expected) + } + } +} + +func TestError(t *testing.T) { + err := newError(400, []byte("bad parameter")) + expected := Error{Status: 400, Message: "bad parameter"} + if !reflect.DeepEqual(expected, *err) { + t.Errorf("Wrong error type. Want %#v. Got %#v.", expected, *err) + } + message := "API error (400): bad parameter" + if err.Error() != message { + t.Errorf("Wrong error message. Want %q. Got %q.", message, err.Error()) + } +} + +func TestQueryString(t *testing.T) { + v := float32(2.4) + f32QueryString := fmt.Sprintf("w=%s&x=10&y=10.35", strconv.FormatFloat(float64(v), 'f', -1, 64)) + jsonPerson := url.QueryEscape(`{"Name":"gopher","age":4}`) + var tests = []struct { + input interface{} + want string + }{ + {&ListContainersOptions{All: true}, "all=1"}, + {ListContainersOptions{All: true}, "all=1"}, + {ListContainersOptions{Before: "something"}, "before=something"}, + {ListContainersOptions{Before: "something", Since: "other"}, "before=something&since=other"}, + {ListContainersOptions{Filters: map[string][]string{"status": {"paused", "running"}}}, "filters=%7B%22status%22%3A%5B%22paused%22%2C%22running%22%5D%7D"}, + {dumb{X: 10, Y: 10.35000}, "x=10&y=10.35"}, + {dumb{W: v, X: 10, Y: 10.35000}, f32QueryString}, + {dumb{X: 10, Y: 10.35000, Z: 10}, "x=10&y=10.35&zee=10"}, + {dumb{v: 4, X: 10, Y: 10.35000}, "x=10&y=10.35"}, + {dumb{T: 10, Y: 10.35000}, "y=10.35"}, + {dumb{Person: &person{Name: "gopher", Age: 4}}, "p=" + jsonPerson}, + {nil, ""}, + {10, ""}, + {"not_a_struct", ""}, + } + for _, tt := range tests { + got := queryString(tt.input) + if got != tt.want { + t.Errorf("queryString(%v). Want %q. Got %q.", tt.input, tt.want, got) + } + } +} + +func TestNewAPIVersionFailures(t *testing.T) { + var tests = []struct { + input string + expectedError string + }{ + {"1-0", `Unable to parse version "1-0"`}, + {"1.0-beta", `Unable to parse version "1.0-beta": "0-beta" is not an integer`}, + } + for _, tt := range tests { + v, err := NewAPIVersion(tt.input) + if v != nil { + t.Errorf("Expected version, got %v.", v) + } + if err.Error() != tt.expectedError { + t.Errorf("NewAPIVersion(%q): wrong error. Want %q. Got %q", tt.input, tt.expectedError, err.Error()) + } + } +} + +func TestAPIVersions(t *testing.T) { + var tests = []struct { + a string + b string + expectedALessThanB bool + expectedALessThanOrEqualToB bool + expectedAGreaterThanB bool + expectedAGreaterThanOrEqualToB bool + }{ + {"1.11", "1.11", false, true, false, true}, + {"1.10", "1.11", true, true, false, false}, + {"1.11", "1.10", false, false, true, true}, + + {"1.9", "1.11", true, true, false, false}, + {"1.11", "1.9", false, false, true, true}, + + {"1.1.1", "1.1", false, false, true, true}, + {"1.1", "1.1.1", true, true, false, false}, + + {"2.1", "1.1.1", false, false, true, true}, + {"2.1", "1.3.1", false, false, true, true}, + {"1.1.1", "2.1", true, true, false, false}, + {"1.3.1", "2.1", true, true, false, false}, + } + + for _, tt := range tests { + a, _ := NewAPIVersion(tt.a) + b, _ := NewAPIVersion(tt.b) + + if tt.expectedALessThanB && !a.LessThan(b) { + t.Errorf("Expected %#v < %#v", a, b) + } + if tt.expectedALessThanOrEqualToB && !a.LessThanOrEqualTo(b) { + t.Errorf("Expected %#v <= %#v", a, b) + } + if tt.expectedAGreaterThanB && !a.GreaterThan(b) { + t.Errorf("Expected %#v > %#v", a, b) + } + if tt.expectedAGreaterThanOrEqualToB && !a.GreaterThanOrEqualTo(b) { + t.Errorf("Expected %#v >= %#v", a, b) + } + } +} + +func TestPing(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + err := client.Ping() + if err != nil { + t.Fatal(err) + } +} + +func TestPingFailing(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusInternalServerError} + client := newTestClient(fakeRT) + err := client.Ping() + if err == nil { + t.Fatal("Expected non nil error, got nil") + } + expectedErrMsg := "API error (500): " + if err.Error() != expectedErrMsg { + t.Fatalf("Expected error to be %q, got: %q", expectedErrMsg, err.Error()) + } +} + +func TestPingFailingWrongStatus(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusAccepted} + client := newTestClient(fakeRT) + err := client.Ping() + if err == nil { + t.Fatal("Expected non nil error, got nil") + } + expectedErrMsg := "API error (202): " + if err.Error() != expectedErrMsg { + t.Fatalf("Expected error to be %q, got: %q", expectedErrMsg, err.Error()) + } +} + +func TestPingErrorWithUnixSocket(t *testing.T) { + go func() { + li, err := net.Listen("unix", "/tmp/echo.sock") + if err != nil { + t.Fatal(err) + } + defer li.Close() + if err != nil { + t.Fatalf("Expected to get listner, but failed: %#v", err) + } + + fd, err := li.Accept() + if err != nil { + t.Fatalf("Expected to accept connection, but failed: %#v", err) + } + + buf := make([]byte, 512) + nr, err := fd.Read(buf) + + // Create invalid response message to occur error + data := buf[0:nr] + for i := 0; i < 10; i++ { + data[i] = 63 + } + + _, err = fd.Write(data) + if err != nil { + t.Fatalf("Expected to write to socket, but failed: %#v", err) + } + + return + }() + + // Wait for unix socket to listen + time.Sleep(10 * time.Millisecond) + + endpoint := "unix:///tmp/echo.sock" + u, _ := parseEndpoint(endpoint, false) + client := Client{ + HTTPClient: http.DefaultClient, + endpoint: endpoint, + endpointURL: u, + SkipServerVersionCheck: true, + } + + err := client.Ping() + if err == nil { + t.Fatal("Expected non nil error, got nil") + } +} + +type FakeRoundTripper struct { + message string + status int + header map[string]string + requests []*http.Request +} + +func (rt *FakeRoundTripper) RoundTrip(r *http.Request) (*http.Response, error) { + body := strings.NewReader(rt.message) + rt.requests = append(rt.requests, r) + res := &http.Response{ + StatusCode: rt.status, + Body: ioutil.NopCloser(body), + Header: make(http.Header), + } + for k, v := range rt.header { + res.Header.Set(k, v) + } + return res, nil +} + +func (rt *FakeRoundTripper) Reset() { + rt.requests = nil +} + +type person struct { + Name string + Age int `json:"age"` +} + +type dumb struct { + T int `qs:"-"` + v int + W float32 + X int + Y float64 + Z int `qs:"zee"` + Person *person `qs:"p"` +} + +type fakeEndpointURL struct { + Scheme string +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go new file mode 100644 index 0000000000000..89430975ba50d --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container.go @@ -0,0 +1,1058 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "bytes" + "encoding/json" + "errors" + "fmt" + "io" + "net/http" + "net/url" + "strconv" + "strings" + "time" +) + +// ErrContainerAlreadyExists is the error returned by CreateContainer when the +// container already exists. +var ErrContainerAlreadyExists = errors.New("container already exists") + +// ListContainersOptions specify parameters to the ListContainers function. +// +// See http://goo.gl/6Y4Gz7 for more details. +type ListContainersOptions struct { + All bool + Size bool + Limit int + Since string + Before string + Filters map[string][]string +} + +// APIPort is a type that represents a port mapping returned by the Docker API +type APIPort struct { + PrivatePort int64 `json:"PrivatePort,omitempty" yaml:"PrivatePort,omitempty"` + PublicPort int64 `json:"PublicPort,omitempty" yaml:"PublicPort,omitempty"` + Type string `json:"Type,omitempty" yaml:"Type,omitempty"` + IP string `json:"IP,omitempty" yaml:"IP,omitempty"` +} + +// APIContainers represents a container. +// +// See http://goo.gl/QeFH7U for more details. +type APIContainers struct { + ID string `json:"Id" yaml:"Id"` + Image string `json:"Image,omitempty" yaml:"Image,omitempty"` + Command string `json:"Command,omitempty" yaml:"Command,omitempty"` + Created int64 `json:"Created,omitempty" yaml:"Created,omitempty"` + Status string `json:"Status,omitempty" yaml:"Status,omitempty"` + Ports []APIPort `json:"Ports,omitempty" yaml:"Ports,omitempty"` + SizeRw int64 `json:"SizeRw,omitempty" yaml:"SizeRw,omitempty"` + SizeRootFs int64 `json:"SizeRootFs,omitempty" yaml:"SizeRootFs,omitempty"` + Names []string `json:"Names,omitempty" yaml:"Names,omitempty"` +} + +// ListContainers returns a slice of containers matching the given criteria. +// +// See http://goo.gl/6Y4Gz7 for more details. +func (c *Client) ListContainers(opts ListContainersOptions) ([]APIContainers, error) { + path := "/containers/json?" + queryString(opts) + body, _, err := c.do("GET", path, doOptions{}) + if err != nil { + return nil, err + } + var containers []APIContainers + err = json.Unmarshal(body, &containers) + if err != nil { + return nil, err + } + return containers, nil +} + +// Port represents the port number and the protocol, in the form +// /. For example: 80/tcp. +type Port string + +// Port returns the number of the port. +func (p Port) Port() string { + return strings.Split(string(p), "/")[0] +} + +// Proto returns the name of the protocol. +func (p Port) Proto() string { + parts := strings.Split(string(p), "/") + if len(parts) == 1 { + return "tcp" + } + return parts[1] +} + +// State represents the state of a container. +type State struct { + Running bool `json:"Running,omitempty" yaml:"Running,omitempty"` + Paused bool `json:"Paused,omitempty" yaml:"Paused,omitempty"` + Restarting bool `json:"Restarting,omitempty" yaml:"Restarting,omitempty"` + OOMKilled bool `json:"OOMKilled,omitempty" yaml:"OOMKilled,omitempty"` + Pid int `json:"Pid,omitempty" yaml:"Pid,omitempty"` + ExitCode int `json:"ExitCode,omitempty" yaml:"ExitCode,omitempty"` + Error string `json:"Error,omitempty" yaml:"Error,omitempty"` + StartedAt time.Time `json:"StartedAt,omitempty" yaml:"StartedAt,omitempty"` + FinishedAt time.Time `json:"FinishedAt,omitempty" yaml:"FinishedAt,omitempty"` +} + +// String returns the string representation of a state. +func (s *State) String() string { + if s.Running { + if s.Paused { + return "paused" + } + return fmt.Sprintf("Up %s", time.Now().UTC().Sub(s.StartedAt)) + } + return fmt.Sprintf("Exit %d", s.ExitCode) +} + +// PortBinding represents the host/container port mapping as returned in the +// `docker inspect` json +type PortBinding struct { + HostIP string `json:"HostIP,omitempty" yaml:"HostIP,omitempty"` + HostPort string `json:"HostPort,omitempty" yaml:"HostPort,omitempty"` +} + +// PortMapping represents a deprecated field in the `docker inspect` output, +// and its value as found in NetworkSettings should always be nil +type PortMapping map[string]string + +// NetworkSettings contains network-related information about a container +type NetworkSettings struct { + IPAddress string `json:"IPAddress,omitempty" yaml:"IPAddress,omitempty"` + IPPrefixLen int `json:"IPPrefixLen,omitempty" yaml:"IPPrefixLen,omitempty"` + MacAddress string `json:"MacAddress,omitempty" yaml:"MacAddress,omitempty"` + Gateway string `json:"Gateway,omitempty" yaml:"Gateway,omitempty"` + Bridge string `json:"Bridge,omitempty" yaml:"Bridge,omitempty"` + PortMapping map[string]PortMapping `json:"PortMapping,omitempty" yaml:"PortMapping,omitempty"` + Ports map[Port][]PortBinding `json:"Ports,omitempty" yaml:"Ports,omitempty"` + NetworkID string `json:"NetworkID,omitempty" yaml:"NetworkID,omitempty"` + EndpointID string `json:"EndpointID,omitempty" yaml:"EndpointID,omitempty"` + SandboxKey string `json:"SandboxKey,omitempty" yaml:"SandboxKey,omitempty"` + GlobalIPv6Address string `json:"GlobalIPv6Address,omitempty" yaml:"GlobalIPv6Address,omitempty"` + GlobalIPv6PrefixLen int `json:"GlobalIPv6PrefixLen,omitempty" yaml:"GlobalIPv6PrefixLen,omitempty"` + IPv6Gateway string `json:"IPv6Gateway,omitempty" yaml:"IPv6Gateway,omitempty"` + LinkLocalIPv6Address string `json:"LinkLocalIPv6Address,omitempty" yaml:"LinkLocalIPv6Address,omitempty"` + LinkLocalIPv6PrefixLen int `json:"LinkLocalIPv6PrefixLen,omitempty" yaml:"LinkLocalIPv6PrefixLen,omitempty"` + SecondaryIPAddresses []string `json:"SecondaryIPAddresses,omitempty" yaml:"SecondaryIPAddresses,omitempty"` + SecondaryIPv6Addresses []string `json:"SecondaryIPv6Addresses,omitempty" yaml:"SecondaryIPv6Addresses,omitempty"` +} + +// PortMappingAPI translates the port mappings as contained in NetworkSettings +// into the format in which they would appear when returned by the API +func (settings *NetworkSettings) PortMappingAPI() []APIPort { + var mapping []APIPort + for port, bindings := range settings.Ports { + p, _ := parsePort(port.Port()) + if len(bindings) == 0 { + mapping = append(mapping, APIPort{ + PublicPort: int64(p), + Type: port.Proto(), + }) + continue + } + for _, binding := range bindings { + p, _ := parsePort(port.Port()) + h, _ := parsePort(binding.HostPort) + mapping = append(mapping, APIPort{ + PrivatePort: int64(p), + PublicPort: int64(h), + Type: port.Proto(), + IP: binding.HostIP, + }) + } + } + return mapping +} + +func parsePort(rawPort string) (int, error) { + port, err := strconv.ParseUint(rawPort, 10, 16) + if err != nil { + return 0, err + } + return int(port), nil +} + +// Config is the list of configuration options used when creating a container. +// Config does not contain the options that are specific to starting a container on a +// given host. Those are contained in HostConfig +type Config struct { + Hostname string `json:"Hostname,omitempty" yaml:"Hostname,omitempty"` + Domainname string `json:"Domainname,omitempty" yaml:"Domainname,omitempty"` + User string `json:"User,omitempty" yaml:"User,omitempty"` + Memory int64 `json:"Memory,omitempty" yaml:"Memory,omitempty"` + MemorySwap int64 `json:"MemorySwap,omitempty" yaml:"MemorySwap,omitempty"` + CPUShares int64 `json:"CpuShares,omitempty" yaml:"CpuShares,omitempty"` + CPUSet string `json:"Cpuset,omitempty" yaml:"Cpuset,omitempty"` + AttachStdin bool `json:"AttachStdin,omitempty" yaml:"AttachStdin,omitempty"` + AttachStdout bool `json:"AttachStdout,omitempty" yaml:"AttachStdout,omitempty"` + AttachStderr bool `json:"AttachStderr,omitempty" yaml:"AttachStderr,omitempty"` + PortSpecs []string `json:"PortSpecs,omitempty" yaml:"PortSpecs,omitempty"` + ExposedPorts map[Port]struct{} `json:"ExposedPorts,omitempty" yaml:"ExposedPorts,omitempty"` + Tty bool `json:"Tty,omitempty" yaml:"Tty,omitempty"` + OpenStdin bool `json:"OpenStdin,omitempty" yaml:"OpenStdin,omitempty"` + StdinOnce bool `json:"StdinOnce,omitempty" yaml:"StdinOnce,omitempty"` + Env []string `json:"Env,omitempty" yaml:"Env,omitempty"` + Cmd []string `json:"Cmd" yaml:"Cmd"` + DNS []string `json:"Dns,omitempty" yaml:"Dns,omitempty"` // For Docker API v1.9 and below only + Image string `json:"Image,omitempty" yaml:"Image,omitempty"` + Volumes map[string]struct{} `json:"Volumes,omitempty" yaml:"Volumes,omitempty"` + VolumesFrom string `json:"VolumesFrom,omitempty" yaml:"VolumesFrom,omitempty"` + WorkingDir string `json:"WorkingDir,omitempty" yaml:"WorkingDir,omitempty"` + MacAddress string `json:"MacAddress,omitempty" yaml:"MacAddress,omitempty"` + Entrypoint []string `json:"Entrypoint" yaml:"Entrypoint"` + NetworkDisabled bool `json:"NetworkDisabled,omitempty" yaml:"NetworkDisabled,omitempty"` + SecurityOpts []string `json:"SecurityOpts,omitempty" yaml:"SecurityOpts,omitempty"` + OnBuild []string `json:"OnBuild,omitempty" yaml:"OnBuild,omitempty"` + Labels map[string]string `json:"Labels,omitempty" yaml:"Labels,omitempty"` +} + +// LogConfig defines the log driver type and the configuration for it. +type LogConfig struct { + Type string `json:"Type,omitempty" yaml:"Type,omitempty"` + Config map[string]string `json:"Config,omitempty" yaml:"Config,omitempty"` +} + +// ULimit defines system-wide resource limitations +// This can help a lot in system administration, e.g. when a user starts too many processes and therefore makes the system unresponsive for other users. +type ULimit struct { + Name string `json:"Name,omitempty" yaml:"Name,omitempty"` + Soft int64 `json:"Soft,omitempty" yaml:"Soft,omitempty"` + Hard int64 `json:"Hard,omitempty" yaml:"Hard,omitempty"` +} + +// SwarmNode containers information about which Swarm node the container is on +type SwarmNode struct { + ID string `json:"ID,omitempty" yaml:"ID,omitempty"` + IP string `json:"IP,omitempty" yaml:"IP,omitempty"` + Addr string `json:"Addr,omitempty" yaml:"Addr,omitempty"` + Name string `json:"Name,omitempty" yaml:"Name,omitempty"` + CPUs int64 `json:"CPUs,omitempty" yaml:"CPUs,omitempty"` + Memory int64 `json:"Memory,omitempty" yaml:"Memory,omitempty"` + Labels map[string]string `json:"Labels,omitempty" yaml:"Labels,omitempty"` +} + +// Container is the type encompasing everything about a container - its config, +// hostconfig, etc. +type Container struct { + ID string `json:"Id" yaml:"Id"` + + Created time.Time `json:"Created,omitempty" yaml:"Created,omitempty"` + + Path string `json:"Path,omitempty" yaml:"Path,omitempty"` + Args []string `json:"Args,omitempty" yaml:"Args,omitempty"` + + Config *Config `json:"Config,omitempty" yaml:"Config,omitempty"` + State State `json:"State,omitempty" yaml:"State,omitempty"` + Image string `json:"Image,omitempty" yaml:"Image,omitempty"` + + Node *SwarmNode `json:"Node,omitempty" yaml:"Node,omitempty"` + + NetworkSettings *NetworkSettings `json:"NetworkSettings,omitempty" yaml:"NetworkSettings,omitempty"` + + SysInitPath string `json:"SysInitPath,omitempty" yaml:"SysInitPath,omitempty"` + ResolvConfPath string `json:"ResolvConfPath,omitempty" yaml:"ResolvConfPath,omitempty"` + HostnamePath string `json:"HostnamePath,omitempty" yaml:"HostnamePath,omitempty"` + HostsPath string `json:"HostsPath,omitempty" yaml:"HostsPath,omitempty"` + LogPath string `json:"LogPath,omitempty" yaml:"LogPath,omitempty"` + Name string `json:"Name,omitempty" yaml:"Name,omitempty"` + Driver string `json:"Driver,omitempty" yaml:"Driver,omitempty"` + + Volumes map[string]string `json:"Volumes,omitempty" yaml:"Volumes,omitempty"` + VolumesRW map[string]bool `json:"VolumesRW,omitempty" yaml:"VolumesRW,omitempty"` + HostConfig *HostConfig `json:"HostConfig,omitempty" yaml:"HostConfig,omitempty"` + ExecIDs []string `json:"ExecIDs,omitempty" yaml:"ExecIDs,omitempty"` + + RestartCount int `json:"RestartCount,omitempty" yaml:"RestartCount,omitempty"` + + AppArmorProfile string `json:"AppArmorProfile,omitempty" yaml:"AppArmorProfile,omitempty"` +} + +// RenameContainerOptions specify parameters to the RenameContainer function. +// +// See http://goo.gl/L00hoj for more details. +type RenameContainerOptions struct { + // ID of container to rename + ID string `qs:"-"` + + // New name + Name string `json:"name,omitempty" yaml:"name,omitempty"` +} + +// RenameContainer updates and existing containers name +// +// See http://goo.gl/L00hoj for more details. +func (c *Client) RenameContainer(opts RenameContainerOptions) error { + _, _, err := c.do("POST", fmt.Sprintf("/containers/"+opts.ID+"/rename?%s", queryString(opts)), doOptions{}) + return err +} + +// InspectContainer returns information about a container by its ID. +// +// See http://goo.gl/CxVuJ5 for more details. +func (c *Client) InspectContainer(id string) (*Container, error) { + path := "/containers/" + id + "/json" + body, status, err := c.do("GET", path, doOptions{}) + if status == http.StatusNotFound { + return nil, &NoSuchContainer{ID: id} + } + if err != nil { + return nil, err + } + var container Container + err = json.Unmarshal(body, &container) + if err != nil { + return nil, err + } + return &container, nil +} + +// ContainerChanges returns changes in the filesystem of the given container. +// +// See http://goo.gl/QkW9sH for more details. +func (c *Client) ContainerChanges(id string) ([]Change, error) { + path := "/containers/" + id + "/changes" + body, status, err := c.do("GET", path, doOptions{}) + if status == http.StatusNotFound { + return nil, &NoSuchContainer{ID: id} + } + if err != nil { + return nil, err + } + var changes []Change + err = json.Unmarshal(body, &changes) + if err != nil { + return nil, err + } + return changes, nil +} + +// CreateContainerOptions specify parameters to the CreateContainer function. +// +// See http://goo.gl/2xxQQK for more details. +type CreateContainerOptions struct { + Name string + Config *Config `qs:"-"` + HostConfig *HostConfig `qs:"-"` +} + +// CreateContainer creates a new container, returning the container instance, +// or an error in case of failure. +// +// See http://goo.gl/mErxNp for more details. +func (c *Client) CreateContainer(opts CreateContainerOptions) (*Container, error) { + path := "/containers/create?" + queryString(opts) + body, status, err := c.do( + "POST", + path, + doOptions{ + data: struct { + *Config + HostConfig *HostConfig `json:"HostConfig,omitempty" yaml:"HostConfig,omitempty"` + }{ + opts.Config, + opts.HostConfig, + }, + }, + ) + + if status == http.StatusNotFound { + return nil, ErrNoSuchImage + } + if status == http.StatusConflict { + return nil, ErrContainerAlreadyExists + } + if err != nil { + return nil, err + } + var container Container + err = json.Unmarshal(body, &container) + if err != nil { + return nil, err + } + + container.Name = opts.Name + + return &container, nil +} + +// KeyValuePair is a type for generic key/value pairs as used in the Lxc +// configuration +type KeyValuePair struct { + Key string `json:"Key,omitempty" yaml:"Key,omitempty"` + Value string `json:"Value,omitempty" yaml:"Value,omitempty"` +} + +// RestartPolicy represents the policy for automatically restarting a container. +// +// Possible values are: +// +// - always: the docker daemon will always restart the container +// - on-failure: the docker daemon will restart the container on failures, at +// most MaximumRetryCount times +// - no: the docker daemon will not restart the container automatically +type RestartPolicy struct { + Name string `json:"Name,omitempty" yaml:"Name,omitempty"` + MaximumRetryCount int `json:"MaximumRetryCount,omitempty" yaml:"MaximumRetryCount,omitempty"` +} + +// AlwaysRestart returns a restart policy that tells the Docker daemon to +// always restart the container. +func AlwaysRestart() RestartPolicy { + return RestartPolicy{Name: "always"} +} + +// RestartOnFailure returns a restart policy that tells the Docker daemon to +// restart the container on failures, trying at most maxRetry times. +func RestartOnFailure(maxRetry int) RestartPolicy { + return RestartPolicy{Name: "on-failure", MaximumRetryCount: maxRetry} +} + +// NeverRestart returns a restart policy that tells the Docker daemon to never +// restart the container on failures. +func NeverRestart() RestartPolicy { + return RestartPolicy{Name: "no"} +} + +// Device represents a device mapping between the Docker host and the +// container. +type Device struct { + PathOnHost string `json:"PathOnHost,omitempty" yaml:"PathOnHost,omitempty"` + PathInContainer string `json:"PathInContainer,omitempty" yaml:"PathInContainer,omitempty"` + CgroupPermissions string `json:"CgroupPermissions,omitempty" yaml:"CgroupPermissions,omitempty"` +} + +// HostConfig contains the container options related to starting a container on +// a given host +type HostConfig struct { + Binds []string `json:"Binds,omitempty" yaml:"Binds,omitempty"` + CapAdd []string `json:"CapAdd,omitempty" yaml:"CapAdd,omitempty"` + CapDrop []string `json:"CapDrop,omitempty" yaml:"CapDrop,omitempty"` + ContainerIDFile string `json:"ContainerIDFile,omitempty" yaml:"ContainerIDFile,omitempty"` + LxcConf []KeyValuePair `json:"LxcConf,omitempty" yaml:"LxcConf,omitempty"` + Privileged bool `json:"Privileged,omitempty" yaml:"Privileged,omitempty"` + PortBindings map[Port][]PortBinding `json:"PortBindings,omitempty" yaml:"PortBindings,omitempty"` + Links []string `json:"Links,omitempty" yaml:"Links,omitempty"` + PublishAllPorts bool `json:"PublishAllPorts,omitempty" yaml:"PublishAllPorts,omitempty"` + DNS []string `json:"Dns,omitempty" yaml:"Dns,omitempty"` // For Docker API v1.10 and above only + DNSSearch []string `json:"DnsSearch,omitempty" yaml:"DnsSearch,omitempty"` + ExtraHosts []string `json:"ExtraHosts,omitempty" yaml:"ExtraHosts,omitempty"` + VolumesFrom []string `json:"VolumesFrom,omitempty" yaml:"VolumesFrom,omitempty"` + NetworkMode string `json:"NetworkMode,omitempty" yaml:"NetworkMode,omitempty"` + IpcMode string `json:"IpcMode,omitempty" yaml:"IpcMode,omitempty"` + PidMode string `json:"PidMode,omitempty" yaml:"PidMode,omitempty"` + UTSMode string `json:"UTSMode,omitempty" yaml:"UTSMode,omitempty"` + RestartPolicy RestartPolicy `json:"RestartPolicy,omitempty" yaml:"RestartPolicy,omitempty"` + Devices []Device `json:"Devices,omitempty" yaml:"Devices,omitempty"` + LogConfig LogConfig `json:"LogConfig,omitempty" yaml:"LogConfig,omitempty"` + ReadonlyRootfs bool `json:"ReadonlyRootfs,omitempty" yaml:"ReadonlyRootfs,omitempty"` + SecurityOpt []string `json:"SecurityOpt,omitempty" yaml:"SecurityOpt,omitempty"` + CgroupParent string `json:"CgroupParent,omitempty" yaml:"CgroupParent,omitempty"` + Memory int64 `json:"Memory,omitempty" yaml:"Memory,omitempty"` + MemorySwap int64 `json:"MemorySwap,omitempty" yaml:"MemorySwap,omitempty"` + CPUShares int64 `json:"CpuShares,omitempty" yaml:"CpuShares,omitempty"` + CPUSet string `json:"Cpuset,omitempty" yaml:"Cpuset,omitempty"` + CPUQuota int64 `json:"CpuQuota,omitempty" yaml:"CpuQuota,omitempty"` + CPUPeriod int64 `json:"CpuPeriod,omitempty" yaml:"CpuPeriod,omitempty"` + Ulimits []ULimit `json:"Ulimits,omitempty" yaml:"Ulimits,omitempty"` +} + +// StartContainer starts a container, returning an error in case of failure. +// +// See http://goo.gl/iM5GYs for more details. +func (c *Client) StartContainer(id string, hostConfig *HostConfig) error { + path := "/containers/" + id + "/start" + _, status, err := c.do("POST", path, doOptions{data: hostConfig, forceJSON: true}) + if status == http.StatusNotFound { + return &NoSuchContainer{ID: id, Err: err} + } + if status == http.StatusNotModified { + return &ContainerAlreadyRunning{ID: id} + } + if err != nil { + return err + } + return nil +} + +// StopContainer stops a container, killing it after the given timeout (in +// seconds). +// +// See http://goo.gl/EbcpXt for more details. +func (c *Client) StopContainer(id string, timeout uint) error { + path := fmt.Sprintf("/containers/%s/stop?t=%d", id, timeout) + _, status, err := c.do("POST", path, doOptions{}) + if status == http.StatusNotFound { + return &NoSuchContainer{ID: id} + } + if status == http.StatusNotModified { + return &ContainerNotRunning{ID: id} + } + if err != nil { + return err + } + return nil +} + +// RestartContainer stops a container, killing it after the given timeout (in +// seconds), during the stop process. +// +// See http://goo.gl/VOzR2n for more details. +func (c *Client) RestartContainer(id string, timeout uint) error { + path := fmt.Sprintf("/containers/%s/restart?t=%d", id, timeout) + _, status, err := c.do("POST", path, doOptions{}) + if status == http.StatusNotFound { + return &NoSuchContainer{ID: id} + } + if err != nil { + return err + } + return nil +} + +// PauseContainer pauses the given container. +// +// See http://goo.gl/AM5t42 for more details. +func (c *Client) PauseContainer(id string) error { + path := fmt.Sprintf("/containers/%s/pause", id) + _, status, err := c.do("POST", path, doOptions{}) + if status == http.StatusNotFound { + return &NoSuchContainer{ID: id} + } + if err != nil { + return err + } + return nil +} + +// UnpauseContainer unpauses the given container. +// +// See http://goo.gl/eBrNSL for more details. +func (c *Client) UnpauseContainer(id string) error { + path := fmt.Sprintf("/containers/%s/unpause", id) + _, status, err := c.do("POST", path, doOptions{}) + if status == http.StatusNotFound { + return &NoSuchContainer{ID: id} + } + if err != nil { + return err + } + return nil +} + +// TopResult represents the list of processes running in a container, as +// returned by /containers//top. +// +// See http://goo.gl/qu4gse for more details. +type TopResult struct { + Titles []string + Processes [][]string +} + +// TopContainer returns processes running inside a container +// +// See http://goo.gl/qu4gse for more details. +func (c *Client) TopContainer(id string, psArgs string) (TopResult, error) { + var args string + var result TopResult + if psArgs != "" { + args = fmt.Sprintf("?ps_args=%s", psArgs) + } + path := fmt.Sprintf("/containers/%s/top%s", id, args) + body, status, err := c.do("GET", path, doOptions{}) + if status == http.StatusNotFound { + return result, &NoSuchContainer{ID: id} + } + if err != nil { + return result, err + } + err = json.Unmarshal(body, &result) + if err != nil { + return result, err + } + return result, nil +} + +// Stats represents container statistics, returned by /containers//stats. +// +// See http://goo.gl/DFMiYD for more details. +type Stats struct { + Read time.Time `json:"read,omitempty" yaml:"read,omitempty"` + Network struct { + RxDropped uint64 `json:"rx_dropped,omitempty" yaml:"rx_dropped,omitempty"` + RxBytes uint64 `json:"rx_bytes,omitempty" yaml:"rx_bytes,omitempty"` + RxErrors uint64 `json:"rx_errors,omitempty" yaml:"rx_errors,omitempty"` + TxPackets uint64 `json:"tx_packets,omitempty" yaml:"tx_packets,omitempty"` + TxDropped uint64 `json:"tx_dropped,omitempty" yaml:"tx_dropped,omitempty"` + RxPackets uint64 `json:"rx_packets,omitempty" yaml:"rx_packets,omitempty"` + TxErrors uint64 `json:"tx_errors,omitempty" yaml:"tx_errors,omitempty"` + TxBytes uint64 `json:"tx_bytes,omitempty" yaml:"tx_bytes,omitempty"` + } `json:"network,omitempty" yaml:"network,omitempty"` + MemoryStats struct { + Stats struct { + TotalPgmafault uint64 `json:"total_pgmafault,omitempty" yaml:"total_pgmafault,omitempty"` + Cache uint64 `json:"cache,omitempty" yaml:"cache,omitempty"` + MappedFile uint64 `json:"mapped_file,omitempty" yaml:"mapped_file,omitempty"` + TotalInactiveFile uint64 `json:"total_inactive_file,omitempty" yaml:"total_inactive_file,omitempty"` + Pgpgout uint64 `json:"pgpgout,omitempty" yaml:"pgpgout,omitempty"` + Rss uint64 `json:"rss,omitempty" yaml:"rss,omitempty"` + TotalMappedFile uint64 `json:"total_mapped_file,omitempty" yaml:"total_mapped_file,omitempty"` + Writeback uint64 `json:"writeback,omitempty" yaml:"writeback,omitempty"` + Unevictable uint64 `json:"unevictable,omitempty" yaml:"unevictable,omitempty"` + Pgpgin uint64 `json:"pgpgin,omitempty" yaml:"pgpgin,omitempty"` + TotalUnevictable uint64 `json:"total_unevictable,omitempty" yaml:"total_unevictable,omitempty"` + Pgmajfault uint64 `json:"pgmajfault,omitempty" yaml:"pgmajfault,omitempty"` + TotalRss uint64 `json:"total_rss,omitempty" yaml:"total_rss,omitempty"` + TotalRssHuge uint64 `json:"total_rss_huge,omitempty" yaml:"total_rss_huge,omitempty"` + TotalWriteback uint64 `json:"total_writeback,omitempty" yaml:"total_writeback,omitempty"` + TotalInactiveAnon uint64 `json:"total_inactive_anon,omitempty" yaml:"total_inactive_anon,omitempty"` + RssHuge uint64 `json:"rss_huge,omitempty" yaml:"rss_huge,omitempty"` + HierarchicalMemoryLimit uint64 `json:"hierarchical_memory_limit,omitempty" yaml:"hierarchical_memory_limit,omitempty"` + TotalPgfault uint64 `json:"total_pgfault,omitempty" yaml:"total_pgfault,omitempty"` + TotalActiveFile uint64 `json:"total_active_file,omitempty" yaml:"total_active_file,omitempty"` + ActiveAnon uint64 `json:"active_anon,omitempty" yaml:"active_anon,omitempty"` + TotalActiveAnon uint64 `json:"total_active_anon,omitempty" yaml:"total_active_anon,omitempty"` + TotalPgpgout uint64 `json:"total_pgpgout,omitempty" yaml:"total_pgpgout,omitempty"` + TotalCache uint64 `json:"total_cache,omitempty" yaml:"total_cache,omitempty"` + InactiveAnon uint64 `json:"inactive_anon,omitempty" yaml:"inactive_anon,omitempty"` + ActiveFile uint64 `json:"active_file,omitempty" yaml:"active_file,omitempty"` + Pgfault uint64 `json:"pgfault,omitempty" yaml:"pgfault,omitempty"` + InactiveFile uint64 `json:"inactive_file,omitempty" yaml:"inactive_file,omitempty"` + TotalPgpgin uint64 `json:"total_pgpgin,omitempty" yaml:"total_pgpgin,omitempty"` + } `json:"stats,omitempty" yaml:"stats,omitempty"` + MaxUsage uint64 `json:"max_usage,omitempty" yaml:"max_usage,omitempty"` + Usage uint64 `json:"usage,omitempty" yaml:"usage,omitempty"` + Failcnt uint64 `json:"failcnt,omitempty" yaml:"failcnt,omitempty"` + Limit uint64 `json:"limit,omitempty" yaml:"limit,omitempty"` + } `json:"memory_stats,omitempty" yaml:"memory_stats,omitempty"` + BlkioStats struct { + IOServiceBytesRecursive []BlkioStatsEntry `json:"io_service_bytes_recursive,omitempty" yaml:"io_service_bytes_recursive,omitempty"` + IOServicedRecursive []BlkioStatsEntry `json:"io_serviced_recursive,omitempty" yaml:"io_serviced_recursive,omitempty"` + IOQueueRecursive []BlkioStatsEntry `json:"io_queue_recursive,omitempty" yaml:"io_queue_recursive,omitempty"` + IOServiceTimeRecursive []BlkioStatsEntry `json:"io_service_time_recursive,omitempty" yaml:"io_service_time_recursive,omitempty"` + IOWaitTimeRecursive []BlkioStatsEntry `json:"io_wait_time_recursive,omitempty" yaml:"io_wait_time_recursive,omitempty"` + IOMergedRecursive []BlkioStatsEntry `json:"io_merged_recursive,omitempty" yaml:"io_merged_recursive,omitempty"` + IOTimeRecursive []BlkioStatsEntry `json:"io_time_recursive,omitempty" yaml:"io_time_recursive,omitempty"` + SectorsRecursive []BlkioStatsEntry `json:"sectors_recursive,omitempty" yaml:"sectors_recursive,omitempty"` + } `json:"blkio_stats,omitempty" yaml:"blkio_stats,omitempty"` + CPUStats CPUStats `json:"cpu_stats,omitempty" yaml:"cpu_stats,omitempty"` + PreCPUStats CPUStats `json:"precpu_stats,omitempty"` +} + +// CPUStats is a stats entry for cpu stats +type CPUStats struct { + CPUUsage struct { + PercpuUsage []uint64 `json:"percpu_usage,omitempty" yaml:"percpu_usage,omitempty"` + UsageInUsermode uint64 `json:"usage_in_usermode,omitempty" yaml:"usage_in_usermode,omitempty"` + TotalUsage uint64 `json:"total_usage,omitempty" yaml:"total_usage,omitempty"` + UsageInKernelmode uint64 `json:"usage_in_kernelmode,omitempty" yaml:"usage_in_kernelmode,omitempty"` + } `json:"cpu_usage,omitempty" yaml:"cpu_usage,omitempty"` + SystemCPUUsage uint64 `json:"system_cpu_usage,omitempty" yaml:"system_cpu_usage,omitempty"` + ThrottlingData struct { + Periods uint64 `json:"periods,omitempty"` + ThrottledPeriods uint64 `json:"throttled_periods,omitempty"` + ThrottledTime uint64 `json:"throttled_time,omitempty"` + } `json:"throttling_data,omitempty" yaml:"throttling_data,omitempty"` +} + +// BlkioStatsEntry is a stats entry for blkio_stats +type BlkioStatsEntry struct { + Major uint64 `json:"major,omitempty" yaml:"major,omitempty"` + Minor uint64 `json:"minor,omitempty" yaml:"minor,omitempty"` + Op string `json:"op,omitempty" yaml:"op,omitempty"` + Value uint64 `json:"value,omitempty" yaml:"value,omitempty"` +} + +// StatsOptions specify parameters to the Stats function. +// +// See http://goo.gl/DFMiYD for more details. +type StatsOptions struct { + ID string + Stats chan<- *Stats + Stream bool + // A flag that enables stopping the stats operation + Done <-chan bool + // Initial connection timeout + Timeout time.Duration +} + +// Stats sends container statistics for the given container to the given channel. +// +// This function is blocking, similar to a streaming call for logs, and should be run +// on a separate goroutine from the caller. Note that this function will block until +// the given container is removed, not just exited. When finished, this function +// will close the given channel. Alternatively, function can be stopped by signaling on the Done channel +// +// See http://goo.gl/DFMiYD for more details. +func (c *Client) Stats(opts StatsOptions) (retErr error) { + errC := make(chan error, 1) + readCloser, writeCloser := io.Pipe() + + defer func() { + close(opts.Stats) + + select { + case err := <-errC: + if err != nil && retErr == nil { + retErr = err + } + default: + // No errors + } + + if err := readCloser.Close(); err != nil && retErr == nil { + retErr = err + } + }() + + go func() { + err := c.stream("GET", fmt.Sprintf("/containers/%s/stats?stream=%v", opts.ID, opts.Stream), streamOptions{ + rawJSONStream: true, + useJSONDecoder: true, + stdout: writeCloser, + timeout: opts.Timeout, + }) + if err != nil { + dockerError, ok := err.(*Error) + if ok { + if dockerError.Status == http.StatusNotFound { + err = &NoSuchContainer{ID: opts.ID} + } + } + } + if closeErr := writeCloser.Close(); closeErr != nil && err == nil { + err = closeErr + } + errC <- err + close(errC) + }() + + quit := make(chan struct{}) + defer close(quit) + go func() { + // block here waiting for the signal to stop function + select { + case <-opts.Done: + readCloser.Close() + case <-quit: + return + } + }() + + decoder := json.NewDecoder(readCloser) + stats := new(Stats) + for err := decoder.Decode(&stats); err != io.EOF; err = decoder.Decode(stats) { + if err != nil { + return err + } + opts.Stats <- stats + stats = new(Stats) + } + return nil +} + +// KillContainerOptions represents the set of options that can be used in a +// call to KillContainer. +// +// See http://goo.gl/TFkECx for more details. +type KillContainerOptions struct { + // The ID of the container. + ID string `qs:"-"` + + // The signal to send to the container. When omitted, Docker server + // will assume SIGKILL. + Signal Signal +} + +// KillContainer kills a container, returning an error in case of failure. +// +// See http://goo.gl/TFkECx for more details. +func (c *Client) KillContainer(opts KillContainerOptions) error { + path := "/containers/" + opts.ID + "/kill" + "?" + queryString(opts) + _, status, err := c.do("POST", path, doOptions{}) + if status == http.StatusNotFound { + return &NoSuchContainer{ID: opts.ID} + } + if err != nil { + return err + } + return nil +} + +// RemoveContainerOptions encapsulates options to remove a container. +// +// See http://goo.gl/ZB83ji for more details. +type RemoveContainerOptions struct { + // The ID of the container. + ID string `qs:"-"` + + // A flag that indicates whether Docker should remove the volumes + // associated to the container. + RemoveVolumes bool `qs:"v"` + + // A flag that indicates whether Docker should remove the container + // even if it is currently running. + Force bool +} + +// RemoveContainer removes a container, returning an error in case of failure. +// +// See http://goo.gl/ZB83ji for more details. +func (c *Client) RemoveContainer(opts RemoveContainerOptions) error { + path := "/containers/" + opts.ID + "?" + queryString(opts) + _, status, err := c.do("DELETE", path, doOptions{}) + if status == http.StatusNotFound { + return &NoSuchContainer{ID: opts.ID} + } + if err != nil { + return err + } + return nil +} + +// CopyFromContainerOptions is the set of options that can be used when copying +// files or folders from a container. +// +// See http://goo.gl/rINMlw for more details. +type CopyFromContainerOptions struct { + OutputStream io.Writer `json:"-"` + Container string `json:"-"` + Resource string +} + +// CopyFromContainer copy files or folders from a container, using a given +// resource. +// +// See http://goo.gl/rINMlw for more details. +func (c *Client) CopyFromContainer(opts CopyFromContainerOptions) error { + if opts.Container == "" { + return &NoSuchContainer{ID: opts.Container} + } + url := fmt.Sprintf("/containers/%s/copy", opts.Container) + body, status, err := c.do("POST", url, doOptions{data: opts}) + if status == http.StatusNotFound { + return &NoSuchContainer{ID: opts.Container} + } + if err != nil { + return err + } + _, err = io.Copy(opts.OutputStream, bytes.NewBuffer(body)) + return err +} + +// WaitContainer blocks until the given container stops, return the exit code +// of the container status. +// +// See http://goo.gl/J88DHU for more details. +func (c *Client) WaitContainer(id string) (int, error) { + body, status, err := c.do("POST", "/containers/"+id+"/wait", doOptions{}) + if status == http.StatusNotFound { + return 0, &NoSuchContainer{ID: id} + } + if err != nil { + return 0, err + } + var r struct{ StatusCode int } + err = json.Unmarshal(body, &r) + if err != nil { + return 0, err + } + return r.StatusCode, nil +} + +// CommitContainerOptions aggregates parameters to the CommitContainer method. +// +// See http://goo.gl/Jn8pe8 for more details. +type CommitContainerOptions struct { + Container string + Repository string `qs:"repo"` + Tag string + Message string `qs:"m"` + Author string + Run *Config `qs:"-"` +} + +// CommitContainer creates a new image from a container's changes. +// +// See http://goo.gl/Jn8pe8 for more details. +func (c *Client) CommitContainer(opts CommitContainerOptions) (*Image, error) { + path := "/commit?" + queryString(opts) + body, status, err := c.do("POST", path, doOptions{data: opts.Run}) + if status == http.StatusNotFound { + return nil, &NoSuchContainer{ID: opts.Container} + } + if err != nil { + return nil, err + } + var image Image + err = json.Unmarshal(body, &image) + if err != nil { + return nil, err + } + return &image, nil +} + +// AttachToContainerOptions is the set of options that can be used when +// attaching to a container. +// +// See http://goo.gl/RRAhws for more details. +type AttachToContainerOptions struct { + Container string `qs:"-"` + InputStream io.Reader `qs:"-"` + OutputStream io.Writer `qs:"-"` + ErrorStream io.Writer `qs:"-"` + + // Get container logs, sending it to OutputStream. + Logs bool + + // Stream the response? + Stream bool + + // Attach to stdin, and use InputStream. + Stdin bool + + // Attach to stdout, and use OutputStream. + Stdout bool + + // Attach to stderr, and use ErrorStream. + Stderr bool + + // If set, after a successful connect, a sentinel will be sent and then the + // client will block on receive before continuing. + // + // It must be an unbuffered channel. Using a buffered channel can lead + // to unexpected behavior. + Success chan struct{} + + // Use raw terminal? Usually true when the container contains a TTY. + RawTerminal bool `qs:"-"` +} + +// AttachToContainer attaches to a container, using the given options. +// +// See http://goo.gl/RRAhws for more details. +func (c *Client) AttachToContainer(opts AttachToContainerOptions) error { + if opts.Container == "" { + return &NoSuchContainer{ID: opts.Container} + } + path := "/containers/" + opts.Container + "/attach?" + queryString(opts) + return c.hijack("POST", path, hijackOptions{ + success: opts.Success, + setRawTerminal: opts.RawTerminal, + in: opts.InputStream, + stdout: opts.OutputStream, + stderr: opts.ErrorStream, + }) +} + +// LogsOptions represents the set of options used when getting logs from a +// container. +// +// See http://goo.gl/rLhKSU for more details. +type LogsOptions struct { + Container string `qs:"-"` + OutputStream io.Writer `qs:"-"` + ErrorStream io.Writer `qs:"-"` + Follow bool + Stdout bool + Stderr bool + Since int64 + Timestamps bool + Tail string + + // Use raw terminal? Usually true when the container contains a TTY. + RawTerminal bool `qs:"-"` +} + +// Logs gets stdout and stderr logs from the specified container. +// +// See http://goo.gl/rLhKSU for more details. +func (c *Client) Logs(opts LogsOptions) error { + if opts.Container == "" { + return &NoSuchContainer{ID: opts.Container} + } + if opts.Tail == "" { + opts.Tail = "all" + } + path := "/containers/" + opts.Container + "/logs?" + queryString(opts) + return c.stream("GET", path, streamOptions{ + setRawTerminal: opts.RawTerminal, + stdout: opts.OutputStream, + stderr: opts.ErrorStream, + }) +} + +// ResizeContainerTTY resizes the terminal to the given height and width. +func (c *Client) ResizeContainerTTY(id string, height, width int) error { + params := make(url.Values) + params.Set("h", strconv.Itoa(height)) + params.Set("w", strconv.Itoa(width)) + _, _, err := c.do("POST", "/containers/"+id+"/resize?"+params.Encode(), doOptions{}) + return err +} + +// ExportContainerOptions is the set of parameters to the ExportContainer +// method. +// +// See http://goo.gl/hnzE62 for more details. +type ExportContainerOptions struct { + ID string + OutputStream io.Writer +} + +// ExportContainer export the contents of container id as tar archive +// and prints the exported contents to stdout. +// +// See http://goo.gl/hnzE62 for more details. +func (c *Client) ExportContainer(opts ExportContainerOptions) error { + if opts.ID == "" { + return &NoSuchContainer{ID: opts.ID} + } + url := fmt.Sprintf("/containers/%s/export", opts.ID) + return c.stream("GET", url, streamOptions{ + setRawTerminal: true, + stdout: opts.OutputStream, + }) +} + +// NoSuchContainer is the error returned when a given container does not exist. +type NoSuchContainer struct { + ID string + Err error +} + +func (err *NoSuchContainer) Error() string { + if err.Err != nil { + return err.Err.Error() + } + return "No such container: " + err.ID +} + +// ContainerAlreadyRunning is the error returned when a given container is +// already running. +type ContainerAlreadyRunning struct { + ID string +} + +func (err *ContainerAlreadyRunning) Error() string { + return "Container already running: " + err.ID +} + +// ContainerNotRunning is the error returned when a given container is not +// running. +type ContainerNotRunning struct { + ID string +} + +func (err *ContainerNotRunning) Error() string { + return "Container not running: " + err.ID +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container_test.go new file mode 100644 index 0000000000000..00966aa19cea9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/container_test.go @@ -0,0 +1,1941 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "bytes" + "encoding/json" + "errors" + "fmt" + "io/ioutil" + "net" + "net/http" + "net/http/httptest" + "net/url" + "os" + "reflect" + "regexp" + "runtime" + "strconv" + "strings" + "testing" + "time" +) + +func TestStateString(t *testing.T) { + started := time.Now().Add(-3 * time.Hour) + var tests = []struct { + input State + expected string + }{ + {State{Running: true, Paused: true}, "^paused$"}, + {State{Running: true, StartedAt: started}, "^Up 3h.*$"}, + {State{Running: false, ExitCode: 7}, "^Exit 7$"}, + } + for _, tt := range tests { + re := regexp.MustCompile(tt.expected) + if got := tt.input.String(); !re.MatchString(got) { + t.Errorf("State.String(): wrong result. Want %q. Got %q.", tt.expected, got) + } + } +} + +func TestListContainers(t *testing.T) { + jsonContainers := `[ + { + "Id": "8dfafdbc3a40", + "Image": "base:latest", + "Command": "echo 1", + "Created": 1367854155, + "Ports":[{"PrivatePort": 2222, "PublicPort": 3333, "Type": "tcp"}], + "Status": "Exit 0" + }, + { + "Id": "9cd87474be90", + "Image": "base:latest", + "Command": "echo 222222", + "Created": 1367854155, + "Ports":[{"PrivatePort": 2222, "PublicPort": 3333, "Type": "tcp"}], + "Status": "Exit 0" + }, + { + "Id": "3176a2479c92", + "Image": "base:latest", + "Command": "echo 3333333333333333", + "Created": 1367854154, + "Ports":[{"PrivatePort": 2221, "PublicPort": 3331, "Type": "tcp"}], + "Status": "Exit 0" + }, + { + "Id": "4cb07b47f9fb", + "Image": "base:latest", + "Command": "echo 444444444444444444444444444444444", + "Ports":[{"PrivatePort": 2223, "PublicPort": 3332, "Type": "tcp"}], + "Created": 1367854152, + "Status": "Exit 0" + } +]` + var expected []APIContainers + err := json.Unmarshal([]byte(jsonContainers), &expected) + if err != nil { + t.Fatal(err) + } + client := newTestClient(&FakeRoundTripper{message: jsonContainers, status: http.StatusOK}) + containers, err := client.ListContainers(ListContainersOptions{}) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(containers, expected) { + t.Errorf("ListContainers: Expected %#v. Got %#v.", expected, containers) + } +} + +func TestListContainersParams(t *testing.T) { + var tests = []struct { + input ListContainersOptions + params map[string][]string + }{ + {ListContainersOptions{}, map[string][]string{}}, + {ListContainersOptions{All: true}, map[string][]string{"all": {"1"}}}, + {ListContainersOptions{All: true, Limit: 10}, map[string][]string{"all": {"1"}, "limit": {"10"}}}, + { + ListContainersOptions{All: true, Limit: 10, Since: "adf9983", Before: "abdeef"}, + map[string][]string{"all": {"1"}, "limit": {"10"}, "since": {"adf9983"}, "before": {"abdeef"}}, + }, + { + ListContainersOptions{Filters: map[string][]string{"status": {"paused", "running"}}}, + map[string][]string{"filters": {"{\"status\":[\"paused\",\"running\"]}"}}, + }, + { + ListContainersOptions{All: true, Filters: map[string][]string{"exited": {"0"}, "status": {"exited"}}}, + map[string][]string{"all": {"1"}, "filters": {"{\"exited\":[\"0\"],\"status\":[\"exited\"]}"}}, + }, + } + fakeRT := &FakeRoundTripper{message: "[]", status: http.StatusOK} + client := newTestClient(fakeRT) + u, _ := url.Parse(client.getURL("/containers/json")) + for _, tt := range tests { + if _, err := client.ListContainers(tt.input); err != nil { + t.Error(err) + } + got := map[string][]string(fakeRT.requests[0].URL.Query()) + if !reflect.DeepEqual(got, tt.params) { + t.Errorf("Expected %#v, got %#v.", tt.params, got) + } + if path := fakeRT.requests[0].URL.Path; path != u.Path { + t.Errorf("Wrong path on request. Want %q. Got %q.", u.Path, path) + } + if meth := fakeRT.requests[0].Method; meth != "GET" { + t.Errorf("Wrong HTTP method. Want GET. Got %s.", meth) + } + fakeRT.Reset() + } +} + +func TestListContainersFailure(t *testing.T) { + var tests = []struct { + status int + message string + }{ + {400, "bad parameter"}, + {500, "internal server error"}, + } + for _, tt := range tests { + client := newTestClient(&FakeRoundTripper{message: tt.message, status: tt.status}) + expected := Error{Status: tt.status, Message: tt.message} + containers, err := client.ListContainers(ListContainersOptions{}) + if !reflect.DeepEqual(expected, *err.(*Error)) { + t.Errorf("Wrong error in ListContainers. Want %#v. Got %#v.", expected, err) + } + if len(containers) > 0 { + t.Errorf("ListContainers failure. Expected empty list. Got %#v.", containers) + } + } +} + +func TestInspectContainer(t *testing.T) { + jsonContainer := `{ + "Id": "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2", + "AppArmorProfile": "Profile", + "Created": "2013-05-07T14:51:42.087658+02:00", + "Path": "date", + "Args": [], + "Config": { + "Hostname": "4fa6e0f0c678", + "User": "", + "Memory": 17179869184, + "MemorySwap": 34359738368, + "AttachStdin": false, + "AttachStdout": true, + "AttachStderr": true, + "PortSpecs": null, + "Tty": false, + "OpenStdin": false, + "StdinOnce": false, + "Env": null, + "Cmd": [ + "date" + ], + "Image": "base", + "Volumes": {}, + "VolumesFrom": "", + "SecurityOpt": [ + "label:user:USER" + ], + "Ulimits": [ + { "Name": "nofile", "Soft": 1024, "Hard": 2048 } + ] + }, + "State": { + "Running": false, + "Pid": 0, + "ExitCode": 0, + "StartedAt": "2013-05-07T14:51:42.087658+02:00", + "Ghost": false + }, + "Node": { + "ID": "4I4E:QR4I:Z733:QEZK:5X44:Q4T7:W2DD:JRDY:KB2O:PODO:Z5SR:XRB6", + "IP": "192.168.99.105", + "Addra": "192.168.99.105:2376", + "Name": "node-01", + "Cpus": 4, + "Memory": 1048436736, + "Labels": { + "executiondriver": "native-0.2", + "kernelversion": "3.18.5-tinycore64", + "operatingsystem": "Boot2Docker 1.5.0 (TCL 5.4); master : a66bce5 - Tue Feb 10 23:31:27 UTC 2015", + "provider": "virtualbox", + "storagedriver": "aufs" + } + }, + "Image": "b750fe79269d2ec9a3c593ef05b4332b1d1a02a62b4accb2c21d589ff2f5f2dc", + "NetworkSettings": { + "IpAddress": "", + "IpPrefixLen": 0, + "Gateway": "", + "Bridge": "", + "PortMapping": null + }, + "SysInitPath": "/home/kitty/go/src/github.com/dotcloud/docker/bin/docker", + "ResolvConfPath": "/etc/resolv.conf", + "Volumes": {}, + "HostConfig": { + "Binds": null, + "ContainerIDFile": "", + "LxcConf": [], + "Privileged": false, + "PortBindings": { + "80/tcp": [ + { + "HostIp": "0.0.0.0", + "HostPort": "49153" + } + ] + }, + "Links": null, + "PublishAllPorts": false, + "CgroupParent": "/mesos", + "Memory": 17179869184, + "MemorySwap": 34359738368 + } +}` + var expected Container + err := json.Unmarshal([]byte(jsonContainer), &expected) + if err != nil { + t.Fatal(err) + } + fakeRT := &FakeRoundTripper{message: jsonContainer, status: http.StatusOK} + client := newTestClient(fakeRT) + id := "4fa6e0f0c678" + container, err := client.InspectContainer(id) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(*container, expected) { + t.Errorf("InspectContainer(%q): Expected %#v. Got %#v.", id, expected, container) + } + expectedURL, _ := url.Parse(client.getURL("/containers/4fa6e0f0c678/json")) + if gotPath := fakeRT.requests[0].URL.Path; gotPath != expectedURL.Path { + t.Errorf("InspectContainer(%q): Wrong path in request. Want %q. Got %q.", id, expectedURL.Path, gotPath) + } +} + +func TestInspectContainerNegativeSwap(t *testing.T) { + jsonContainer := `{ + "Id": "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2", + "Created": "2013-05-07T14:51:42.087658+02:00", + "Path": "date", + "Args": [], + "Config": { + "Hostname": "4fa6e0f0c678", + "User": "", + "Memory": 17179869184, + "MemorySwap": -1, + "AttachStdin": false, + "AttachStdout": true, + "AttachStderr": true, + "PortSpecs": null, + "Tty": false, + "OpenStdin": false, + "StdinOnce": false, + "Env": null, + "Cmd": [ + "date" + ], + "Image": "base", + "Volumes": {}, + "VolumesFrom": "" + }, + "State": { + "Running": false, + "Pid": 0, + "ExitCode": 0, + "StartedAt": "2013-05-07T14:51:42.087658+02:00", + "Ghost": false + }, + "Image": "b750fe79269d2ec9a3c593ef05b4332b1d1a02a62b4accb2c21d589ff2f5f2dc", + "NetworkSettings": { + "IpAddress": "", + "IpPrefixLen": 0, + "Gateway": "", + "Bridge": "", + "PortMapping": null + }, + "SysInitPath": "/home/kitty/go/src/github.com/dotcloud/docker/bin/docker", + "ResolvConfPath": "/etc/resolv.conf", + "Volumes": {}, + "HostConfig": { + "Binds": null, + "ContainerIDFile": "", + "LxcConf": [], + "Privileged": false, + "PortBindings": { + "80/tcp": [ + { + "HostIp": "0.0.0.0", + "HostPort": "49153" + } + ] + }, + "Links": null, + "PublishAllPorts": false + } +}` + var expected Container + err := json.Unmarshal([]byte(jsonContainer), &expected) + if err != nil { + t.Fatal(err) + } + fakeRT := &FakeRoundTripper{message: jsonContainer, status: http.StatusOK} + client := newTestClient(fakeRT) + id := "4fa6e0f0c678" + container, err := client.InspectContainer(id) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(*container, expected) { + t.Errorf("InspectContainer(%q): Expected %#v. Got %#v.", id, expected, container) + } + expectedURL, _ := url.Parse(client.getURL("/containers/4fa6e0f0c678/json")) + if gotPath := fakeRT.requests[0].URL.Path; gotPath != expectedURL.Path { + t.Errorf("InspectContainer(%q): Wrong path in request. Want %q. Got %q.", id, expectedURL.Path, gotPath) + } +} + +func TestInspectContainerFailure(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "server error", status: 500}) + expected := Error{Status: 500, Message: "server error"} + container, err := client.InspectContainer("abe033") + if container != nil { + t.Errorf("InspectContainer: Expected container, got %#v", container) + } + if !reflect.DeepEqual(expected, *err.(*Error)) { + t.Errorf("InspectContainer: Wrong error information. Want %#v. Got %#v.", expected, err) + } +} + +func TestInspectContainerNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such container", status: 404}) + container, err := client.InspectContainer("abe033") + if container != nil { + t.Errorf("InspectContainer: Expected container, got %#v", container) + } + expected := &NoSuchContainer{ID: "abe033"} + if !reflect.DeepEqual(err, expected) { + t.Errorf("InspectContainer: Wrong error information. Want %#v. Got %#v.", expected, err) + } +} + +func TestContainerChanges(t *testing.T) { + jsonChanges := `[ + { + "Path":"/dev", + "Kind":0 + }, + { + "Path":"/dev/kmsg", + "Kind":1 + }, + { + "Path":"/test", + "Kind":1 + } +]` + var expected []Change + err := json.Unmarshal([]byte(jsonChanges), &expected) + if err != nil { + t.Fatal(err) + } + fakeRT := &FakeRoundTripper{message: jsonChanges, status: http.StatusOK} + client := newTestClient(fakeRT) + id := "4fa6e0f0c678" + changes, err := client.ContainerChanges(id) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(changes, expected) { + t.Errorf("ContainerChanges(%q): Expected %#v. Got %#v.", id, expected, changes) + } + expectedURL, _ := url.Parse(client.getURL("/containers/4fa6e0f0c678/changes")) + if gotPath := fakeRT.requests[0].URL.Path; gotPath != expectedURL.Path { + t.Errorf("ContainerChanges(%q): Wrong path in request. Want %q. Got %q.", id, expectedURL.Path, gotPath) + } +} + +func TestContainerChangesFailure(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "server error", status: 500}) + expected := Error{Status: 500, Message: "server error"} + changes, err := client.ContainerChanges("abe033") + if changes != nil { + t.Errorf("ContainerChanges: Expected changes, got %#v", changes) + } + if !reflect.DeepEqual(expected, *err.(*Error)) { + t.Errorf("ContainerChanges: Wrong error information. Want %#v. Got %#v.", expected, err) + } +} + +func TestContainerChangesNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such container", status: 404}) + changes, err := client.ContainerChanges("abe033") + if changes != nil { + t.Errorf("ContainerChanges: Expected changes, got %#v", changes) + } + expected := &NoSuchContainer{ID: "abe033"} + if !reflect.DeepEqual(err, expected) { + t.Errorf("ContainerChanges: Wrong error information. Want %#v. Got %#v.", expected, err) + } +} + +func TestCreateContainer(t *testing.T) { + jsonContainer := `{ + "Id": "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2", + "Warnings": [] +}` + var expected Container + err := json.Unmarshal([]byte(jsonContainer), &expected) + if err != nil { + t.Fatal(err) + } + fakeRT := &FakeRoundTripper{message: jsonContainer, status: http.StatusOK} + client := newTestClient(fakeRT) + config := Config{AttachStdout: true, AttachStdin: true} + opts := CreateContainerOptions{Name: "TestCreateContainer", Config: &config} + container, err := client.CreateContainer(opts) + if err != nil { + t.Fatal(err) + } + id := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + if container.ID != id { + t.Errorf("CreateContainer: wrong ID. Want %q. Got %q.", id, container.ID) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("CreateContainer: wrong HTTP method. Want %q. Got %q.", "POST", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/containers/create")) + if gotPath := req.URL.Path; gotPath != expectedURL.Path { + t.Errorf("CreateContainer: Wrong path in request. Want %q. Got %q.", expectedURL.Path, gotPath) + } + var gotBody Config + err = json.NewDecoder(req.Body).Decode(&gotBody) + if err != nil { + t.Fatal(err) + } +} + +func TestCreateContainerImageNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "No such image", status: http.StatusNotFound}) + config := Config{AttachStdout: true, AttachStdin: true} + container, err := client.CreateContainer(CreateContainerOptions{Config: &config}) + if container != nil { + t.Errorf("CreateContainer: expected container, got %#v.", container) + } + if !reflect.DeepEqual(err, ErrNoSuchImage) { + t.Errorf("CreateContainer: Wrong error type. Want %#v. Got %#v.", ErrNoSuchImage, err) + } +} + +func TestCreateContainerDuplicateName(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "No such image", status: http.StatusConflict}) + config := Config{AttachStdout: true, AttachStdin: true} + container, err := client.CreateContainer(CreateContainerOptions{Config: &config}) + if container != nil { + t.Errorf("CreateContainer: expected container, got %#v.", container) + } + if err != ErrContainerAlreadyExists { + t.Errorf("CreateContainer: Wrong error type. Want %#v. Got %#v.", ErrContainerAlreadyExists, err) + } +} + +func TestCreateContainerWithHostConfig(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "{}", status: http.StatusOK} + client := newTestClient(fakeRT) + config := Config{} + hostConfig := HostConfig{PublishAllPorts: true} + opts := CreateContainerOptions{Name: "TestCreateContainerWithHostConfig", Config: &config, HostConfig: &hostConfig} + _, err := client.CreateContainer(opts) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + var gotBody map[string]interface{} + err = json.NewDecoder(req.Body).Decode(&gotBody) + if err != nil { + t.Fatal(err) + } + if _, ok := gotBody["HostConfig"]; !ok { + t.Errorf("CreateContainer: wrong body. HostConfig was not serialized") + } +} + +func TestStartContainer(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + id := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + err := client.StartContainer(id, &HostConfig{}) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("StartContainer(%q): wrong HTTP method. Want %q. Got %q.", id, "POST", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/containers/" + id + "/start")) + if gotPath := req.URL.Path; gotPath != expectedURL.Path { + t.Errorf("StartContainer(%q): Wrong path in request. Want %q. Got %q.", id, expectedURL.Path, gotPath) + } + expectedContentType := "application/json" + if contentType := req.Header.Get("Content-Type"); contentType != expectedContentType { + t.Errorf("StartContainer(%q): Wrong content-type in request. Want %q. Got %q.", id, expectedContentType, contentType) + } +} + +func TestStartContainerNilHostConfig(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + id := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + err := client.StartContainer(id, nil) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("StartContainer(%q): wrong HTTP method. Want %q. Got %q.", id, "POST", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/containers/" + id + "/start")) + if gotPath := req.URL.Path; gotPath != expectedURL.Path { + t.Errorf("StartContainer(%q): Wrong path in request. Want %q. Got %q.", id, expectedURL.Path, gotPath) + } + expectedContentType := "application/json" + if contentType := req.Header.Get("Content-Type"); contentType != expectedContentType { + t.Errorf("StartContainer(%q): Wrong content-type in request. Want %q. Got %q.", id, expectedContentType, contentType) + } + var buf [4]byte + req.Body.Read(buf[:]) + if string(buf[:]) != "null" { + t.Errorf("Startcontainer(%q): Wrong body. Want null. Got %s", id, buf[:]) + } +} + +func TestStartContainerNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such container", status: http.StatusNotFound}) + err := client.StartContainer("a2344", &HostConfig{}) + expected := &NoSuchContainer{ID: "a2344", Err: err.(*NoSuchContainer).Err} + if !reflect.DeepEqual(err, expected) { + t.Errorf("StartContainer: Wrong error returned. Want %#v. Got %#v.", expected, err) + } +} + +func TestStartContainerAlreadyRunning(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "container already running", status: http.StatusNotModified}) + err := client.StartContainer("a2334", &HostConfig{}) + expected := &ContainerAlreadyRunning{ID: "a2334"} + if !reflect.DeepEqual(err, expected) { + t.Errorf("StartContainer: Wrong error returned. Want %#v. Got %#v.", expected, err) + } +} + +func TestStopContainer(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusNoContent} + client := newTestClient(fakeRT) + id := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + err := client.StopContainer(id, 10) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("StopContainer(%q, 10): wrong HTTP method. Want %q. Got %q.", id, "POST", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/containers/" + id + "/stop")) + if gotPath := req.URL.Path; gotPath != expectedURL.Path { + t.Errorf("StopContainer(%q, 10): Wrong path in request. Want %q. Got %q.", id, expectedURL.Path, gotPath) + } +} + +func TestStopContainerNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such container", status: http.StatusNotFound}) + err := client.StopContainer("a2334", 10) + expected := &NoSuchContainer{ID: "a2334"} + if !reflect.DeepEqual(err, expected) { + t.Errorf("StopContainer: Wrong error returned. Want %#v. Got %#v.", expected, err) + } +} + +func TestStopContainerNotRunning(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "container not running", status: http.StatusNotModified}) + err := client.StopContainer("a2334", 10) + expected := &ContainerNotRunning{ID: "a2334"} + if !reflect.DeepEqual(err, expected) { + t.Errorf("StopContainer: Wrong error returned. Want %#v. Got %#v.", expected, err) + } +} + +func TestRestartContainer(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusNoContent} + client := newTestClient(fakeRT) + id := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + err := client.RestartContainer(id, 10) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("RestartContainer(%q, 10): wrong HTTP method. Want %q. Got %q.", id, "POST", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/containers/" + id + "/restart")) + if gotPath := req.URL.Path; gotPath != expectedURL.Path { + t.Errorf("RestartContainer(%q, 10): Wrong path in request. Want %q. Got %q.", id, expectedURL.Path, gotPath) + } +} + +func TestRestartContainerNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such container", status: http.StatusNotFound}) + err := client.RestartContainer("a2334", 10) + expected := &NoSuchContainer{ID: "a2334"} + if !reflect.DeepEqual(err, expected) { + t.Errorf("RestartContainer: Wrong error returned. Want %#v. Got %#v.", expected, err) + } +} + +func TestPauseContainer(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusNoContent} + client := newTestClient(fakeRT) + id := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + err := client.PauseContainer(id) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("PauseContainer(%q): wrong HTTP method. Want %q. Got %q.", id, "POST", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/containers/" + id + "/pause")) + if gotPath := req.URL.Path; gotPath != expectedURL.Path { + t.Errorf("PauseContainer(%q): Wrong path in request. Want %q. Got %q.", id, expectedURL.Path, gotPath) + } +} + +func TestPauseContainerNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such container", status: http.StatusNotFound}) + err := client.PauseContainer("a2334") + expected := &NoSuchContainer{ID: "a2334"} + if !reflect.DeepEqual(err, expected) { + t.Errorf("PauseContainer: Wrong error returned. Want %#v. Got %#v.", expected, err) + } +} + +func TestUnpauseContainer(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusNoContent} + client := newTestClient(fakeRT) + id := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + err := client.UnpauseContainer(id) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("PauseContainer(%q): wrong HTTP method. Want %q. Got %q.", id, "POST", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/containers/" + id + "/unpause")) + if gotPath := req.URL.Path; gotPath != expectedURL.Path { + t.Errorf("PauseContainer(%q): Wrong path in request. Want %q. Got %q.", id, expectedURL.Path, gotPath) + } +} + +func TestUnpauseContainerNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such container", status: http.StatusNotFound}) + err := client.UnpauseContainer("a2334") + expected := &NoSuchContainer{ID: "a2334"} + if !reflect.DeepEqual(err, expected) { + t.Errorf("PauseContainer: Wrong error returned. Want %#v. Got %#v.", expected, err) + } +} + +func TestKillContainer(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusNoContent} + client := newTestClient(fakeRT) + id := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + err := client.KillContainer(KillContainerOptions{ID: id}) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("KillContainer(%q): wrong HTTP method. Want %q. Got %q.", id, "POST", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/containers/" + id + "/kill")) + if gotPath := req.URL.Path; gotPath != expectedURL.Path { + t.Errorf("KillContainer(%q): Wrong path in request. Want %q. Got %q.", id, expectedURL.Path, gotPath) + } +} + +func TestKillContainerSignal(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusNoContent} + client := newTestClient(fakeRT) + id := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + err := client.KillContainer(KillContainerOptions{ID: id, Signal: SIGTERM}) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("KillContainer(%q): wrong HTTP method. Want %q. Got %q.", id, "POST", req.Method) + } + if signal := req.URL.Query().Get("signal"); signal != "15" { + t.Errorf("KillContainer(%q): Wrong query string in request. Want %q. Got %q.", id, "15", signal) + } +} + +func TestKillContainerNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such container", status: http.StatusNotFound}) + err := client.KillContainer(KillContainerOptions{ID: "a2334"}) + expected := &NoSuchContainer{ID: "a2334"} + if !reflect.DeepEqual(err, expected) { + t.Errorf("KillContainer: Wrong error returned. Want %#v. Got %#v.", expected, err) + } +} + +func TestRemoveContainer(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + id := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + opts := RemoveContainerOptions{ID: id} + err := client.RemoveContainer(opts) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + if req.Method != "DELETE" { + t.Errorf("RemoveContainer(%q): wrong HTTP method. Want %q. Got %q.", id, "DELETE", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/containers/" + id)) + if gotPath := req.URL.Path; gotPath != expectedURL.Path { + t.Errorf("RemoveContainer(%q): Wrong path in request. Want %q. Got %q.", id, expectedURL.Path, gotPath) + } +} + +func TestRemoveContainerRemoveVolumes(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + id := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + opts := RemoveContainerOptions{ID: id, RemoveVolumes: true} + err := client.RemoveContainer(opts) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + params := map[string][]string(req.URL.Query()) + expected := map[string][]string{"v": {"1"}} + if !reflect.DeepEqual(params, expected) { + t.Errorf("RemoveContainer(%q): wrong parameters. Want %#v. Got %#v.", id, expected, params) + } +} + +func TestRemoveContainerNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such container", status: http.StatusNotFound}) + err := client.RemoveContainer(RemoveContainerOptions{ID: "a2334"}) + expected := &NoSuchContainer{ID: "a2334"} + if !reflect.DeepEqual(err, expected) { + t.Errorf("RemoveContainer: Wrong error returned. Want %#v. Got %#v.", expected, err) + } +} + +func TestResizeContainerTTY(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + id := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + err := client.ResizeContainerTTY(id, 40, 80) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("ResizeContainerTTY(%q): wrong HTTP method. Want %q. Got %q.", id, "POST", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/containers/" + id + "/resize")) + if gotPath := req.URL.Path; gotPath != expectedURL.Path { + t.Errorf("ResizeContainerTTY(%q): Wrong path in request. Want %q. Got %q.", id, expectedURL.Path, gotPath) + } + got := map[string][]string(req.URL.Query()) + expectedParams := map[string][]string{ + "w": {"80"}, + "h": {"40"}, + } + if !reflect.DeepEqual(got, expectedParams) { + t.Errorf("Expected %#v, got %#v.", expectedParams, got) + } +} + +func TestWaitContainer(t *testing.T) { + fakeRT := &FakeRoundTripper{message: `{"StatusCode": 56}`, status: http.StatusOK} + client := newTestClient(fakeRT) + id := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + status, err := client.WaitContainer(id) + if err != nil { + t.Fatal(err) + } + if status != 56 { + t.Errorf("WaitContainer(%q): wrong return. Want 56. Got %d.", id, status) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("WaitContainer(%q): wrong HTTP method. Want %q. Got %q.", id, "POST", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/containers/" + id + "/wait")) + if gotPath := req.URL.Path; gotPath != expectedURL.Path { + t.Errorf("WaitContainer(%q): Wrong path in request. Want %q. Got %q.", id, expectedURL.Path, gotPath) + } +} + +func TestWaitContainerNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such container", status: http.StatusNotFound}) + _, err := client.WaitContainer("a2334") + expected := &NoSuchContainer{ID: "a2334"} + if !reflect.DeepEqual(err, expected) { + t.Errorf("WaitContainer: Wrong error returned. Want %#v. Got %#v.", expected, err) + } +} + +func TestCommitContainer(t *testing.T) { + response := `{"Id":"596069db4bf5"}` + client := newTestClient(&FakeRoundTripper{message: response, status: http.StatusOK}) + id := "596069db4bf5" + image, err := client.CommitContainer(CommitContainerOptions{}) + if err != nil { + t.Fatal(err) + } + if image.ID != id { + t.Errorf("CommitContainer: Wrong image id. Want %q. Got %q.", id, image.ID) + } +} + +func TestCommitContainerParams(t *testing.T) { + cfg := Config{Memory: 67108864} + json, _ := json.Marshal(&cfg) + var tests = []struct { + input CommitContainerOptions + params map[string][]string + body []byte + }{ + {CommitContainerOptions{}, map[string][]string{}, nil}, + {CommitContainerOptions{Container: "44c004db4b17"}, map[string][]string{"container": {"44c004db4b17"}}, nil}, + { + CommitContainerOptions{Container: "44c004db4b17", Repository: "tsuru/python", Message: "something"}, + map[string][]string{"container": {"44c004db4b17"}, "repo": {"tsuru/python"}, "m": {"something"}}, + nil, + }, + { + CommitContainerOptions{Container: "44c004db4b17", Run: &cfg}, + map[string][]string{"container": {"44c004db4b17"}}, + json, + }, + } + fakeRT := &FakeRoundTripper{message: "{}", status: http.StatusOK} + client := newTestClient(fakeRT) + u, _ := url.Parse(client.getURL("/commit")) + for _, tt := range tests { + if _, err := client.CommitContainer(tt.input); err != nil { + t.Error(err) + } + got := map[string][]string(fakeRT.requests[0].URL.Query()) + if !reflect.DeepEqual(got, tt.params) { + t.Errorf("Expected %#v, got %#v.", tt.params, got) + } + if path := fakeRT.requests[0].URL.Path; path != u.Path { + t.Errorf("Wrong path on request. Want %q. Got %q.", u.Path, path) + } + if meth := fakeRT.requests[0].Method; meth != "POST" { + t.Errorf("Wrong HTTP method. Want POST. Got %s.", meth) + } + if tt.body != nil { + if requestBody, err := ioutil.ReadAll(fakeRT.requests[0].Body); err == nil { + if bytes.Compare(requestBody, tt.body) != 0 { + t.Errorf("Expected body %#v, got %#v", tt.body, requestBody) + } + } else { + t.Errorf("Error reading request body: %#v", err) + } + } + fakeRT.Reset() + } +} + +func TestCommitContainerFailure(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such container", status: http.StatusInternalServerError}) + _, err := client.CommitContainer(CommitContainerOptions{}) + if err == nil { + t.Error("Expected non-nil error, got .") + } +} + +func TestCommitContainerNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such container", status: http.StatusNotFound}) + _, err := client.CommitContainer(CommitContainerOptions{}) + expected := &NoSuchContainer{ID: ""} + if !reflect.DeepEqual(err, expected) { + t.Errorf("CommitContainer: Wrong error returned. Want %#v. Got %#v.", expected, err) + } +} + +func TestAttachToContainerLogs(t *testing.T) { + var req http.Request + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + w.Write([]byte{1, 0, 0, 0, 0, 0, 0, 19}) + w.Write([]byte("something happened!")) + req = *r + })) + defer server.Close() + client, _ := NewClient(server.URL) + client.SkipServerVersionCheck = true + var buf bytes.Buffer + opts := AttachToContainerOptions{ + Container: "a123456", + OutputStream: &buf, + Stdout: true, + Stderr: true, + Logs: true, + } + err := client.AttachToContainer(opts) + if err != nil { + t.Fatal(err) + } + expected := "something happened!" + if buf.String() != expected { + t.Errorf("AttachToContainer for logs: wrong output. Want %q. Got %q.", expected, buf.String()) + } + if req.Method != "POST" { + t.Errorf("AttachToContainer: wrong HTTP method. Want POST. Got %s.", req.Method) + } + u, _ := url.Parse(client.getURL("/containers/a123456/attach")) + if req.URL.Path != u.Path { + t.Errorf("AttachToContainer for logs: wrong HTTP path. Want %q. Got %q.", u.Path, req.URL.Path) + } + expectedQs := map[string][]string{ + "logs": {"1"}, + "stdout": {"1"}, + "stderr": {"1"}, + } + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expectedQs) { + t.Errorf("AttachToContainer: wrong query string. Want %#v. Got %#v.", expectedQs, got) + } +} + +func TestAttachToContainer(t *testing.T) { + var reader = strings.NewReader("send value") + var req http.Request + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + w.Write([]byte{1, 0, 0, 0, 0, 0, 0, 5}) + w.Write([]byte("hello")) + req = *r + })) + defer server.Close() + client, _ := NewClient(server.URL) + client.SkipServerVersionCheck = true + var stdout, stderr bytes.Buffer + opts := AttachToContainerOptions{ + Container: "a123456", + OutputStream: &stdout, + ErrorStream: &stderr, + InputStream: reader, + Stdin: true, + Stdout: true, + Stderr: true, + Stream: true, + RawTerminal: true, + } + err := client.AttachToContainer(opts) + if err != nil { + t.Fatal(err) + } + expected := map[string][]string{ + "stdin": {"1"}, + "stdout": {"1"}, + "stderr": {"1"}, + "stream": {"1"}, + } + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expected) { + t.Errorf("AttachToContainer: wrong query string. Want %#v. Got %#v.", expected, got) + } +} + +func TestAttachToContainerSentinel(t *testing.T) { + var reader = strings.NewReader("send value") + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + w.Write([]byte{1, 0, 0, 0, 0, 0, 0, 5}) + w.Write([]byte("hello")) + })) + defer server.Close() + client, _ := NewClient(server.URL) + client.SkipServerVersionCheck = true + var stdout, stderr bytes.Buffer + success := make(chan struct{}) + opts := AttachToContainerOptions{ + Container: "a123456", + OutputStream: &stdout, + ErrorStream: &stderr, + InputStream: reader, + Stdin: true, + Stdout: true, + Stderr: true, + Stream: true, + RawTerminal: true, + Success: success, + } + go func() { + if err := client.AttachToContainer(opts); err != nil { + t.Error(err) + } + }() + success <- <-success +} + +func TestAttachToContainerNilStdout(t *testing.T) { + var reader = strings.NewReader("send value") + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + w.Write([]byte{1, 0, 0, 0, 0, 0, 0, 5}) + w.Write([]byte("hello")) + })) + defer server.Close() + client, _ := NewClient(server.URL) + client.SkipServerVersionCheck = true + var stderr bytes.Buffer + opts := AttachToContainerOptions{ + Container: "a123456", + OutputStream: nil, + ErrorStream: &stderr, + InputStream: reader, + Stdin: true, + Stdout: true, + Stderr: true, + Stream: true, + RawTerminal: true, + } + err := client.AttachToContainer(opts) + if err != nil { + t.Fatal(err) + } +} + +func TestAttachToContainerNilStderr(t *testing.T) { + var reader = strings.NewReader("send value") + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + w.Write([]byte{1, 0, 0, 0, 0, 0, 0, 5}) + w.Write([]byte("hello")) + })) + defer server.Close() + client, _ := NewClient(server.URL) + client.SkipServerVersionCheck = true + var stdout bytes.Buffer + opts := AttachToContainerOptions{ + Container: "a123456", + OutputStream: &stdout, + InputStream: reader, + Stdin: true, + Stdout: true, + Stderr: true, + Stream: true, + RawTerminal: true, + } + err := client.AttachToContainer(opts) + if err != nil { + t.Fatal(err) + } +} + +func TestAttachToContainerRawTerminalFalse(t *testing.T) { + input := strings.NewReader("send value") + var req http.Request + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + req = *r + w.WriteHeader(http.StatusOK) + hj, ok := w.(http.Hijacker) + if !ok { + t.Fatal("cannot hijack server connection") + } + conn, _, err := hj.Hijack() + if err != nil { + t.Fatal(err) + } + conn.Write([]byte{1, 0, 0, 0, 0, 0, 0, 5}) + conn.Write([]byte("hello")) + conn.Write([]byte{2, 0, 0, 0, 0, 0, 0, 6}) + conn.Write([]byte("hello!")) + conn.Close() + })) + defer server.Close() + client, _ := NewClient(server.URL) + client.SkipServerVersionCheck = true + var stdout, stderr bytes.Buffer + opts := AttachToContainerOptions{ + Container: "a123456", + OutputStream: &stdout, + ErrorStream: &stderr, + InputStream: input, + Stdin: true, + Stdout: true, + Stderr: true, + Stream: true, + RawTerminal: false, + } + err := client.AttachToContainer(opts) + if err != nil { + t.Fatal(err) + } + expected := map[string][]string{ + "stdin": {"1"}, + "stdout": {"1"}, + "stderr": {"1"}, + "stream": {"1"}, + } + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expected) { + t.Errorf("AttachToContainer: wrong query string. Want %#v. Got %#v.", expected, got) + } + if stdout.String() != "hello" { + t.Errorf("AttachToContainer: wrong content written to stdout. Want %q. Got %q.", "hello", stdout.String()) + } + if stderr.String() != "hello!" { + t.Errorf("AttachToContainer: wrong content written to stderr. Want %q. Got %q.", "hello!", stderr.String()) + } +} + +func TestAttachToContainerWithoutContainer(t *testing.T) { + var client Client + err := client.AttachToContainer(AttachToContainerOptions{}) + expected := &NoSuchContainer{ID: ""} + if !reflect.DeepEqual(err, expected) { + t.Errorf("AttachToContainer: wrong error. Want %#v. Got %#v.", expected, err) + } +} + +func TestLogs(t *testing.T) { + var req http.Request + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + prefix := []byte{1, 0, 0, 0, 0, 0, 0, 19} + w.Write(prefix) + w.Write([]byte("something happened!")) + req = *r + })) + defer server.Close() + client, _ := NewClient(server.URL) + client.SkipServerVersionCheck = true + var buf bytes.Buffer + opts := LogsOptions{ + Container: "a123456", + OutputStream: &buf, + Follow: true, + Stdout: true, + Stderr: true, + Timestamps: true, + } + err := client.Logs(opts) + if err != nil { + t.Fatal(err) + } + expected := "something happened!" + if buf.String() != expected { + t.Errorf("Logs: wrong output. Want %q. Got %q.", expected, buf.String()) + } + if req.Method != "GET" { + t.Errorf("Logs: wrong HTTP method. Want GET. Got %s.", req.Method) + } + u, _ := url.Parse(client.getURL("/containers/a123456/logs")) + if req.URL.Path != u.Path { + t.Errorf("AttachToContainer for logs: wrong HTTP path. Want %q. Got %q.", u.Path, req.URL.Path) + } + expectedQs := map[string][]string{ + "follow": {"1"}, + "stdout": {"1"}, + "stderr": {"1"}, + "timestamps": {"1"}, + "tail": {"all"}, + } + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expectedQs) { + t.Errorf("Logs: wrong query string. Want %#v. Got %#v.", expectedQs, got) + } +} + +func TestLogsNilStdoutDoesntFail(t *testing.T) { + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + prefix := []byte{1, 0, 0, 0, 0, 0, 0, 19} + w.Write(prefix) + w.Write([]byte("something happened!")) + })) + defer server.Close() + client, _ := NewClient(server.URL) + client.SkipServerVersionCheck = true + opts := LogsOptions{ + Container: "a123456", + Follow: true, + Stdout: true, + Stderr: true, + Timestamps: true, + } + err := client.Logs(opts) + if err != nil { + t.Fatal(err) + } +} + +func TestLogsNilStderrDoesntFail(t *testing.T) { + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + prefix := []byte{2, 0, 0, 0, 0, 0, 0, 19} + w.Write(prefix) + w.Write([]byte("something happened!")) + })) + defer server.Close() + client, _ := NewClient(server.URL) + client.SkipServerVersionCheck = true + opts := LogsOptions{ + Container: "a123456", + Follow: true, + Stdout: true, + Stderr: true, + Timestamps: true, + } + err := client.Logs(opts) + if err != nil { + t.Fatal(err) + } +} + +func TestLogsSpecifyingTail(t *testing.T) { + var req http.Request + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + prefix := []byte{1, 0, 0, 0, 0, 0, 0, 19} + w.Write(prefix) + w.Write([]byte("something happened!")) + req = *r + })) + defer server.Close() + client, _ := NewClient(server.URL) + client.SkipServerVersionCheck = true + var buf bytes.Buffer + opts := LogsOptions{ + Container: "a123456", + OutputStream: &buf, + Follow: true, + Stdout: true, + Stderr: true, + Timestamps: true, + Tail: "100", + } + err := client.Logs(opts) + if err != nil { + t.Fatal(err) + } + expected := "something happened!" + if buf.String() != expected { + t.Errorf("Logs: wrong output. Want %q. Got %q.", expected, buf.String()) + } + if req.Method != "GET" { + t.Errorf("Logs: wrong HTTP method. Want GET. Got %s.", req.Method) + } + u, _ := url.Parse(client.getURL("/containers/a123456/logs")) + if req.URL.Path != u.Path { + t.Errorf("AttachToContainer for logs: wrong HTTP path. Want %q. Got %q.", u.Path, req.URL.Path) + } + expectedQs := map[string][]string{ + "follow": {"1"}, + "stdout": {"1"}, + "stderr": {"1"}, + "timestamps": {"1"}, + "tail": {"100"}, + } + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expectedQs) { + t.Errorf("Logs: wrong query string. Want %#v. Got %#v.", expectedQs, got) + } +} + +func TestLogsRawTerminal(t *testing.T) { + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + w.Write([]byte("something happened!")) + })) + defer server.Close() + client, _ := NewClient(server.URL) + client.SkipServerVersionCheck = true + var buf bytes.Buffer + opts := LogsOptions{ + Container: "a123456", + OutputStream: &buf, + Follow: true, + RawTerminal: true, + Stdout: true, + Stderr: true, + Timestamps: true, + Tail: "100", + } + err := client.Logs(opts) + if err != nil { + t.Fatal(err) + } + expected := "something happened!" + if buf.String() != expected { + t.Errorf("Logs: wrong output. Want %q. Got %q.", expected, buf.String()) + } +} + +func TestLogsNoContainer(t *testing.T) { + var client Client + err := client.Logs(LogsOptions{}) + expected := &NoSuchContainer{ID: ""} + if !reflect.DeepEqual(err, expected) { + t.Errorf("AttachToContainer: wrong error. Want %#v. Got %#v.", expected, err) + } +} + +func TestNoSuchContainerError(t *testing.T) { + var err = &NoSuchContainer{ID: "i345"} + expected := "No such container: i345" + if got := err.Error(); got != expected { + t.Errorf("NoSuchContainer: wrong message. Want %q. Got %q.", expected, got) + } +} + +func TestNoSuchContainerErrorMessage(t *testing.T) { + var err = &NoSuchContainer{ID: "i345", Err: errors.New("some advanced error info")} + expected := "some advanced error info" + if got := err.Error(); got != expected { + t.Errorf("NoSuchContainer: wrong message. Want %q. Got %q.", expected, got) + } +} + +func TestExportContainer(t *testing.T) { + content := "exported container tar content" + out := stdoutMock{bytes.NewBufferString(content)} + client := newTestClient(&FakeRoundTripper{status: http.StatusOK}) + opts := ExportContainerOptions{ID: "4fa6e0f0c678", OutputStream: out} + err := client.ExportContainer(opts) + if err != nil { + t.Errorf("ExportContainer: caugh error %#v while exporting container, expected nil", err.Error()) + } + if out.String() != content { + t.Errorf("ExportContainer: wrong stdout. Want %#v. Got %#v.", content, out.String()) + } +} + +func TestExportContainerViaUnixSocket(t *testing.T) { + if runtime.GOOS != "darwin" { + t.Skip(fmt.Sprintf("skipping test on %s", runtime.GOOS)) + } + content := "exported container tar content" + var buf []byte + out := bytes.NewBuffer(buf) + tempSocket := tempfile("export_socket") + defer os.Remove(tempSocket) + endpoint := "unix://" + tempSocket + u, _ := parseEndpoint(endpoint, false) + client := Client{ + HTTPClient: http.DefaultClient, + endpoint: endpoint, + endpointURL: u, + SkipServerVersionCheck: true, + } + listening := make(chan string) + done := make(chan int) + go runStreamConnServer(t, "unix", tempSocket, listening, done) + <-listening // wait for server to start + opts := ExportContainerOptions{ID: "4fa6e0f0c678", OutputStream: out} + err := client.ExportContainer(opts) + <-done // make sure server stopped + if err != nil { + t.Errorf("ExportContainer: caugh error %#v while exporting container, expected nil", err.Error()) + } + if out.String() != content { + t.Errorf("ExportContainer: wrong stdout. Want %#v. Got %#v.", content, out.String()) + } +} + +func runStreamConnServer(t *testing.T, network, laddr string, listening chan<- string, done chan<- int) { + defer close(done) + l, err := net.Listen(network, laddr) + if err != nil { + t.Errorf("Listen(%q, %q) failed: %v", network, laddr, err) + listening <- "" + return + } + defer l.Close() + listening <- l.Addr().String() + c, err := l.Accept() + if err != nil { + t.Logf("Accept failed: %v", err) + return + } + c.Write([]byte("HTTP/1.1 200 OK\n\nexported container tar content")) + c.Close() +} + +func tempfile(filename string) string { + return os.TempDir() + "/" + filename + "." + strconv.Itoa(os.Getpid()) +} + +func TestExportContainerNoId(t *testing.T) { + client := Client{} + out := stdoutMock{bytes.NewBufferString("")} + err := client.ExportContainer(ExportContainerOptions{OutputStream: out}) + e, ok := err.(*NoSuchContainer) + if !ok { + t.Errorf("ExportContainer: wrong error. Want NoSuchContainer. Got %#v.", e) + } + if e.ID != "" { + t.Errorf("ExportContainer: wrong ID. Want %q. Got %q", "", e.ID) + } +} + +func TestCopyFromContainer(t *testing.T) { + content := "File content" + out := stdoutMock{bytes.NewBufferString(content)} + client := newTestClient(&FakeRoundTripper{status: http.StatusOK}) + opts := CopyFromContainerOptions{ + Container: "a123456", + OutputStream: out, + } + err := client.CopyFromContainer(opts) + if err != nil { + t.Errorf("CopyFromContainer: caugh error %#v while copying from container, expected nil", err.Error()) + } + if out.String() != content { + t.Errorf("CopyFromContainer: wrong stdout. Want %#v. Got %#v.", content, out.String()) + } +} + +func TestCopyFromContainerEmptyContainer(t *testing.T) { + client := newTestClient(&FakeRoundTripper{status: http.StatusOK}) + err := client.CopyFromContainer(CopyFromContainerOptions{}) + _, ok := err.(*NoSuchContainer) + if !ok { + t.Errorf("CopyFromContainer: invalid error returned. Want NoSuchContainer, got %#v.", err) + } +} + +func TestPassingNameOptToCreateContainerReturnsItInContainer(t *testing.T) { + jsonContainer := `{ + "Id": "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2", + "Warnings": [] +}` + fakeRT := &FakeRoundTripper{message: jsonContainer, status: http.StatusOK} + client := newTestClient(fakeRT) + config := Config{AttachStdout: true, AttachStdin: true} + opts := CreateContainerOptions{Name: "TestCreateContainer", Config: &config} + container, err := client.CreateContainer(opts) + if err != nil { + t.Fatal(err) + } + if container.Name != "TestCreateContainer" { + t.Errorf("Container name expected to be TestCreateContainer, was %s", container.Name) + } +} + +func TestAlwaysRestart(t *testing.T) { + policy := AlwaysRestart() + if policy.Name != "always" { + t.Errorf("AlwaysRestart(): wrong policy name. Want %q. Got %q", "always", policy.Name) + } + if policy.MaximumRetryCount != 0 { + t.Errorf("AlwaysRestart(): wrong MaximumRetryCount. Want 0. Got %d", policy.MaximumRetryCount) + } +} + +func TestRestartOnFailure(t *testing.T) { + const retry = 5 + policy := RestartOnFailure(retry) + if policy.Name != "on-failure" { + t.Errorf("RestartOnFailure(%d): wrong policy name. Want %q. Got %q", retry, "on-failure", policy.Name) + } + if policy.MaximumRetryCount != retry { + t.Errorf("RestartOnFailure(%d): wrong MaximumRetryCount. Want %d. Got %d", retry, retry, policy.MaximumRetryCount) + } +} + +func TestNeverRestart(t *testing.T) { + policy := NeverRestart() + if policy.Name != "no" { + t.Errorf("NeverRestart(): wrong policy name. Want %q. Got %q", "always", policy.Name) + } + if policy.MaximumRetryCount != 0 { + t.Errorf("NeverRestart(): wrong MaximumRetryCount. Want 0. Got %d", policy.MaximumRetryCount) + } +} + +func TestTopContainer(t *testing.T) { + jsonTop := `{ + "Processes": [ + [ + "ubuntu", + "3087", + "815", + "0", + "01:44", + "?", + "00:00:00", + "cmd1" + ], + [ + "root", + "3158", + "3087", + "0", + "01:44", + "?", + "00:00:01", + "cmd2" + ] + ], + "Titles": [ + "UID", + "PID", + "PPID", + "C", + "STIME", + "TTY", + "TIME", + "CMD" + ] +}` + var expected TopResult + err := json.Unmarshal([]byte(jsonTop), &expected) + if err != nil { + t.Fatal(err) + } + id := "4fa6e0f0" + fakeRT := &FakeRoundTripper{message: jsonTop, status: http.StatusOK} + client := newTestClient(fakeRT) + processes, err := client.TopContainer(id, "") + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(processes, expected) { + t.Errorf("TopContainer: Expected %#v. Got %#v.", expected, processes) + } + if len(processes.Processes) != 2 || len(processes.Processes[0]) != 8 || + processes.Processes[0][7] != "cmd1" { + t.Errorf("TopContainer: Process list to include cmd1. Got %#v.", processes) + } + expectedURI := "/containers/" + id + "/top" + if !strings.HasSuffix(fakeRT.requests[0].URL.String(), expectedURI) { + t.Errorf("TopContainer: Expected URI to have %q. Got %q.", expectedURI, fakeRT.requests[0].URL.String()) + } +} + +func TestTopContainerNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such container", status: http.StatusNotFound}) + _, err := client.TopContainer("abef348", "") + expected := &NoSuchContainer{ID: "abef348"} + if !reflect.DeepEqual(err, expected) { + t.Errorf("StopContainer: Wrong error returned. Want %#v. Got %#v.", expected, err) + } +} + +func TestTopContainerWithPsArgs(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "no such container", status: http.StatusNotFound} + client := newTestClient(fakeRT) + expectedErr := &NoSuchContainer{ID: "abef348"} + if _, err := client.TopContainer("abef348", "aux"); !reflect.DeepEqual(expectedErr, err) { + t.Errorf("TopContainer: Expected %v. Got %v.", expectedErr, err) + } + expectedURI := "/containers/abef348/top?ps_args=aux" + if !strings.HasSuffix(fakeRT.requests[0].URL.String(), expectedURI) { + t.Errorf("TopContainer: Expected URI to have %q. Got %q.", expectedURI, fakeRT.requests[0].URL.String()) + } +} + +func TestStatsTimeout(t *testing.T) { + + l, err := net.Listen("unix", "/tmp/docker_test.sock") + if err != nil { + t.Fatal(err) + } + received := false + defer l.Close() + go func() { + l.Accept() + received = true + time.Sleep(time.Millisecond * 250) + }() + client, _ := NewClient("unix:///tmp/docker_test.sock") + client.SkipServerVersionCheck = true + errC := make(chan error, 1) + statsC := make(chan *Stats) + done := make(chan bool) + go func() { + errC <- client.Stats(StatsOptions{"c", statsC, true, done, time.Millisecond * 100}) + close(errC) + }() + err = <-errC + e, ok := err.(net.Error) + if !ok || !e.Timeout() { + t.Error("Failed to receive timeout exception") + } + if !received { + t.Fatal("Failed to receive message") + } +} + +func TestStats(t *testing.T) { + jsonStats1 := `{ + "read" : "2015-01-08T22:57:31.547920715Z", + "network" : { + "rx_dropped" : 0, + "rx_bytes" : 648, + "rx_errors" : 0, + "tx_packets" : 8, + "tx_dropped" : 0, + "rx_packets" : 8, + "tx_errors" : 0, + "tx_bytes" : 648 + }, + "memory_stats" : { + "stats" : { + "total_pgmajfault" : 0, + "cache" : 0, + "mapped_file" : 0, + "total_inactive_file" : 0, + "pgpgout" : 414, + "rss" : 6537216, + "total_mapped_file" : 0, + "writeback" : 0, + "unevictable" : 0, + "pgpgin" : 477, + "total_unevictable" : 0, + "pgmajfault" : 0, + "total_rss" : 6537216, + "total_rss_huge" : 6291456, + "total_writeback" : 0, + "total_inactive_anon" : 0, + "rss_huge" : 6291456, + "hierarchical_memory_limit": 189204833, + "total_pgfault" : 964, + "total_active_file" : 0, + "active_anon" : 6537216, + "total_active_anon" : 6537216, + "total_pgpgout" : 414, + "total_cache" : 0, + "inactive_anon" : 0, + "active_file" : 0, + "pgfault" : 964, + "inactive_file" : 0, + "total_pgpgin" : 477 + }, + "max_usage" : 6651904, + "usage" : 6537216, + "failcnt" : 0, + "limit" : 67108864 + }, + "blkio_stats": { + "io_service_bytes_recursive": [ + { + "major": 8, + "minor": 0, + "op": "Read", + "value": 428795731968 + }, + { + "major": 8, + "minor": 0, + "op": "Write", + "value": 388177920 + } + ], + "io_serviced_recursive": [ + { + "major": 8, + "minor": 0, + "op": "Read", + "value": 25994442 + }, + { + "major": 8, + "minor": 0, + "op": "Write", + "value": 1734 + } + ], + "io_queue_recursive": [], + "io_service_time_recursive": [], + "io_wait_time_recursive": [], + "io_merged_recursive": [], + "io_time_recursive": [], + "sectors_recursive": [] + }, + "cpu_stats" : { + "cpu_usage" : { + "percpu_usage" : [ + 16970827, + 1839451, + 7107380, + 10571290 + ], + "usage_in_usermode" : 10000000, + "total_usage" : 36488948, + "usage_in_kernelmode" : 20000000 + }, + "system_cpu_usage" : 20091722000000000 + }, + "precpu_stats" : { + "cpu_usage" : { + "percpu_usage" : [ + 16970827, + 1839451, + 7107380, + 10571290 + ], + "usage_in_usermode" : 10000000, + "total_usage" : 36488948, + "usage_in_kernelmode" : 20000000 + }, + "system_cpu_usage" : 20091722000000000 + } + }` + // 1 second later, cache is 100 + jsonStats2 := `{ + "read" : "2015-01-08T22:57:32.547920715Z", + "network" : { + "rx_dropped" : 0, + "rx_bytes" : 648, + "rx_errors" : 0, + "tx_packets" : 8, + "tx_dropped" : 0, + "rx_packets" : 8, + "tx_errors" : 0, + "tx_bytes" : 648 + }, + "memory_stats" : { + "stats" : { + "total_pgmajfault" : 0, + "cache" : 100, + "mapped_file" : 0, + "total_inactive_file" : 0, + "pgpgout" : 414, + "rss" : 6537216, + "total_mapped_file" : 0, + "writeback" : 0, + "unevictable" : 0, + "pgpgin" : 477, + "total_unevictable" : 0, + "pgmajfault" : 0, + "total_rss" : 6537216, + "total_rss_huge" : 6291456, + "total_writeback" : 0, + "total_inactive_anon" : 0, + "rss_huge" : 6291456, + "total_pgfault" : 964, + "total_active_file" : 0, + "active_anon" : 6537216, + "total_active_anon" : 6537216, + "total_pgpgout" : 414, + "total_cache" : 0, + "inactive_anon" : 0, + "active_file" : 0, + "pgfault" : 964, + "inactive_file" : 0, + "total_pgpgin" : 477 + }, + "max_usage" : 6651904, + "usage" : 6537216, + "failcnt" : 0, + "limit" : 67108864 + }, + "blkio_stats": { + "io_service_bytes_recursive": [ + { + "major": 8, + "minor": 0, + "op": "Read", + "value": 428795731968 + }, + { + "major": 8, + "minor": 0, + "op": "Write", + "value": 388177920 + } + ], + "io_serviced_recursive": [ + { + "major": 8, + "minor": 0, + "op": "Read", + "value": 25994442 + }, + { + "major": 8, + "minor": 0, + "op": "Write", + "value": 1734 + } + ], + "io_queue_recursive": [], + "io_service_time_recursive": [], + "io_wait_time_recursive": [], + "io_merged_recursive": [], + "io_time_recursive": [], + "sectors_recursive": [] + }, + "cpu_stats" : { + "cpu_usage" : { + "percpu_usage" : [ + 16970827, + 1839451, + 7107380, + 10571290 + ], + "usage_in_usermode" : 10000000, + "total_usage" : 36488948, + "usage_in_kernelmode" : 20000000 + }, + "system_cpu_usage" : 20091722000000000 + }, + "precpu_stats" : { + "cpu_usage" : { + "percpu_usage" : [ + 16970827, + 1839451, + 7107380, + 10571290 + ], + "usage_in_usermode" : 10000000, + "total_usage" : 36488948, + "usage_in_kernelmode" : 20000000 + }, + "system_cpu_usage" : 20091722000000000 + } + }` + var expected1 Stats + var expected2 Stats + err := json.Unmarshal([]byte(jsonStats1), &expected1) + if err != nil { + t.Fatal(err) + } + err = json.Unmarshal([]byte(jsonStats2), &expected2) + if err != nil { + t.Fatal(err) + } + id := "4fa6e0f0" + + var req http.Request + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + w.Header().Set("Content-Type", "application/json") + w.Write([]byte(jsonStats1)) + w.Write([]byte(jsonStats2)) + req = *r + })) + defer server.Close() + client, _ := NewClient(server.URL) + client.SkipServerVersionCheck = true + errC := make(chan error, 1) + statsC := make(chan *Stats) + done := make(chan bool) + go func() { + errC <- client.Stats(StatsOptions{id, statsC, true, done, 0}) + close(errC) + }() + var resultStats []*Stats + for { + stats, ok := <-statsC + if !ok { + break + } + resultStats = append(resultStats, stats) + } + err = <-errC + if err != nil { + t.Fatal(err) + } + if len(resultStats) != 2 { + t.Fatalf("Stats: Expected 2 results. Got %d.", len(resultStats)) + } + if !reflect.DeepEqual(resultStats[0], &expected1) { + t.Errorf("Stats: Expected:\n%+v\nGot:\n%+v", expected1, resultStats[0]) + } + if !reflect.DeepEqual(resultStats[1], &expected2) { + t.Errorf("Stats: Expected:\n%+v\nGot:\n%+v", expected2, resultStats[1]) + } + if req.Method != "GET" { + t.Errorf("Stats: wrong HTTP method. Want GET. Got %s.", req.Method) + } + u, _ := url.Parse(client.getURL("/containers/" + id + "/stats")) + if req.URL.Path != u.Path { + t.Errorf("Stats: wrong HTTP path. Want %q. Got %q.", u.Path, req.URL.Path) + } +} + +func TestStatsContainerNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such container", status: http.StatusNotFound}) + statsC := make(chan *Stats) + done := make(chan bool) + err := client.Stats(StatsOptions{"abef348", statsC, true, done, 0}) + expected := &NoSuchContainer{ID: "abef348"} + if !reflect.DeepEqual(err, expected) { + t.Errorf("Stats: Wrong error returned. Want %#v. Got %#v.", expected, err) + } +} + +func TestRenameContainer(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + opts := RenameContainerOptions{ID: "something_old", Name: "something_new"} + err := client.RenameContainer(opts) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("RenameContainer: wrong HTTP method. Want %q. Got %q.", "POST", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/containers/something_old/rename?name=something_new")) + if gotPath := req.URL.Path; gotPath != expectedURL.Path { + t.Errorf("RenameContainer: Wrong path in request. Want %q. Got %q.", expectedURL.Path, gotPath) + } + expectedValues := expectedURL.Query()["name"] + actualValues := req.URL.Query()["name"] + if len(actualValues) != 1 || expectedValues[0] != actualValues[0] { + t.Errorf("RenameContainer: Wrong params in request. Want %q. Got %q.", expectedValues, actualValues) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go new file mode 100644 index 0000000000000..c54b0b0e80211 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env.go @@ -0,0 +1,168 @@ +// Copyright 2014 Docker authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the DOCKER-LICENSE file. + +package docker + +import ( + "encoding/json" + "fmt" + "io" + "strconv" + "strings" +) + +// Env represents a list of key-pair represented in the form KEY=VALUE. +type Env []string + +// Get returns the string value of the given key. +func (env *Env) Get(key string) (value string) { + return env.Map()[key] +} + +// Exists checks whether the given key is defined in the internal Env +// representation. +func (env *Env) Exists(key string) bool { + _, exists := env.Map()[key] + return exists +} + +// GetBool returns a boolean representation of the given key. The key is false +// whenever its value if 0, no, false, none or an empty string. Any other value +// will be interpreted as true. +func (env *Env) GetBool(key string) (value bool) { + s := strings.ToLower(strings.Trim(env.Get(key), " \t")) + if s == "" || s == "0" || s == "no" || s == "false" || s == "none" { + return false + } + return true +} + +// SetBool defines a boolean value to the given key. +func (env *Env) SetBool(key string, value bool) { + if value { + env.Set(key, "1") + } else { + env.Set(key, "0") + } +} + +// GetInt returns the value of the provided key, converted to int. +// +// It the value cannot be represented as an integer, it returns -1. +func (env *Env) GetInt(key string) int { + return int(env.GetInt64(key)) +} + +// SetInt defines an integer value to the given key. +func (env *Env) SetInt(key string, value int) { + env.Set(key, strconv.Itoa(value)) +} + +// GetInt64 returns the value of the provided key, converted to int64. +// +// It the value cannot be represented as an integer, it returns -1. +func (env *Env) GetInt64(key string) int64 { + s := strings.Trim(env.Get(key), " \t") + val, err := strconv.ParseInt(s, 10, 64) + if err != nil { + return -1 + } + return val +} + +// SetInt64 defines an integer (64-bit wide) value to the given key. +func (env *Env) SetInt64(key string, value int64) { + env.Set(key, strconv.FormatInt(value, 10)) +} + +// GetJSON unmarshals the value of the provided key in the provided iface. +// +// iface is a value that can be provided to the json.Unmarshal function. +func (env *Env) GetJSON(key string, iface interface{}) error { + sval := env.Get(key) + if sval == "" { + return nil + } + return json.Unmarshal([]byte(sval), iface) +} + +// SetJSON marshals the given value to JSON format and stores it using the +// provided key. +func (env *Env) SetJSON(key string, value interface{}) error { + sval, err := json.Marshal(value) + if err != nil { + return err + } + env.Set(key, string(sval)) + return nil +} + +// GetList returns a list of strings matching the provided key. It handles the +// list as a JSON representation of a list of strings. +// +// If the given key matches to a single string, it will return a list +// containing only the value that matches the key. +func (env *Env) GetList(key string) []string { + sval := env.Get(key) + if sval == "" { + return nil + } + var l []string + if err := json.Unmarshal([]byte(sval), &l); err != nil { + l = append(l, sval) + } + return l +} + +// SetList stores the given list in the provided key, after serializing it to +// JSON format. +func (env *Env) SetList(key string, value []string) error { + return env.SetJSON(key, value) +} + +// Set defines the value of a key to the given string. +func (env *Env) Set(key, value string) { + *env = append(*env, key+"="+value) +} + +// Decode decodes `src` as a json dictionary, and adds each decoded key-value +// pair to the environment. +// +// If `src` cannot be decoded as a json dictionary, an error is returned. +func (env *Env) Decode(src io.Reader) error { + m := make(map[string]interface{}) + if err := json.NewDecoder(src).Decode(&m); err != nil { + return err + } + for k, v := range m { + env.SetAuto(k, v) + } + return nil +} + +// SetAuto will try to define the Set* method to call based on the given value. +func (env *Env) SetAuto(key string, value interface{}) { + if fval, ok := value.(float64); ok { + env.SetInt64(key, int64(fval)) + } else if sval, ok := value.(string); ok { + env.Set(key, sval) + } else if val, err := json.Marshal(value); err == nil { + env.Set(key, string(val)) + } else { + env.Set(key, fmt.Sprintf("%v", value)) + } +} + +// Map returns the map representation of the env. +func (env *Env) Map() map[string]string { + if len(*env) == 0 { + return nil + } + m := make(map[string]string) + for _, kv := range *env { + parts := strings.SplitN(kv, "=", 2) + m[parts[0]] = parts[1] + } + return m +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env_test.go new file mode 100644 index 0000000000000..df5169d06bada --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/env_test.go @@ -0,0 +1,351 @@ +// Copyright 2014 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the DOCKER-LICENSE file. + +package docker + +import ( + "bytes" + "errors" + "reflect" + "sort" + "testing" +) + +func TestGet(t *testing.T) { + var tests = []struct { + input []string + query string + expected string + }{ + {[]string{"PATH=/usr/bin:/bin", "PYTHONPATH=/usr/local"}, "PATH", "/usr/bin:/bin"}, + {[]string{"PATH=/usr/bin:/bin", "PYTHONPATH=/usr/local"}, "PYTHONPATH", "/usr/local"}, + {[]string{"PATH=/usr/bin:/bin", "PYTHONPATH=/usr/local"}, "PYTHONPATHI", ""}, + {[]string{"WAT="}, "WAT", ""}, + } + for _, tt := range tests { + env := Env(tt.input) + got := env.Get(tt.query) + if got != tt.expected { + t.Errorf("Env.Get(%q): wrong result. Want %q. Got %q", tt.query, tt.expected, got) + } + } +} + +func TestExists(t *testing.T) { + var tests = []struct { + input []string + query string + expected bool + }{ + {[]string{"WAT=", "PYTHONPATH=/usr/local"}, "WAT", true}, + {[]string{"PATH=/usr/bin:/bin", "PYTHONPATH=/usr/local"}, "PYTHONPATH", true}, + {[]string{"PATH=/usr/bin:/bin", "PYTHONPATH=/usr/local"}, "PYTHONPATHI", false}, + } + for _, tt := range tests { + env := Env(tt.input) + got := env.Exists(tt.query) + if got != tt.expected { + t.Errorf("Env.Exists(%q): wrong result. Want %v. Got %v", tt.query, tt.expected, got) + } + } +} + +func TestGetBool(t *testing.T) { + var tests = []struct { + input string + expected bool + }{ + {"EMTPY_VAR", false}, {"ZERO_VAR", false}, {"NO_VAR", false}, + {"FALSE_VAR", false}, {"NONE_VAR", false}, {"TRUE_VAR", true}, + {"WAT", true}, {"PATH", true}, {"ONE_VAR", true}, {"NO_VAR_TAB", false}, + } + env := Env([]string{ + "EMPTY_VAR=", "ZERO_VAR=0", "NO_VAR=no", "FALSE_VAR=false", + "NONE_VAR=none", "TRUE_VAR=true", "WAT=wat", "PATH=/usr/bin:/bin", + "ONE_VAR=1", "NO_VAR_TAB=0 \t\t\t", + }) + for _, tt := range tests { + got := env.GetBool(tt.input) + if got != tt.expected { + t.Errorf("Env.GetBool(%q): wrong result. Want %v. Got %v.", tt.input, tt.expected, got) + } + } +} + +func TestSetBool(t *testing.T) { + var tests = []struct { + input bool + expected string + }{ + {true, "1"}, {false, "0"}, + } + for _, tt := range tests { + var env Env + env.SetBool("SOME", tt.input) + if got := env.Get("SOME"); got != tt.expected { + t.Errorf("Env.SetBool(%v): wrong result. Want %q. Got %q", tt.input, tt.expected, got) + } + } +} + +func TestGetInt(t *testing.T) { + var tests = []struct { + input string + expected int + }{ + {"NEGATIVE_INTEGER", -10}, {"NON_INTEGER", -1}, {"ONE", 1}, {"TWO", 2}, + } + env := Env([]string{"NEGATIVE_INTEGER=-10", "NON_INTEGER=wat", "ONE=1", "TWO=2"}) + for _, tt := range tests { + got := env.GetInt(tt.input) + if got != tt.expected { + t.Errorf("Env.GetInt(%q): wrong result. Want %d. Got %d", tt.input, tt.expected, got) + } + } +} + +func TestSetInt(t *testing.T) { + var tests = []struct { + input int + expected string + }{ + {10, "10"}, {13, "13"}, {7, "7"}, {33, "33"}, + {0, "0"}, {-34, "-34"}, + } + for _, tt := range tests { + var env Env + env.SetInt("SOME", tt.input) + if got := env.Get("SOME"); got != tt.expected { + t.Errorf("Env.SetBool(%d): wrong result. Want %q. Got %q", tt.input, tt.expected, got) + } + } +} + +func TestGetInt64(t *testing.T) { + var tests = []struct { + input string + expected int64 + }{ + {"NEGATIVE_INTEGER", -10}, {"NON_INTEGER", -1}, {"ONE", 1}, {"TWO", 2}, + } + env := Env([]string{"NEGATIVE_INTEGER=-10", "NON_INTEGER=wat", "ONE=1", "TWO=2"}) + for _, tt := range tests { + got := env.GetInt64(tt.input) + if got != tt.expected { + t.Errorf("Env.GetInt64(%q): wrong result. Want %d. Got %d", tt.input, tt.expected, got) + } + } +} + +func TestSetInt64(t *testing.T) { + var tests = []struct { + input int64 + expected string + }{ + {10, "10"}, {13, "13"}, {7, "7"}, {33, "33"}, + {0, "0"}, {-34, "-34"}, + } + for _, tt := range tests { + var env Env + env.SetInt64("SOME", tt.input) + if got := env.Get("SOME"); got != tt.expected { + t.Errorf("Env.SetBool(%d): wrong result. Want %q. Got %q", tt.input, tt.expected, got) + } + } +} + +func TestGetJSON(t *testing.T) { + var p struct { + Name string `json:"name"` + Age int `json:"age"` + } + var env Env + env.Set("person", `{"name":"Gopher","age":5}`) + err := env.GetJSON("person", &p) + if err != nil { + t.Error(err) + } + if p.Name != "Gopher" { + t.Errorf("Env.GetJSON(%q): wrong name. Want %q. Got %q", "person", "Gopher", p.Name) + } + if p.Age != 5 { + t.Errorf("Env.GetJSON(%q): wrong age. Want %d. Got %d", "person", 5, p.Age) + } +} + +func TestGetJSONAbsent(t *testing.T) { + var l []string + var env Env + err := env.GetJSON("person", &l) + if err != nil { + t.Error(err) + } + if l != nil { + t.Errorf("Env.GetJSON(): get unexpected list %v", l) + } +} + +func TestGetJSONFailure(t *testing.T) { + var p []string + var env Env + env.Set("list-person", `{"name":"Gopher","age":5}`) + err := env.GetJSON("list-person", &p) + if err == nil { + t.Errorf("Env.GetJSON(%q): got unexpected error.", "list-person") + } +} + +func TestSetJSON(t *testing.T) { + var p1 = struct { + Name string `json:"name"` + Age int `json:"age"` + }{Name: "Gopher", Age: 5} + var env Env + err := env.SetJSON("person", p1) + if err != nil { + t.Error(err) + } + var p2 struct { + Name string `json:"name"` + Age int `json:"age"` + } + err = env.GetJSON("person", &p2) + if err != nil { + t.Error(err) + } + if !reflect.DeepEqual(p1, p2) { + t.Errorf("Env.SetJSON(%q): wrong result. Want %v. Got %v", "person", p1, p2) + } +} + +func TestSetJSONFailure(t *testing.T) { + var env Env + err := env.SetJSON("person", unmarshable{}) + if err == nil { + t.Error("Env.SetJSON(): got unexpected error") + } + if env.Exists("person") { + t.Errorf("Env.SetJSON(): should not define the key %q, but did", "person") + } +} + +func TestGetList(t *testing.T) { + var tests = []struct { + input string + expected []string + }{ + {"WAT=wat", []string{"wat"}}, + {`WAT=["wat","wet","wit","wot","wut"]`, []string{"wat", "wet", "wit", "wot", "wut"}}, + {"WAT=", nil}, + } + for _, tt := range tests { + env := Env([]string{tt.input}) + got := env.GetList("WAT") + if !reflect.DeepEqual(got, tt.expected) { + t.Errorf("Env.GetList(%q): wrong result. Want %v. Got %v", "WAT", tt.expected, got) + } + } +} + +func TestSetList(t *testing.T) { + list := []string{"a", "b", "c"} + var env Env + if err := env.SetList("SOME", list); err != nil { + t.Error(err) + } + if got := env.GetList("SOME"); !reflect.DeepEqual(got, list) { + t.Errorf("Env.SetList(%v): wrong result. Got %v", list, got) + } +} + +func TestSet(t *testing.T) { + var env Env + env.Set("PATH", "/home/bin:/bin") + env.Set("SOMETHING", "/usr/bin") + env.Set("PATH", "/bin") + if expected, got := "/usr/bin", env.Get("SOMETHING"); got != expected { + t.Errorf("Env.Set(%q): wrong result. Want %q. Got %q", expected, expected, got) + } + if expected, got := "/bin", env.Get("PATH"); got != expected { + t.Errorf("Env.Set(%q): wrong result. Want %q. Got %q", expected, expected, got) + } +} + +func TestDecode(t *testing.T) { + var tests = []struct { + input string + expectedOut []string + expectedErr string + }{ + { + `{"PATH":"/usr/bin:/bin","containers":54,"wat":["123","345"]}`, + []string{"PATH=/usr/bin:/bin", "containers=54", `wat=["123","345"]`}, + "", + }, + {"}}", nil, "invalid character '}' looking for beginning of value"}, + {`{}`, nil, ""}, + } + for _, tt := range tests { + var env Env + err := env.Decode(bytes.NewBufferString(tt.input)) + if tt.expectedErr == "" { + if err != nil { + t.Error(err) + } + } else if tt.expectedErr != err.Error() { + t.Errorf("Env.Decode(): invalid error. Want %q. Got %q.", tt.expectedErr, err) + } + got := []string(env) + sort.Strings(got) + sort.Strings(tt.expectedOut) + if !reflect.DeepEqual(got, tt.expectedOut) { + t.Errorf("Env.Decode(): wrong result. Want %v. Got %v.", tt.expectedOut, got) + } + } +} + +func TestSetAuto(t *testing.T) { + buf := bytes.NewBufferString("oi") + var tests = []struct { + input interface{} + expected string + }{ + {10, "10"}, + {10.3, "10"}, + {"oi", "oi"}, + {buf, "{}"}, + {unmarshable{}, "{}"}, + } + for _, tt := range tests { + var env Env + env.SetAuto("SOME", tt.input) + if got := env.Get("SOME"); got != tt.expected { + t.Errorf("Env.SetAuto(%v): wrong result. Want %q. Got %q", tt.input, tt.expected, got) + } + } +} + +func TestMap(t *testing.T) { + var tests = []struct { + input []string + expected map[string]string + }{ + {[]string{"PATH=/usr/bin:/bin", "PYTHONPATH=/usr/local"}, map[string]string{"PATH": "/usr/bin:/bin", "PYTHONPATH": "/usr/local"}}, + {nil, nil}, + } + for _, tt := range tests { + env := Env(tt.input) + got := env.Map() + if !reflect.DeepEqual(got, tt.expected) { + t.Errorf("Env.Map(): wrong result. Want %v. Got %v", tt.expected, got) + } + } +} + +type unmarshable struct { +} + +func (unmarshable) MarshalJSON() ([]byte, error) { + return nil, errors.New("cannot marshal") +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go new file mode 100644 index 0000000000000..5a85983cc896f --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event.go @@ -0,0 +1,305 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "crypto/tls" + "encoding/json" + "errors" + "fmt" + "io" + "math" + "net" + "net/http" + "net/http/httputil" + "sync" + "sync/atomic" + "time" +) + +// APIEvents represents an event returned by the API. +type APIEvents struct { + Status string `json:"Status,omitempty" yaml:"Status,omitempty"` + ID string `json:"ID,omitempty" yaml:"ID,omitempty"` + From string `json:"From,omitempty" yaml:"From,omitempty"` + Time int64 `json:"Time,omitempty" yaml:"Time,omitempty"` +} + +type eventMonitoringState struct { + sync.RWMutex + sync.WaitGroup + enabled bool + lastSeen *int64 + C chan *APIEvents + errC chan error + listeners []chan<- *APIEvents +} + +const ( + maxMonitorConnRetries = 5 + retryInitialWaitTime = 10. +) + +var ( + // ErrNoListeners is the error returned when no listeners are available + // to receive an event. + ErrNoListeners = errors.New("no listeners present to receive event") + + // ErrListenerAlreadyExists is the error returned when the listerner already + // exists. + ErrListenerAlreadyExists = errors.New("listener already exists for docker events") + + // EOFEvent is sent when the event listener receives an EOF error. + EOFEvent = &APIEvents{ + Status: "EOF", + } +) + +// AddEventListener adds a new listener to container events in the Docker API. +// +// The parameter is a channel through which events will be sent. +func (c *Client) AddEventListener(listener chan<- *APIEvents) error { + var err error + if !c.eventMonitor.isEnabled() { + err = c.eventMonitor.enableEventMonitoring(c) + if err != nil { + return err + } + } + err = c.eventMonitor.addListener(listener) + if err != nil { + return err + } + return nil +} + +// RemoveEventListener removes a listener from the monitor. +func (c *Client) RemoveEventListener(listener chan *APIEvents) error { + err := c.eventMonitor.removeListener(listener) + if err != nil { + return err + } + if len(c.eventMonitor.listeners) == 0 { + c.eventMonitor.disableEventMonitoring() + } + return nil +} + +func (eventState *eventMonitoringState) addListener(listener chan<- *APIEvents) error { + eventState.Lock() + defer eventState.Unlock() + if listenerExists(listener, &eventState.listeners) { + return ErrListenerAlreadyExists + } + eventState.Add(1) + eventState.listeners = append(eventState.listeners, listener) + return nil +} + +func (eventState *eventMonitoringState) removeListener(listener chan<- *APIEvents) error { + eventState.Lock() + defer eventState.Unlock() + if listenerExists(listener, &eventState.listeners) { + var newListeners []chan<- *APIEvents + for _, l := range eventState.listeners { + if l != listener { + newListeners = append(newListeners, l) + } + } + eventState.listeners = newListeners + eventState.Add(-1) + } + return nil +} + +func (eventState *eventMonitoringState) closeListeners() { + for _, l := range eventState.listeners { + close(l) + eventState.Add(-1) + } + eventState.listeners = nil +} + +func listenerExists(a chan<- *APIEvents, list *[]chan<- *APIEvents) bool { + for _, b := range *list { + if b == a { + return true + } + } + return false +} + +func (eventState *eventMonitoringState) enableEventMonitoring(c *Client) error { + eventState.Lock() + defer eventState.Unlock() + if !eventState.enabled { + eventState.enabled = true + var lastSeenDefault = int64(0) + eventState.lastSeen = &lastSeenDefault + eventState.C = make(chan *APIEvents, 100) + eventState.errC = make(chan error, 1) + go eventState.monitorEvents(c) + } + return nil +} + +func (eventState *eventMonitoringState) disableEventMonitoring() error { + eventState.Lock() + defer eventState.Unlock() + + eventState.closeListeners() + + eventState.Wait() + + if eventState.enabled { + eventState.enabled = false + close(eventState.C) + close(eventState.errC) + } + return nil +} + +func (eventState *eventMonitoringState) monitorEvents(c *Client) { + var err error + for eventState.noListeners() { + time.Sleep(10 * time.Millisecond) + } + if err = eventState.connectWithRetry(c); err != nil { + // terminate if connect failed + eventState.disableEventMonitoring() + return + } + for eventState.isEnabled() { + timeout := time.After(100 * time.Millisecond) + select { + case ev, ok := <-eventState.C: + if !ok { + return + } + if ev == EOFEvent { + eventState.disableEventMonitoring() + return + } + eventState.updateLastSeen(ev) + go eventState.sendEvent(ev) + case err = <-eventState.errC: + if err == ErrNoListeners { + eventState.disableEventMonitoring() + return + } else if err != nil { + defer func() { go eventState.monitorEvents(c) }() + return + } + case <-timeout: + continue + } + } +} + +func (eventState *eventMonitoringState) connectWithRetry(c *Client) error { + var retries int + var err error + for err = c.eventHijack(atomic.LoadInt64(eventState.lastSeen), eventState.C, eventState.errC); err != nil && retries < maxMonitorConnRetries; retries++ { + waitTime := int64(retryInitialWaitTime * math.Pow(2, float64(retries))) + time.Sleep(time.Duration(waitTime) * time.Millisecond) + err = c.eventHijack(atomic.LoadInt64(eventState.lastSeen), eventState.C, eventState.errC) + } + return err +} + +func (eventState *eventMonitoringState) noListeners() bool { + eventState.RLock() + defer eventState.RUnlock() + return len(eventState.listeners) == 0 +} + +func (eventState *eventMonitoringState) isEnabled() bool { + eventState.RLock() + defer eventState.RUnlock() + return eventState.enabled +} + +func (eventState *eventMonitoringState) sendEvent(event *APIEvents) { + eventState.RLock() + defer eventState.RUnlock() + eventState.Add(1) + defer eventState.Done() + if eventState.enabled { + if len(eventState.listeners) == 0 { + eventState.errC <- ErrNoListeners + return + } + + for _, listener := range eventState.listeners { + listener <- event + } + } +} + +func (eventState *eventMonitoringState) updateLastSeen(e *APIEvents) { + eventState.Lock() + defer eventState.Unlock() + if atomic.LoadInt64(eventState.lastSeen) < e.Time { + atomic.StoreInt64(eventState.lastSeen, e.Time) + } +} + +func (c *Client) eventHijack(startTime int64, eventChan chan *APIEvents, errChan chan error) error { + uri := "/events" + if startTime != 0 { + uri += fmt.Sprintf("?since=%d", startTime) + } + protocol := c.endpointURL.Scheme + address := c.endpointURL.Path + if protocol != "unix" { + protocol = "tcp" + address = c.endpointURL.Host + } + var dial net.Conn + var err error + if c.TLSConfig == nil { + dial, err = net.Dial(protocol, address) + } else { + dial, err = tls.Dial(protocol, address, c.TLSConfig) + } + if err != nil { + return err + } + conn := httputil.NewClientConn(dial, nil) + req, err := http.NewRequest("GET", uri, nil) + if err != nil { + return err + } + res, err := conn.Do(req) + if err != nil { + return err + } + go func(res *http.Response, conn *httputil.ClientConn) { + defer conn.Close() + defer res.Body.Close() + decoder := json.NewDecoder(res.Body) + for { + var event APIEvents + if err = decoder.Decode(&event); err != nil { + if err == io.EOF || err == io.ErrUnexpectedEOF { + if c.eventMonitor.isEnabled() { + // Signal that we're exiting. + eventChan <- EOFEvent + } + break + } + errChan <- err + } + if event.Time == 0 { + continue + } + if !c.eventMonitor.isEnabled() { + return + } + eventChan <- &event + } + }(res, conn) + return nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event_test.go new file mode 100644 index 0000000000000..a308538ccd5f4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/event_test.go @@ -0,0 +1,132 @@ +// Copyright 2014 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "bufio" + "crypto/tls" + "crypto/x509" + "fmt" + "io/ioutil" + "net/http" + "net/http/httptest" + "strings" + "testing" + "time" +) + +func TestEventListeners(t *testing.T) { + testEventListeners("TestEventListeners", t, httptest.NewServer, NewClient) +} + +func TestTLSEventListeners(t *testing.T) { + testEventListeners("TestTLSEventListeners", t, func(handler http.Handler) *httptest.Server { + server := httptest.NewUnstartedServer(handler) + + cert, err := tls.LoadX509KeyPair("testing/data/server.pem", "testing/data/serverkey.pem") + if err != nil { + t.Fatalf("Error loading server key pair: %s", err) + } + + caCert, err := ioutil.ReadFile("testing/data/ca.pem") + if err != nil { + t.Fatalf("Error loading ca certificate: %s", err) + } + caPool := x509.NewCertPool() + if !caPool.AppendCertsFromPEM(caCert) { + t.Fatalf("Could not add ca certificate") + } + + server.TLS = &tls.Config{ + Certificates: []tls.Certificate{cert}, + RootCAs: caPool, + } + server.StartTLS() + return server + }, func(url string) (*Client, error) { + return NewTLSClient(url, "testing/data/cert.pem", "testing/data/key.pem", "testing/data/ca.pem") + }) +} + +func testEventListeners(testName string, t *testing.T, buildServer func(http.Handler) *httptest.Server, buildClient func(string) (*Client, error)) { + response := `{"status":"create","id":"dfdf82bd3881","from":"base:latest","time":1374067924} +{"status":"start","id":"dfdf82bd3881","from":"base:latest","time":1374067924} +{"status":"stop","id":"dfdf82bd3881","from":"base:latest","time":1374067966} +{"status":"destroy","id":"dfdf82bd3881","from":"base:latest","time":1374067970} +` + + server := buildServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + rsc := bufio.NewScanner(strings.NewReader(response)) + for rsc.Scan() { + w.Write([]byte(rsc.Text())) + w.(http.Flusher).Flush() + time.Sleep(10 * time.Millisecond) + } + })) + defer server.Close() + + client, err := buildClient(server.URL) + if err != nil { + t.Errorf("Failed to create client: %s", err) + } + client.SkipServerVersionCheck = true + + listener := make(chan *APIEvents, 10) + defer func() { + time.Sleep(10 * time.Millisecond) + if err := client.RemoveEventListener(listener); err != nil { + t.Error(err) + } + }() + + err = client.AddEventListener(listener) + if err != nil { + t.Errorf("Failed to add event listener: %s", err) + } + + timeout := time.After(1 * time.Second) + var count int + + for { + select { + case msg := <-listener: + t.Logf("Received: %v", *msg) + count++ + err = checkEvent(count, msg) + if err != nil { + t.Fatalf("Check event failed: %s", err) + } + if count == 4 { + return + } + case <-timeout: + t.Fatalf("%s timed out waiting on events", testName) + } + } +} + +func checkEvent(index int, event *APIEvents) error { + if event.ID != "dfdf82bd3881" { + return fmt.Errorf("event ID did not match. Expected dfdf82bd3881 got %s", event.ID) + } + if event.From != "base:latest" { + return fmt.Errorf("event from did not match. Expected base:latest got %s", event.From) + } + var status string + switch index { + case 1: + status = "create" + case 2: + status = "start" + case 3: + status = "stop" + case 4: + status = "destroy" + } + if event.Status != status { + return fmt.Errorf("event status did not match. Expected %s got %s", status, event.Status) + } + return nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/example_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/example_test.go new file mode 100644 index 0000000000000..8c2c719e6e8c2 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/example_test.go @@ -0,0 +1,168 @@ +// Copyright 2014 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker_test + +import ( + "archive/tar" + "bytes" + "fmt" + "io" + "log" + "time" + + "github.com/fsouza/go-dockerclient" +) + +func ExampleClient_AttachToContainer() { + client, err := docker.NewClient("http://localhost:4243") + if err != nil { + log.Fatal(err) + } + client.SkipServerVersionCheck = true + // Reading logs from container a84849 and sending them to buf. + var buf bytes.Buffer + err = client.AttachToContainer(docker.AttachToContainerOptions{ + Container: "a84849", + OutputStream: &buf, + Logs: true, + Stdout: true, + Stderr: true, + }) + if err != nil { + log.Fatal(err) + } + log.Println(buf.String()) + buf.Reset() + err = client.AttachToContainer(docker.AttachToContainerOptions{ + Container: "a84849", + OutputStream: &buf, + Stdout: true, + Stream: true, + }) + if err != nil { + log.Fatal(err) + } + log.Println(buf.String()) +} + +func ExampleClient_CopyFromContainer() { + client, err := docker.NewClient("http://localhost:4243") + if err != nil { + log.Fatal(err) + } + cid := "a84849" + var buf bytes.Buffer + filename := "/tmp/output.txt" + err = client.CopyFromContainer(docker.CopyFromContainerOptions{ + Container: cid, + Resource: filename, + OutputStream: &buf, + }) + if err != nil { + log.Fatalf("Error while copying from %s: %s\n", cid, err) + } + content := new(bytes.Buffer) + r := bytes.NewReader(buf.Bytes()) + tr := tar.NewReader(r) + tr.Next() + if err != nil && err != io.EOF { + log.Fatal(err) + } + if _, err := io.Copy(content, tr); err != nil { + log.Fatal(err) + } + log.Println(buf.String()) +} + +func ExampleClient_BuildImage() { + client, err := docker.NewClient("http://localhost:4243") + if err != nil { + log.Fatal(err) + } + + t := time.Now() + inputbuf, outputbuf := bytes.NewBuffer(nil), bytes.NewBuffer(nil) + tr := tar.NewWriter(inputbuf) + tr.WriteHeader(&tar.Header{Name: "Dockerfile", Size: 10, ModTime: t, AccessTime: t, ChangeTime: t}) + tr.Write([]byte("FROM base\n")) + tr.Close() + opts := docker.BuildImageOptions{ + Name: "test", + InputStream: inputbuf, + OutputStream: outputbuf, + } + if err := client.BuildImage(opts); err != nil { + log.Fatal(err) + } +} + +func ExampleClient_ListenEvents() { + client, err := docker.NewClient("http://localhost:4243") + if err != nil { + log.Fatal(err) + } + + listener := make(chan *docker.APIEvents) + err = client.AddEventListener(listener) + if err != nil { + log.Fatal(err) + } + + defer func() { + + err = client.RemoveEventListener(listener) + if err != nil { + log.Fatal(err) + } + + }() + + timeout := time.After(1 * time.Second) + + for { + select { + case msg := <-listener: + log.Println(msg) + case <-timeout: + break + } + } + +} + +func ExampleEnv_Map() { + e := docker.Env([]string{"A=1", "B=2", "C=3"}) + envs := e.Map() + for k, v := range envs { + fmt.Printf("%s=%q\n", k, v) + } +} + +func ExampleEnv_SetJSON() { + type Person struct { + Name string + Age int + } + p := Person{Name: "Gopher", Age: 4} + var e docker.Env + err := e.SetJSON("person", p) + if err != nil { + log.Fatal(err) + } +} + +func ExampleEnv_GetJSON() { + type Person struct { + Name string + Age int + } + p := Person{Name: "Gopher", Age: 4} + var e docker.Env + e.Set("person", `{"name":"Gopher","age":4}`) + err := e.GetJSON("person", &p) + if err != nil { + log.Fatal(err) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/exec.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/exec.go new file mode 100644 index 0000000000000..bc7c5cfcd80f5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/exec.go @@ -0,0 +1,185 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// Docs can currently be found at https://github.com/docker/docker/blob/master/docs/sources/reference/api/docker_remote_api_v1.15.md#exec-create + +package docker + +import ( + "encoding/json" + "fmt" + "io" + "net/http" + "net/url" + "strconv" +) + +// CreateExecOptions specify parameters to the CreateExecContainer function. +// +// See http://goo.gl/8izrzI for more details +type CreateExecOptions struct { + AttachStdin bool `json:"AttachStdin,omitempty" yaml:"AttachStdin,omitempty"` + AttachStdout bool `json:"AttachStdout,omitempty" yaml:"AttachStdout,omitempty"` + AttachStderr bool `json:"AttachStderr,omitempty" yaml:"AttachStderr,omitempty"` + Tty bool `json:"Tty,omitempty" yaml:"Tty,omitempty"` + Cmd []string `json:"Cmd,omitempty" yaml:"Cmd,omitempty"` + Container string `json:"Container,omitempty" yaml:"Container,omitempty"` + User string `json:"User,omitempty" yaml:"User,omitempty"` +} + +// StartExecOptions specify parameters to the StartExecContainer function. +// +// See http://goo.gl/JW8Lxl for more details +type StartExecOptions struct { + Detach bool `json:"Detach,omitempty" yaml:"Detach,omitempty"` + + Tty bool `json:"Tty,omitempty" yaml:"Tty,omitempty"` + + InputStream io.Reader `qs:"-"` + OutputStream io.Writer `qs:"-"` + ErrorStream io.Writer `qs:"-"` + + // Use raw terminal? Usually true when the container contains a TTY. + RawTerminal bool `qs:"-"` + + // If set, after a successful connect, a sentinel will be sent and then the + // client will block on receive before continuing. + // + // It must be an unbuffered channel. Using a buffered channel can lead + // to unexpected behavior. + Success chan struct{} `json:"-"` +} + +// Exec is the type representing a `docker exec` instance and containing the +// instance ID +type Exec struct { + ID string `json:"Id,omitempty" yaml:"Id,omitempty"` +} + +// ExecProcessConfig is a type describing the command associated to a Exec +// instance. It's used in the ExecInspect type. +// +// See http://goo.gl/ypQULN for more details +type ExecProcessConfig struct { + Privileged bool `json:"privileged,omitempty" yaml:"privileged,omitempty"` + User string `json:"user,omitempty" yaml:"user,omitempty"` + Tty bool `json:"tty,omitempty" yaml:"tty,omitempty"` + EntryPoint string `json:"entrypoint,omitempty" yaml:"entrypoint,omitempty"` + Arguments []string `json:"arguments,omitempty" yaml:"arguments,omitempty"` +} + +// ExecInspect is a type with details about a exec instance, including the +// exit code if the command has finished running. It's returned by a api +// call to /exec/(id)/json +// +// See http://goo.gl/ypQULN for more details +type ExecInspect struct { + ID string `json:"ID,omitempty" yaml:"ID,omitempty"` + Running bool `json:"Running,omitempty" yaml:"Running,omitempty"` + ExitCode int `json:"ExitCode,omitempty" yaml:"ExitCode,omitempty"` + OpenStdin bool `json:"OpenStdin,omitempty" yaml:"OpenStdin,omitempty"` + OpenStderr bool `json:"OpenStderr,omitempty" yaml:"OpenStderr,omitempty"` + OpenStdout bool `json:"OpenStdout,omitempty" yaml:"OpenStdout,omitempty"` + ProcessConfig ExecProcessConfig `json:"ProcessConfig,omitempty" yaml:"ProcessConfig,omitempty"` + Container Container `json:"Container,omitempty" yaml:"Container,omitempty"` +} + +// CreateExec sets up an exec instance in a running container `id`, returning the exec +// instance, or an error in case of failure. +// +// See http://goo.gl/8izrzI for more details +func (c *Client) CreateExec(opts CreateExecOptions) (*Exec, error) { + path := fmt.Sprintf("/containers/%s/exec", opts.Container) + body, status, err := c.do("POST", path, doOptions{data: opts}) + if status == http.StatusNotFound { + return nil, &NoSuchContainer{ID: opts.Container} + } + if err != nil { + return nil, err + } + var exec Exec + err = json.Unmarshal(body, &exec) + if err != nil { + return nil, err + } + + return &exec, nil +} + +// StartExec starts a previously set up exec instance id. If opts.Detach is +// true, it returns after starting the exec command. Otherwise, it sets up an +// interactive session with the exec command. +// +// See http://goo.gl/JW8Lxl for more details +func (c *Client) StartExec(id string, opts StartExecOptions) error { + if id == "" { + return &NoSuchExec{ID: id} + } + + path := fmt.Sprintf("/exec/%s/start", id) + + if opts.Detach { + _, status, err := c.do("POST", path, doOptions{data: opts}) + if status == http.StatusNotFound { + return &NoSuchExec{ID: id} + } + if err != nil { + return err + } + return nil + } + + return c.hijack("POST", path, hijackOptions{ + success: opts.Success, + setRawTerminal: opts.RawTerminal, + in: opts.InputStream, + stdout: opts.OutputStream, + stderr: opts.ErrorStream, + data: opts, + }) +} + +// ResizeExecTTY resizes the tty session used by the exec command id. This API +// is valid only if Tty was specified as part of creating and starting the exec +// command. +// +// See http://goo.gl/YDSx1f for more details +func (c *Client) ResizeExecTTY(id string, height, width int) error { + params := make(url.Values) + params.Set("h", strconv.Itoa(height)) + params.Set("w", strconv.Itoa(width)) + + path := fmt.Sprintf("/exec/%s/resize?%s", id, params.Encode()) + _, _, err := c.do("POST", path, doOptions{}) + return err +} + +// InspectExec returns low-level information about the exec command id. +// +// See http://goo.gl/ypQULN for more details +func (c *Client) InspectExec(id string) (*ExecInspect, error) { + path := fmt.Sprintf("/exec/%s/json", id) + body, status, err := c.do("GET", path, doOptions{}) + if status == http.StatusNotFound { + return nil, &NoSuchExec{ID: id} + } + if err != nil { + return nil, err + } + var exec ExecInspect + err = json.Unmarshal(body, &exec) + if err != nil { + return nil, err + } + return &exec, nil +} + +// NoSuchExec is the error returned when a given exec instance does not exist. +type NoSuchExec struct { + ID string +} + +func (err *NoSuchExec) Error() string { + return "No such exec instance: " + err.ID +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/exec_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/exec_test.go new file mode 100644 index 0000000000000..2dc8d2100c608 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/exec_test.go @@ -0,0 +1,262 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "bytes" + "encoding/json" + "net/http" + "net/http/httptest" + "net/url" + "reflect" + "strings" + "testing" +) + +func TestExecCreate(t *testing.T) { + jsonContainer := `{"Id": "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2"}` + var expected struct{ ID string } + err := json.Unmarshal([]byte(jsonContainer), &expected) + if err != nil { + t.Fatal(err) + } + fakeRT := &FakeRoundTripper{message: jsonContainer, status: http.StatusOK} + client := newTestClient(fakeRT) + config := CreateExecOptions{ + Container: "test", + AttachStdin: true, + AttachStdout: true, + AttachStderr: false, + Tty: false, + Cmd: []string{"touch", "/tmp/file"}, + User: "a-user", + } + execObj, err := client.CreateExec(config) + if err != nil { + t.Fatal(err) + } + expectedID := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + if execObj.ID != expectedID { + t.Errorf("ExecCreate: wrong ID. Want %q. Got %q.", expectedID, execObj.ID) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("ExecCreate: wrong HTTP method. Want %q. Got %q.", "POST", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/containers/test/exec")) + if gotPath := req.URL.Path; gotPath != expectedURL.Path { + t.Errorf("ExecCreate: Wrong path in request. Want %q. Got %q.", expectedURL.Path, gotPath) + } + var gotBody struct{ ID string } + err = json.NewDecoder(req.Body).Decode(&gotBody) + if err != nil { + t.Fatal(err) + } +} + +func TestExecStartDetached(t *testing.T) { + execID := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + fakeRT := &FakeRoundTripper{status: http.StatusOK} + client := newTestClient(fakeRT) + config := StartExecOptions{ + Detach: true, + } + err := client.StartExec(execID, config) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("ExecStart: wrong HTTP method. Want %q. Got %q.", "POST", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/exec/" + execID + "/start")) + if gotPath := req.URL.Path; gotPath != expectedURL.Path { + t.Errorf("ExecCreate: Wrong path in request. Want %q. Got %q.", expectedURL.Path, gotPath) + } + t.Log(req.Body) + var gotBody struct{ Detach bool } + err = json.NewDecoder(req.Body).Decode(&gotBody) + if err != nil { + t.Fatal(err) + } + if !gotBody.Detach { + t.Fatal("Expected Detach in StartExecOptions to be true") + } +} + +func TestExecStartAndAttach(t *testing.T) { + var reader = strings.NewReader("send value") + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + w.Write([]byte{1, 0, 0, 0, 0, 0, 0, 5}) + w.Write([]byte("hello")) + })) + defer server.Close() + client, _ := NewClient(server.URL) + client.SkipServerVersionCheck = true + var stdout, stderr bytes.Buffer + success := make(chan struct{}) + execID := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + opts := StartExecOptions{ + OutputStream: &stdout, + ErrorStream: &stderr, + InputStream: reader, + RawTerminal: true, + Success: success, + } + go func() { + if err := client.StartExec(execID, opts); err != nil { + t.Error(err) + } + }() + <-success +} + +func TestExecResize(t *testing.T) { + execID := "4fa6e0f0c6786287e131c3852c58a2e01cc697a68231826813597e4994f1d6e2" + fakeRT := &FakeRoundTripper{status: http.StatusOK} + client := newTestClient(fakeRT) + err := client.ResizeExecTTY(execID, 10, 20) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("ExecStart: wrong HTTP method. Want %q. Got %q.", "POST", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/exec/" + execID + "/resize?h=10&w=20")) + if gotPath := req.URL.RequestURI(); gotPath != expectedURL.RequestURI() { + t.Errorf("ExecCreate: Wrong path in request. Want %q. Got %q.", expectedURL.Path, gotPath) + } +} + +func TestExecInspect(t *testing.T) { + jsonExec := `{ + "ID": "32adfeeec34250f9530ce1dafd40c6233832315e065ea6b362d745e2f63cde0e", + "Running": true, + "ExitCode": 0, + "ProcessConfig": { + "privileged": false, + "user": "", + "tty": true, + "entrypoint": "bash", + "arguments": [] + }, + "OpenStdin": true, + "OpenStderr": true, + "OpenStdout": true, + "Container": { + "State": { + "Running": true, + "Paused": false, + "Restarting": false, + "OOMKilled": false, + "Pid": 29392, + "ExitCode": 0, + "Error": "", + "StartedAt": "2015-01-21T17:08:59.634662178Z", + "FinishedAt": "0001-01-01T00:00:00Z" + }, + "ID": "922cd0568714763dc725b24b7c9801016b2a3de68e2a1dc989bf5abf07740521", + "Created": "2015-01-21T17:08:59.46407212Z", + "Path": "/bin/bash", + "Args": [ + "-lc", + "tsuru_unit_agent http://192.168.50.4:8080 689b30e0ab3adce374346de2e72512138e0e8b75 gtest /var/lib/tsuru/start && tail -f /dev/null" + ], + "Config": { + "Hostname": "922cd0568714", + "Domainname": "", + "User": "ubuntu", + "Memory": 0, + "MemorySwap": 0, + "CpuShares": 100, + "Cpuset": "", + "AttachStdin": false, + "AttachStdout": false, + "AttachStderr": false, + "PortSpecs": null, + "ExposedPorts": { + "8888/tcp": {} + }, + "Tty": false, + "OpenStdin": false, + "StdinOnce": false, + "Env": [ + "PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin" + ], + "Cmd": [ + "/bin/bash", + "-lc", + "tsuru_unit_agent http://192.168.50.4:8080 689b30e0ab3adce374346de2e72512138e0e8b75 gtest /var/lib/tsuru/start && tail -f /dev/null" + ], + "Image": "tsuru/app-gtest", + "Volumes": null, + "WorkingDir": "", + "Entrypoint": null, + "NetworkDisabled": false, + "MacAddress": "", + "OnBuild": null + }, + "Image": "a88060b8b54fde0f7168c86742d0ce83b80f3f10925d85c98fdad9ed00bef544", + "NetworkSettings": { + "IPAddress": "172.17.0.8", + "IPPrefixLen": 16, + "MacAddress": "02:42:ac:11:00:08", + "LinkLocalIPv6Address": "fe80::42:acff:fe11:8", + "LinkLocalIPv6PrefixLen": 64, + "GlobalIPv6Address": "", + "GlobalIPv6PrefixLen": 0, + "Gateway": "172.17.42.1", + "IPv6Gateway": "", + "Bridge": "docker0", + "PortMapping": null, + "Ports": { + "8888/tcp": [ + { + "HostIp": "0.0.0.0", + "HostPort": "49156" + } + ] + } + }, + "ResolvConfPath": "/var/lib/docker/containers/922cd0568714763dc725b24b7c9801016b2a3de68e2a1dc989bf5abf07740521/resolv.conf", + "HostnamePath": "/var/lib/docker/containers/922cd0568714763dc725b24b7c9801016b2a3de68e2a1dc989bf5abf07740521/hostname", + "HostsPath": "/var/lib/docker/containers/922cd0568714763dc725b24b7c9801016b2a3de68e2a1dc989bf5abf07740521/hosts", + "Name": "/c7e43b72288ee9d0270a", + "Driver": "aufs", + "ExecDriver": "native-0.2", + "MountLabel": "", + "ProcessLabel": "", + "AppArmorProfile": "", + "RestartCount": 0, + "UpdateDns": false, + "Volumes": {}, + "VolumesRW": {} + } + }` + var expected ExecInspect + err := json.Unmarshal([]byte(jsonExec), &expected) + if err != nil { + t.Fatal(err) + } + fakeRT := &FakeRoundTripper{message: jsonExec, status: http.StatusOK} + client := newTestClient(fakeRT) + expectedID := "32adfeeec34250f9530ce1dafd40c6233832315e065ea6b362d745e2f63cde0e" + execObj, err := client.InspectExec(expectedID) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(*execObj, expected) { + t.Errorf("ExecInspect: Expected %#v. Got %#v.", expected, *execObj) + } + req := fakeRT.requests[0] + if req.Method != "GET" { + t.Errorf("ExecInspect: wrong HTTP method. Want %q. Got %q.", "GET", req.Method) + } + expectedURL, _ := url.Parse(client.getURL("/exec/" + expectedID + "/json")) + if gotPath := fakeRT.requests[0].URL.Path; gotPath != expectedURL.Path { + t.Errorf("ExecInspect: Wrong path in request. Want %q. Got %q.", expectedURL.Path, gotPath) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/CHANGELOG.md b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/CHANGELOG.md new file mode 100644 index 0000000000000..a38715497a5c5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/CHANGELOG.md @@ -0,0 +1,26 @@ +# (Unreleased) + +logrus/core: improve performance of text formatter by 40% +logrus/core: expose `LevelHooks` type + +# 0.8.2 + +logrus: fix more Fatal family functions + +# 0.8.1 + +logrus: fix not exiting on `Fatalf` and `Fatalln` + +# 0.8.0 + +logrus: defaults to stderr instead of stdout +hooks/sentry: add special field for `*http.Request` +formatter/text: ignore Windows for colors + +# 0.7.3 + +formatter/\*: allow configuration of timestamp layout + +# 0.7.2 + +formatter/text: Add configuration option for time format (#158) diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/LICENSE b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/LICENSE new file mode 100644 index 0000000000000..f090cb42f370b --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/LICENSE @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2014 Simon Eskildsen + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/README.md b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/README.md new file mode 100644 index 0000000000000..4be378476f4c2 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/README.md @@ -0,0 +1,355 @@ +# Logrus :walrus: [![Build Status](https://travis-ci.org/Sirupsen/logrus.svg?branch=master)](https://travis-ci.org/Sirupsen/logrus) [![godoc reference](https://godoc.org/github.com/Sirupsen/logrus?status.png)][godoc] + +Logrus is a structured logger for Go (golang), completely API compatible with +the standard library logger. [Godoc][godoc]. **Please note the Logrus API is not +yet stable (pre 1.0). Logrus itself is completely stable and has been used in +many large deployments. The core API is unlikely to change much but please +version control your Logrus to make sure you aren't fetching latest `master` on +every build.** + +Nicely color-coded in development (when a TTY is attached, otherwise just +plain text): + +![Colored](http://i.imgur.com/PY7qMwd.png) + +With `log.Formatter = new(logrus.JSONFormatter)`, for easy parsing by logstash +or Splunk: + +```json +{"animal":"walrus","level":"info","msg":"A group of walrus emerges from the +ocean","size":10,"time":"2014-03-10 19:57:38.562264131 -0400 EDT"} + +{"level":"warning","msg":"The group's number increased tremendously!", +"number":122,"omg":true,"time":"2014-03-10 19:57:38.562471297 -0400 EDT"} + +{"animal":"walrus","level":"info","msg":"A giant walrus appears!", +"size":10,"time":"2014-03-10 19:57:38.562500591 -0400 EDT"} + +{"animal":"walrus","level":"info","msg":"Tremendously sized cow enters the ocean.", +"size":9,"time":"2014-03-10 19:57:38.562527896 -0400 EDT"} + +{"level":"fatal","msg":"The ice breaks!","number":100,"omg":true, +"time":"2014-03-10 19:57:38.562543128 -0400 EDT"} +``` + +With the default `log.Formatter = new(&log.TextFormatter{})` when a TTY is not +attached, the output is compatible with the +[logfmt](http://godoc.org/github.com/kr/logfmt) format: + +```text +time="2015-03-26T01:27:38-04:00" level=debug msg="Started observing beach" animal=walrus number=8 +time="2015-03-26T01:27:38-04:00" level=info msg="A group of walrus emerges from the ocean" animal=walrus size=10 +time="2015-03-26T01:27:38-04:00" level=warning msg="The group's number increased tremendously!" number=122 omg=true +time="2015-03-26T01:27:38-04:00" level=debug msg="Temperature changes" temperature=-4 +time="2015-03-26T01:27:38-04:00" level=panic msg="It's over 9000!" animal=orca size=9009 +time="2015-03-26T01:27:38-04:00" level=fatal msg="The ice breaks!" err=&{0x2082280c0 map[animal:orca size:9009] 2015-03-26 01:27:38.441574009 -0400 EDT panic It's over 9000!} number=100 omg=true +exit status 1 +``` + +#### Example + +The simplest way to use Logrus is simply the package-level exported logger: + +```go +package main + +import ( + log "github.com/Sirupsen/logrus" +) + +func main() { + log.WithFields(log.Fields{ + "animal": "walrus", + }).Info("A walrus appears") +} +``` + +Note that it's completely api-compatible with the stdlib logger, so you can +replace your `log` imports everywhere with `log "github.com/Sirupsen/logrus"` +and you'll now have the flexibility of Logrus. You can customize it all you +want: + +```go +package main + +import ( + "os" + log "github.com/Sirupsen/logrus" + "github.com/Sirupsen/logrus/hooks/airbrake" +) + +func init() { + // Log as JSON instead of the default ASCII formatter. + log.SetFormatter(&log.JSONFormatter{}) + + // Use the Airbrake hook to report errors that have Error severity or above to + // an exception tracker. You can create custom hooks, see the Hooks section. + log.AddHook(airbrake.NewHook("https://example.com", "xyz", "development")) + + // Output to stderr instead of stdout, could also be a file. + log.SetOutput(os.Stderr) + + // Only log the warning severity or above. + log.SetLevel(log.WarnLevel) +} + +func main() { + log.WithFields(log.Fields{ + "animal": "walrus", + "size": 10, + }).Info("A group of walrus emerges from the ocean") + + log.WithFields(log.Fields{ + "omg": true, + "number": 122, + }).Warn("The group's number increased tremendously!") + + log.WithFields(log.Fields{ + "omg": true, + "number": 100, + }).Fatal("The ice breaks!") + + // A common pattern is to re-use fields between logging statements by re-using + // the logrus.Entry returned from WithFields() + contextLogger := log.WithFields(log.Fields{ + "common": "this is a common field", + "other": "I also should be logged always", + }) + + contextLogger.Info("I'll be logged with common and other field") + contextLogger.Info("Me too") +} +``` + +For more advanced usage such as logging to multiple locations from the same +application, you can also create an instance of the `logrus` Logger: + +```go +package main + +import ( + "github.com/Sirupsen/logrus" +) + +// Create a new instance of the logger. You can have any number of instances. +var log = logrus.New() + +func main() { + // The API for setting attributes is a little different than the package level + // exported logger. See Godoc. + log.Out = os.Stderr + + log.WithFields(logrus.Fields{ + "animal": "walrus", + "size": 10, + }).Info("A group of walrus emerges from the ocean") +} +``` + +#### Fields + +Logrus encourages careful, structured logging though logging fields instead of +long, unparseable error messages. For example, instead of: `log.Fatalf("Failed +to send event %s to topic %s with key %d")`, you should log the much more +discoverable: + +```go +log.WithFields(log.Fields{ + "event": event, + "topic": topic, + "key": key, +}).Fatal("Failed to send event") +``` + +We've found this API forces you to think about logging in a way that produces +much more useful logging messages. We've been in countless situations where just +a single added field to a log statement that was already there would've saved us +hours. The `WithFields` call is optional. + +In general, with Logrus using any of the `printf`-family functions should be +seen as a hint you should add a field, however, you can still use the +`printf`-family functions with Logrus. + +#### Hooks + +You can add hooks for logging levels. For example to send errors to an exception +tracking service on `Error`, `Fatal` and `Panic`, info to StatsD or log to +multiple places simultaneously, e.g. syslog. + +Logrus comes with [built-in hooks](hooks/). Add those, or your custom hook, in +`init`: + +```go +import ( + log "github.com/Sirupsen/logrus" + "github.com/Sirupsen/logrus/hooks/airbrake" + logrus_syslog "github.com/Sirupsen/logrus/hooks/syslog" + "log/syslog" +) + +func init() { + log.AddHook(airbrake.NewHook("https://example.com", "xyz", "development")) + + hook, err := logrus_syslog.NewSyslogHook("udp", "localhost:514", syslog.LOG_INFO, "") + if err != nil { + log.Error("Unable to connect to local syslog daemon") + } else { + log.AddHook(hook) + } +} +``` + + +| Hook | Description | +| ----- | ----------- | +| [Airbrake](https://github.com/Sirupsen/logrus/blob/master/hooks/airbrake/airbrake.go) | Send errors to an exception tracking service compatible with the Airbrake API. Uses [`airbrake-go`](https://github.com/tobi/airbrake-go) behind the scenes. | +| [Papertrail](https://github.com/Sirupsen/logrus/blob/master/hooks/papertrail/papertrail.go) | Send errors to the Papertrail hosted logging service via UDP. | +| [Syslog](https://github.com/Sirupsen/logrus/blob/master/hooks/syslog/syslog.go) | Send errors to remote syslog server. Uses standard library `log/syslog` behind the scenes. | +| [BugSnag](https://github.com/Sirupsen/logrus/blob/master/hooks/bugsnag/bugsnag.go) | Send errors to the Bugsnag exception tracking service. | +| [Sentry](https://github.com/Sirupsen/logrus/blob/master/hooks/sentry/sentry.go) | Send errors to the Sentry error logging and aggregation service. | +| [Hiprus](https://github.com/nubo/hiprus) | Send errors to a channel in hipchat. | +| [Logrusly](https://github.com/sebest/logrusly) | Send logs to [Loggly](https://www.loggly.com/) | +| [Slackrus](https://github.com/johntdyer/slackrus) | Hook for Slack chat. | +| [Journalhook](https://github.com/wercker/journalhook) | Hook for logging to `systemd-journald` | +| [Graylog](https://github.com/gemnasium/logrus-hooks/tree/master/graylog) | Hook for logging to [Graylog](http://graylog2.org/) | +| [Raygun](https://github.com/squirkle/logrus-raygun-hook) | Hook for logging to [Raygun.io](http://raygun.io/) | +| [LFShook](https://github.com/rifflock/lfshook) | Hook for logging to the local filesystem | +| [Honeybadger](https://github.com/agonzalezro/logrus_honeybadger) | Hook for sending exceptions to Honeybadger | +| [Mail](https://github.com/zbindenren/logrus_mail) | Hook for sending exceptions via mail | +| [Rollrus](https://github.com/heroku/rollrus) | Hook for sending errors to rollbar | + +#### Level logging + +Logrus has six logging levels: Debug, Info, Warning, Error, Fatal and Panic. + +```go +log.Debug("Useful debugging information.") +log.Info("Something noteworthy happened!") +log.Warn("You should probably take a look at this.") +log.Error("Something failed but I'm not quitting.") +// Calls os.Exit(1) after logging +log.Fatal("Bye.") +// Calls panic() after logging +log.Panic("I'm bailing.") +``` + +You can set the logging level on a `Logger`, then it will only log entries with +that severity or anything above it: + +```go +// Will log anything that is info or above (warn, error, fatal, panic). Default. +log.SetLevel(log.InfoLevel) +``` + +It may be useful to set `log.Level = logrus.DebugLevel` in a debug or verbose +environment if your application has that. + +#### Entries + +Besides the fields added with `WithField` or `WithFields` some fields are +automatically added to all logging events: + +1. `time`. The timestamp when the entry was created. +2. `msg`. The logging message passed to `{Info,Warn,Error,Fatal,Panic}` after + the `AddFields` call. E.g. `Failed to send event.` +3. `level`. The logging level. E.g. `info`. + +#### Environments + +Logrus has no notion of environment. + +If you wish for hooks and formatters to only be used in specific environments, +you should handle that yourself. For example, if your application has a global +variable `Environment`, which is a string representation of the environment you +could do: + +```go +import ( + log "github.com/Sirupsen/logrus" +) + +init() { + // do something here to set environment depending on an environment variable + // or command-line flag + if Environment == "production" { + log.SetFormatter(&logrus.JSONFormatter{}) + } else { + // The TextFormatter is default, you don't actually have to do this. + log.SetFormatter(&log.TextFormatter{}) + } +} +``` + +This configuration is how `logrus` was intended to be used, but JSON in +production is mostly only useful if you do log aggregation with tools like +Splunk or Logstash. + +#### Formatters + +The built-in logging formatters are: + +* `logrus.TextFormatter`. Logs the event in colors if stdout is a tty, otherwise + without colors. + * *Note:* to force colored output when there is no TTY, set the `ForceColors` + field to `true`. To force no colored output even if there is a TTY set the + `DisableColors` field to `true` +* `logrus.JSONFormatter`. Logs fields as JSON. +* `logrus_logstash.LogstashFormatter`. Logs fields as Logstash Events (http://logstash.net). + + ```go + logrus.SetFormatter(&logrus_logstash.LogstashFormatter{Type: “application_name"}) + ``` + +Third party logging formatters: + +* [`zalgo`](https://github.com/aybabtme/logzalgo): invoking the P͉̫o̳̼̊w̖͈̰͎e̬͔̭͂r͚̼̹̲ ̫͓͉̳͈ō̠͕͖̚f̝͍̠ ͕̲̞͖͑Z̖̫̤̫ͪa͉̬͈̗l͖͎g̳̥o̰̥̅!̣͔̲̻͊̄ ̙̘̦̹̦. + +You can define your formatter by implementing the `Formatter` interface, +requiring a `Format` method. `Format` takes an `*Entry`. `entry.Data` is a +`Fields` type (`map[string]interface{}`) with all your fields as well as the +default ones (see Entries section above): + +```go +type MyJSONFormatter struct { +} + +log.SetFormatter(new(MyJSONFormatter)) + +func (f *JSONFormatter) Format(entry *Entry) ([]byte, error) { + // Note this doesn't include Time, Level and Message which are available on + // the Entry. Consult `godoc` on information about those fields or read the + // source of the official loggers. + serialized, err := json.Marshal(entry.Data) + if err != nil { + return nil, fmt.Errorf("Failed to marshal fields to JSON, %v", err) + } + return append(serialized, '\n'), nil +} +``` + +#### Logger as an `io.Writer` + +Logrus can be transformed into an `io.Writer`. That writer is the end of an `io.Pipe` and it is your responsibility to close it. + +```go +w := logger.Writer() +defer w.Close() + +srv := http.Server{ + // create a stdlib log.Logger that writes to + // logrus.Logger. + ErrorLog: log.New(w, "", 0), +} +``` + +Each line written to that writer will be printed the usual way, using formatters +and hooks. The level for those entries is `info`. + +#### Rotation + +Log rotation is not provided with Logrus. Log rotation should be done by an +external program (like `logrotate(8)`) that can compress and delete old log +entries. It should not be a feature of the application-level logger. + + +[godoc]: https://godoc.org/github.com/Sirupsen/logrus diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go new file mode 100644 index 0000000000000..699ea035cce16 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry.go @@ -0,0 +1,254 @@ +package logrus + +import ( + "bytes" + "fmt" + "io" + "os" + "time" +) + +// An entry is the final or intermediate Logrus logging entry. It contains all +// the fields passed with WithField{,s}. It's finally logged when Debug, Info, +// Warn, Error, Fatal or Panic is called on it. These objects can be reused and +// passed around as much as you wish to avoid field duplication. +type Entry struct { + Logger *Logger + + // Contains all the fields set by the user. + Data Fields + + // Time at which the log entry was created + Time time.Time + + // Level the log entry was logged at: Debug, Info, Warn, Error, Fatal or Panic + Level Level + + // Message passed to Debug, Info, Warn, Error, Fatal or Panic + Message string +} + +func NewEntry(logger *Logger) *Entry { + return &Entry{ + Logger: logger, + // Default is three fields, give a little extra room + Data: make(Fields, 5), + } +} + +// Returns a reader for the entry, which is a proxy to the formatter. +func (entry *Entry) Reader() (*bytes.Buffer, error) { + serialized, err := entry.Logger.Formatter.Format(entry) + return bytes.NewBuffer(serialized), err +} + +// Returns the string representation from the reader and ultimately the +// formatter. +func (entry *Entry) String() (string, error) { + reader, err := entry.Reader() + if err != nil { + return "", err + } + + return reader.String(), err +} + +// Add a single field to the Entry. +func (entry *Entry) WithField(key string, value interface{}) *Entry { + return entry.WithFields(Fields{key: value}) +} + +// Add a map of fields to the Entry. +func (entry *Entry) WithFields(fields Fields) *Entry { + data := Fields{} + for k, v := range entry.Data { + data[k] = v + } + for k, v := range fields { + data[k] = v + } + return &Entry{Logger: entry.Logger, Data: data} +} + +func (entry *Entry) log(level Level, msg string) { + entry.Time = time.Now() + entry.Level = level + entry.Message = msg + + if err := entry.Logger.Hooks.Fire(level, entry); err != nil { + entry.Logger.mu.Lock() + fmt.Fprintf(os.Stderr, "Failed to fire hook: %v\n", err) + entry.Logger.mu.Unlock() + } + + reader, err := entry.Reader() + if err != nil { + entry.Logger.mu.Lock() + fmt.Fprintf(os.Stderr, "Failed to obtain reader, %v\n", err) + entry.Logger.mu.Unlock() + } + + entry.Logger.mu.Lock() + defer entry.Logger.mu.Unlock() + + _, err = io.Copy(entry.Logger.Out, reader) + if err != nil { + fmt.Fprintf(os.Stderr, "Failed to write to log, %v\n", err) + } + + // To avoid Entry#log() returning a value that only would make sense for + // panic() to use in Entry#Panic(), we avoid the allocation by checking + // directly here. + if level <= PanicLevel { + panic(entry) + } +} + +func (entry *Entry) Debug(args ...interface{}) { + if entry.Logger.Level >= DebugLevel { + entry.log(DebugLevel, fmt.Sprint(args...)) + } +} + +func (entry *Entry) Print(args ...interface{}) { + entry.Info(args...) +} + +func (entry *Entry) Info(args ...interface{}) { + if entry.Logger.Level >= InfoLevel { + entry.log(InfoLevel, fmt.Sprint(args...)) + } +} + +func (entry *Entry) Warn(args ...interface{}) { + if entry.Logger.Level >= WarnLevel { + entry.log(WarnLevel, fmt.Sprint(args...)) + } +} + +func (entry *Entry) Warning(args ...interface{}) { + entry.Warn(args...) +} + +func (entry *Entry) Error(args ...interface{}) { + if entry.Logger.Level >= ErrorLevel { + entry.log(ErrorLevel, fmt.Sprint(args...)) + } +} + +func (entry *Entry) Fatal(args ...interface{}) { + if entry.Logger.Level >= FatalLevel { + entry.log(FatalLevel, fmt.Sprint(args...)) + } + os.Exit(1) +} + +func (entry *Entry) Panic(args ...interface{}) { + if entry.Logger.Level >= PanicLevel { + entry.log(PanicLevel, fmt.Sprint(args...)) + } + panic(fmt.Sprint(args...)) +} + +// Entry Printf family functions + +func (entry *Entry) Debugf(format string, args ...interface{}) { + if entry.Logger.Level >= DebugLevel { + entry.Debug(fmt.Sprintf(format, args...)) + } +} + +func (entry *Entry) Infof(format string, args ...interface{}) { + if entry.Logger.Level >= InfoLevel { + entry.Info(fmt.Sprintf(format, args...)) + } +} + +func (entry *Entry) Printf(format string, args ...interface{}) { + entry.Infof(format, args...) +} + +func (entry *Entry) Warnf(format string, args ...interface{}) { + if entry.Logger.Level >= WarnLevel { + entry.Warn(fmt.Sprintf(format, args...)) + } +} + +func (entry *Entry) Warningf(format string, args ...interface{}) { + entry.Warnf(format, args...) +} + +func (entry *Entry) Errorf(format string, args ...interface{}) { + if entry.Logger.Level >= ErrorLevel { + entry.Error(fmt.Sprintf(format, args...)) + } +} + +func (entry *Entry) Fatalf(format string, args ...interface{}) { + if entry.Logger.Level >= FatalLevel { + entry.Fatal(fmt.Sprintf(format, args...)) + } + os.Exit(1) +} + +func (entry *Entry) Panicf(format string, args ...interface{}) { + if entry.Logger.Level >= PanicLevel { + entry.Panic(fmt.Sprintf(format, args...)) + } +} + +// Entry Println family functions + +func (entry *Entry) Debugln(args ...interface{}) { + if entry.Logger.Level >= DebugLevel { + entry.Debug(entry.sprintlnn(args...)) + } +} + +func (entry *Entry) Infoln(args ...interface{}) { + if entry.Logger.Level >= InfoLevel { + entry.Info(entry.sprintlnn(args...)) + } +} + +func (entry *Entry) Println(args ...interface{}) { + entry.Infoln(args...) +} + +func (entry *Entry) Warnln(args ...interface{}) { + if entry.Logger.Level >= WarnLevel { + entry.Warn(entry.sprintlnn(args...)) + } +} + +func (entry *Entry) Warningln(args ...interface{}) { + entry.Warnln(args...) +} + +func (entry *Entry) Errorln(args ...interface{}) { + if entry.Logger.Level >= ErrorLevel { + entry.Error(entry.sprintlnn(args...)) + } +} + +func (entry *Entry) Fatalln(args ...interface{}) { + if entry.Logger.Level >= FatalLevel { + entry.Fatal(entry.sprintlnn(args...)) + } + os.Exit(1) +} + +func (entry *Entry) Panicln(args ...interface{}) { + if entry.Logger.Level >= PanicLevel { + entry.Panic(entry.sprintlnn(args...)) + } +} + +// Sprintlnn => Sprint no newline. This is to get the behavior of how +// fmt.Sprintln where spaces are always added between operands, regardless of +// their type. Instead of vendoring the Sprintln implementation to spare a +// string allocation, we do the simplest thing. +func (entry *Entry) sprintlnn(args ...interface{}) string { + msg := fmt.Sprintln(args...) + return msg[:len(msg)-1] +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry_test.go new file mode 100644 index 0000000000000..cd90aa7dc6440 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/entry_test.go @@ -0,0 +1,53 @@ +package logrus + +import ( + "bytes" + "fmt" + "testing" + + "github.com/fsouza/go-dockerclient/external/github.com/stretchr/testify/assert" +) + +func TestEntryPanicln(t *testing.T) { + errBoom := fmt.Errorf("boom time") + + defer func() { + p := recover() + assert.NotNil(t, p) + + switch pVal := p.(type) { + case *Entry: + assert.Equal(t, "kaboom", pVal.Message) + assert.Equal(t, errBoom, pVal.Data["err"]) + default: + t.Fatalf("want type *Entry, got %T: %#v", pVal, pVal) + } + }() + + logger := New() + logger.Out = &bytes.Buffer{} + entry := NewEntry(logger) + entry.WithField("err", errBoom).Panicln("kaboom") +} + +func TestEntryPanicf(t *testing.T) { + errBoom := fmt.Errorf("boom again") + + defer func() { + p := recover() + assert.NotNil(t, p) + + switch pVal := p.(type) { + case *Entry: + assert.Equal(t, "kaboom true", pVal.Message) + assert.Equal(t, errBoom, pVal.Data["err"]) + default: + t.Fatalf("want type *Entry, got %T: %#v", pVal, pVal) + } + }() + + logger := New() + logger.Out = &bytes.Buffer{} + entry := NewEntry(logger) + entry.WithField("err", errBoom).Panicf("kaboom %v", true) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go new file mode 100644 index 0000000000000..a67e1b802d914 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/exported.go @@ -0,0 +1,188 @@ +package logrus + +import ( + "io" +) + +var ( + // std is the name of the standard logger in stdlib `log` + std = New() +) + +func StandardLogger() *Logger { + return std +} + +// SetOutput sets the standard logger output. +func SetOutput(out io.Writer) { + std.mu.Lock() + defer std.mu.Unlock() + std.Out = out +} + +// SetFormatter sets the standard logger formatter. +func SetFormatter(formatter Formatter) { + std.mu.Lock() + defer std.mu.Unlock() + std.Formatter = formatter +} + +// SetLevel sets the standard logger level. +func SetLevel(level Level) { + std.mu.Lock() + defer std.mu.Unlock() + std.Level = level +} + +// GetLevel returns the standard logger level. +func GetLevel() Level { + std.mu.Lock() + defer std.mu.Unlock() + return std.Level +} + +// AddHook adds a hook to the standard logger hooks. +func AddHook(hook Hook) { + std.mu.Lock() + defer std.mu.Unlock() + std.Hooks.Add(hook) +} + +// WithField creates an entry from the standard logger and adds a field to +// it. If you want multiple fields, use `WithFields`. +// +// Note that it doesn't log until you call Debug, Print, Info, Warn, Fatal +// or Panic on the Entry it returns. +func WithField(key string, value interface{}) *Entry { + return std.WithField(key, value) +} + +// WithFields creates an entry from the standard logger and adds multiple +// fields to it. This is simply a helper for `WithField`, invoking it +// once for each field. +// +// Note that it doesn't log until you call Debug, Print, Info, Warn, Fatal +// or Panic on the Entry it returns. +func WithFields(fields Fields) *Entry { + return std.WithFields(fields) +} + +// Debug logs a message at level Debug on the standard logger. +func Debug(args ...interface{}) { + std.Debug(args...) +} + +// Print logs a message at level Info on the standard logger. +func Print(args ...interface{}) { + std.Print(args...) +} + +// Info logs a message at level Info on the standard logger. +func Info(args ...interface{}) { + std.Info(args...) +} + +// Warn logs a message at level Warn on the standard logger. +func Warn(args ...interface{}) { + std.Warn(args...) +} + +// Warning logs a message at level Warn on the standard logger. +func Warning(args ...interface{}) { + std.Warning(args...) +} + +// Error logs a message at level Error on the standard logger. +func Error(args ...interface{}) { + std.Error(args...) +} + +// Panic logs a message at level Panic on the standard logger. +func Panic(args ...interface{}) { + std.Panic(args...) +} + +// Fatal logs a message at level Fatal on the standard logger. +func Fatal(args ...interface{}) { + std.Fatal(args...) +} + +// Debugf logs a message at level Debug on the standard logger. +func Debugf(format string, args ...interface{}) { + std.Debugf(format, args...) +} + +// Printf logs a message at level Info on the standard logger. +func Printf(format string, args ...interface{}) { + std.Printf(format, args...) +} + +// Infof logs a message at level Info on the standard logger. +func Infof(format string, args ...interface{}) { + std.Infof(format, args...) +} + +// Warnf logs a message at level Warn on the standard logger. +func Warnf(format string, args ...interface{}) { + std.Warnf(format, args...) +} + +// Warningf logs a message at level Warn on the standard logger. +func Warningf(format string, args ...interface{}) { + std.Warningf(format, args...) +} + +// Errorf logs a message at level Error on the standard logger. +func Errorf(format string, args ...interface{}) { + std.Errorf(format, args...) +} + +// Panicf logs a message at level Panic on the standard logger. +func Panicf(format string, args ...interface{}) { + std.Panicf(format, args...) +} + +// Fatalf logs a message at level Fatal on the standard logger. +func Fatalf(format string, args ...interface{}) { + std.Fatalf(format, args...) +} + +// Debugln logs a message at level Debug on the standard logger. +func Debugln(args ...interface{}) { + std.Debugln(args...) +} + +// Println logs a message at level Info on the standard logger. +func Println(args ...interface{}) { + std.Println(args...) +} + +// Infoln logs a message at level Info on the standard logger. +func Infoln(args ...interface{}) { + std.Infoln(args...) +} + +// Warnln logs a message at level Warn on the standard logger. +func Warnln(args ...interface{}) { + std.Warnln(args...) +} + +// Warningln logs a message at level Warn on the standard logger. +func Warningln(args ...interface{}) { + std.Warningln(args...) +} + +// Errorln logs a message at level Error on the standard logger. +func Errorln(args ...interface{}) { + std.Errorln(args...) +} + +// Panicln logs a message at level Panic on the standard logger. +func Panicln(args ...interface{}) { + std.Panicln(args...) +} + +// Fatalln logs a message at level Fatal on the standard logger. +func Fatalln(args ...interface{}) { + std.Fatalln(args...) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/formatter.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/formatter.go new file mode 100644 index 0000000000000..104d689f187eb --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/formatter.go @@ -0,0 +1,48 @@ +package logrus + +import "time" + +const DefaultTimestampFormat = time.RFC3339 + +// The Formatter interface is used to implement a custom Formatter. It takes an +// `Entry`. It exposes all the fields, including the default ones: +// +// * `entry.Data["msg"]`. The message passed from Info, Warn, Error .. +// * `entry.Data["time"]`. The timestamp. +// * `entry.Data["level"]. The level the entry was logged at. +// +// Any additional fields added with `WithField` or `WithFields` are also in +// `entry.Data`. Format is expected to return an array of bytes which are then +// logged to `logger.Out`. +type Formatter interface { + Format(*Entry) ([]byte, error) +} + +// This is to not silently overwrite `time`, `msg` and `level` fields when +// dumping it. If this code wasn't there doing: +// +// logrus.WithField("level", 1).Info("hello") +// +// Would just silently drop the user provided level. Instead with this code +// it'll logged as: +// +// {"level": "info", "fields.level": 1, "msg": "hello", "time": "..."} +// +// It's not exported because it's still using Data in an opinionated way. It's to +// avoid code duplication between the two default formatters. +func prefixFieldClashes(data Fields) { + _, ok := data["time"] + if ok { + data["fields.time"] = data["time"] + } + + _, ok = data["msg"] + if ok { + data["fields.msg"] = data["msg"] + } + + _, ok = data["level"] + if ok { + data["fields.level"] = data["level"] + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/formatter_bench_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/formatter_bench_test.go new file mode 100644 index 0000000000000..c6d290c77f09b --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/formatter_bench_test.go @@ -0,0 +1,98 @@ +package logrus + +import ( + "fmt" + "testing" + "time" +) + +// smallFields is a small size data set for benchmarking +var smallFields = Fields{ + "foo": "bar", + "baz": "qux", + "one": "two", + "three": "four", +} + +// largeFields is a large size data set for benchmarking +var largeFields = Fields{ + "foo": "bar", + "baz": "qux", + "one": "two", + "three": "four", + "five": "six", + "seven": "eight", + "nine": "ten", + "eleven": "twelve", + "thirteen": "fourteen", + "fifteen": "sixteen", + "seventeen": "eighteen", + "nineteen": "twenty", + "a": "b", + "c": "d", + "e": "f", + "g": "h", + "i": "j", + "k": "l", + "m": "n", + "o": "p", + "q": "r", + "s": "t", + "u": "v", + "w": "x", + "y": "z", + "this": "will", + "make": "thirty", + "entries": "yeah", +} + +var errorFields = Fields{ + "foo": fmt.Errorf("bar"), + "baz": fmt.Errorf("qux"), +} + +func BenchmarkErrorTextFormatter(b *testing.B) { + doBenchmark(b, &TextFormatter{DisableColors: true}, errorFields) +} + +func BenchmarkSmallTextFormatter(b *testing.B) { + doBenchmark(b, &TextFormatter{DisableColors: true}, smallFields) +} + +func BenchmarkLargeTextFormatter(b *testing.B) { + doBenchmark(b, &TextFormatter{DisableColors: true}, largeFields) +} + +func BenchmarkSmallColoredTextFormatter(b *testing.B) { + doBenchmark(b, &TextFormatter{ForceColors: true}, smallFields) +} + +func BenchmarkLargeColoredTextFormatter(b *testing.B) { + doBenchmark(b, &TextFormatter{ForceColors: true}, largeFields) +} + +func BenchmarkSmallJSONFormatter(b *testing.B) { + doBenchmark(b, &JSONFormatter{}, smallFields) +} + +func BenchmarkLargeJSONFormatter(b *testing.B) { + doBenchmark(b, &JSONFormatter{}, largeFields) +} + +func doBenchmark(b *testing.B, formatter Formatter, fields Fields) { + entry := &Entry{ + Time: time.Time{}, + Level: InfoLevel, + Message: "message", + Data: fields, + } + var d []byte + var err error + for i := 0; i < b.N; i++ { + d, err = formatter.Format(entry) + if err != nil { + b.Fatal(err) + } + b.SetBytes(int64(len(d))) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/hook_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/hook_test.go new file mode 100644 index 0000000000000..938b974956455 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/hook_test.go @@ -0,0 +1,122 @@ +package logrus + +import ( + "testing" + + "github.com/fsouza/go-dockerclient/external/github.com/stretchr/testify/assert" +) + +type TestHook struct { + Fired bool +} + +func (hook *TestHook) Fire(entry *Entry) error { + hook.Fired = true + return nil +} + +func (hook *TestHook) Levels() []Level { + return []Level{ + DebugLevel, + InfoLevel, + WarnLevel, + ErrorLevel, + FatalLevel, + PanicLevel, + } +} + +func TestHookFires(t *testing.T) { + hook := new(TestHook) + + LogAndAssertJSON(t, func(log *Logger) { + log.Hooks.Add(hook) + assert.Equal(t, hook.Fired, false) + + log.Print("test") + }, func(fields Fields) { + assert.Equal(t, hook.Fired, true) + }) +} + +type ModifyHook struct { +} + +func (hook *ModifyHook) Fire(entry *Entry) error { + entry.Data["wow"] = "whale" + return nil +} + +func (hook *ModifyHook) Levels() []Level { + return []Level{ + DebugLevel, + InfoLevel, + WarnLevel, + ErrorLevel, + FatalLevel, + PanicLevel, + } +} + +func TestHookCanModifyEntry(t *testing.T) { + hook := new(ModifyHook) + + LogAndAssertJSON(t, func(log *Logger) { + log.Hooks.Add(hook) + log.WithField("wow", "elephant").Print("test") + }, func(fields Fields) { + assert.Equal(t, fields["wow"], "whale") + }) +} + +func TestCanFireMultipleHooks(t *testing.T) { + hook1 := new(ModifyHook) + hook2 := new(TestHook) + + LogAndAssertJSON(t, func(log *Logger) { + log.Hooks.Add(hook1) + log.Hooks.Add(hook2) + + log.WithField("wow", "elephant").Print("test") + }, func(fields Fields) { + assert.Equal(t, fields["wow"], "whale") + assert.Equal(t, hook2.Fired, true) + }) +} + +type ErrorHook struct { + Fired bool +} + +func (hook *ErrorHook) Fire(entry *Entry) error { + hook.Fired = true + return nil +} + +func (hook *ErrorHook) Levels() []Level { + return []Level{ + ErrorLevel, + } +} + +func TestErrorHookShouldntFireOnInfo(t *testing.T) { + hook := new(ErrorHook) + + LogAndAssertJSON(t, func(log *Logger) { + log.Hooks.Add(hook) + log.Info("test") + }, func(fields Fields) { + assert.Equal(t, hook.Fired, false) + }) +} + +func TestErrorHookShouldFireOnError(t *testing.T) { + hook := new(ErrorHook) + + LogAndAssertJSON(t, func(log *Logger) { + log.Hooks.Add(hook) + log.Error("test") + }, func(fields Fields) { + assert.Equal(t, hook.Fired, true) + }) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/hooks.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/hooks.go new file mode 100644 index 0000000000000..3f151cdc39275 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/hooks.go @@ -0,0 +1,34 @@ +package logrus + +// A hook to be fired when logging on the logging levels returned from +// `Levels()` on your implementation of the interface. Note that this is not +// fired in a goroutine or a channel with workers, you should handle such +// functionality yourself if your call is non-blocking and you don't wish for +// the logging calls for levels returned from `Levels()` to block. +type Hook interface { + Levels() []Level + Fire(*Entry) error +} + +// Internal type for storing the hooks on a logger instance. +type LevelHooks map[Level][]Hook + +// Add a hook to an instance of logger. This is called with +// `log.Hooks.Add(new(MyHook))` where `MyHook` implements the `Hook` interface. +func (hooks LevelHooks) Add(hook Hook) { + for _, level := range hook.Levels() { + hooks[level] = append(hooks[level], hook) + } +} + +// Fire all the hooks for the passed level. Used by `entry.log` to fire +// appropriate hooks for a log entry. +func (hooks LevelHooks) Fire(level Level, entry *Entry) error { + for _, hook := range hooks[level] { + if err := hook.Fire(entry); err != nil { + return err + } + } + + return nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/json_formatter.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/json_formatter.go new file mode 100644 index 0000000000000..2ad6dc5cf4f04 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/json_formatter.go @@ -0,0 +1,41 @@ +package logrus + +import ( + "encoding/json" + "fmt" +) + +type JSONFormatter struct { + // TimestampFormat sets the format used for marshaling timestamps. + TimestampFormat string +} + +func (f *JSONFormatter) Format(entry *Entry) ([]byte, error) { + data := make(Fields, len(entry.Data)+3) + for k, v := range entry.Data { + switch v := v.(type) { + case error: + // Otherwise errors are ignored by `encoding/json` + // https://github.com/Sirupsen/logrus/issues/137 + data[k] = v.Error() + default: + data[k] = v + } + } + prefixFieldClashes(data) + + timestampFormat := f.TimestampFormat + if timestampFormat == "" { + timestampFormat = DefaultTimestampFormat + } + + data["time"] = entry.Time.Format(timestampFormat) + data["msg"] = entry.Message + data["level"] = entry.Level.String() + + serialized, err := json.Marshal(data) + if err != nil { + return nil, fmt.Errorf("Failed to marshal fields to JSON, %v", err) + } + return append(serialized, '\n'), nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/json_formatter_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/json_formatter_test.go new file mode 100644 index 0000000000000..1d70873254d7b --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/json_formatter_test.go @@ -0,0 +1,120 @@ +package logrus + +import ( + "encoding/json" + "errors" + + "testing" +) + +func TestErrorNotLost(t *testing.T) { + formatter := &JSONFormatter{} + + b, err := formatter.Format(WithField("error", errors.New("wild walrus"))) + if err != nil { + t.Fatal("Unable to format entry: ", err) + } + + entry := make(map[string]interface{}) + err = json.Unmarshal(b, &entry) + if err != nil { + t.Fatal("Unable to unmarshal formatted entry: ", err) + } + + if entry["error"] != "wild walrus" { + t.Fatal("Error field not set") + } +} + +func TestErrorNotLostOnFieldNotNamedError(t *testing.T) { + formatter := &JSONFormatter{} + + b, err := formatter.Format(WithField("omg", errors.New("wild walrus"))) + if err != nil { + t.Fatal("Unable to format entry: ", err) + } + + entry := make(map[string]interface{}) + err = json.Unmarshal(b, &entry) + if err != nil { + t.Fatal("Unable to unmarshal formatted entry: ", err) + } + + if entry["omg"] != "wild walrus" { + t.Fatal("Error field not set") + } +} + +func TestFieldClashWithTime(t *testing.T) { + formatter := &JSONFormatter{} + + b, err := formatter.Format(WithField("time", "right now!")) + if err != nil { + t.Fatal("Unable to format entry: ", err) + } + + entry := make(map[string]interface{}) + err = json.Unmarshal(b, &entry) + if err != nil { + t.Fatal("Unable to unmarshal formatted entry: ", err) + } + + if entry["fields.time"] != "right now!" { + t.Fatal("fields.time not set to original time field") + } + + if entry["time"] != "0001-01-01T00:00:00Z" { + t.Fatal("time field not set to current time, was: ", entry["time"]) + } +} + +func TestFieldClashWithMsg(t *testing.T) { + formatter := &JSONFormatter{} + + b, err := formatter.Format(WithField("msg", "something")) + if err != nil { + t.Fatal("Unable to format entry: ", err) + } + + entry := make(map[string]interface{}) + err = json.Unmarshal(b, &entry) + if err != nil { + t.Fatal("Unable to unmarshal formatted entry: ", err) + } + + if entry["fields.msg"] != "something" { + t.Fatal("fields.msg not set to original msg field") + } +} + +func TestFieldClashWithLevel(t *testing.T) { + formatter := &JSONFormatter{} + + b, err := formatter.Format(WithField("level", "something")) + if err != nil { + t.Fatal("Unable to format entry: ", err) + } + + entry := make(map[string]interface{}) + err = json.Unmarshal(b, &entry) + if err != nil { + t.Fatal("Unable to unmarshal formatted entry: ", err) + } + + if entry["fields.level"] != "something" { + t.Fatal("fields.level not set to original level field") + } +} + +func TestJSONEntryEndsWithNewline(t *testing.T) { + formatter := &JSONFormatter{} + + b, err := formatter.Format(WithField("level", "something")) + if err != nil { + t.Fatal("Unable to format entry: ", err) + } + + if b[len(b)-1] != '\n' { + t.Fatal("Expected JSON log entry to end with a newline") + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go new file mode 100644 index 0000000000000..e4974bfbe7832 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logger.go @@ -0,0 +1,206 @@ +package logrus + +import ( + "io" + "os" + "sync" +) + +type Logger struct { + // The logs are `io.Copy`'d to this in a mutex. It's common to set this to a + // file, or leave it default which is `os.Stdout`. You can also set this to + // something more adventorous, such as logging to Kafka. + Out io.Writer + // Hooks for the logger instance. These allow firing events based on logging + // levels and log entries. For example, to send errors to an error tracking + // service, log to StatsD or dump the core on fatal errors. + Hooks LevelHooks + // All log entries pass through the formatter before logged to Out. The + // included formatters are `TextFormatter` and `JSONFormatter` for which + // TextFormatter is the default. In development (when a TTY is attached) it + // logs with colors, but to a file it wouldn't. You can easily implement your + // own that implements the `Formatter` interface, see the `README` or included + // formatters for examples. + Formatter Formatter + // The logging level the logger should log at. This is typically (and defaults + // to) `logrus.Info`, which allows Info(), Warn(), Error() and Fatal() to be + // logged. `logrus.Debug` is useful in + Level Level + // Used to sync writing to the log. + mu sync.Mutex +} + +// Creates a new logger. Configuration should be set by changing `Formatter`, +// `Out` and `Hooks` directly on the default logger instance. You can also just +// instantiate your own: +// +// var log = &Logger{ +// Out: os.Stderr, +// Formatter: new(JSONFormatter), +// Hooks: make(LevelHooks), +// Level: logrus.DebugLevel, +// } +// +// It's recommended to make this a global instance called `log`. +func New() *Logger { + return &Logger{ + Out: os.Stderr, + Formatter: new(TextFormatter), + Hooks: make(LevelHooks), + Level: InfoLevel, + } +} + +// Adds a field to the log entry, note that you it doesn't log until you call +// Debug, Print, Info, Warn, Fatal or Panic. It only creates a log entry. +// Ff you want multiple fields, use `WithFields`. +func (logger *Logger) WithField(key string, value interface{}) *Entry { + return NewEntry(logger).WithField(key, value) +} + +// Adds a struct of fields to the log entry. All it does is call `WithField` for +// each `Field`. +func (logger *Logger) WithFields(fields Fields) *Entry { + return NewEntry(logger).WithFields(fields) +} + +func (logger *Logger) Debugf(format string, args ...interface{}) { + if logger.Level >= DebugLevel { + NewEntry(logger).Debugf(format, args...) + } +} + +func (logger *Logger) Infof(format string, args ...interface{}) { + if logger.Level >= InfoLevel { + NewEntry(logger).Infof(format, args...) + } +} + +func (logger *Logger) Printf(format string, args ...interface{}) { + NewEntry(logger).Printf(format, args...) +} + +func (logger *Logger) Warnf(format string, args ...interface{}) { + if logger.Level >= WarnLevel { + NewEntry(logger).Warnf(format, args...) + } +} + +func (logger *Logger) Warningf(format string, args ...interface{}) { + if logger.Level >= WarnLevel { + NewEntry(logger).Warnf(format, args...) + } +} + +func (logger *Logger) Errorf(format string, args ...interface{}) { + if logger.Level >= ErrorLevel { + NewEntry(logger).Errorf(format, args...) + } +} + +func (logger *Logger) Fatalf(format string, args ...interface{}) { + if logger.Level >= FatalLevel { + NewEntry(logger).Fatalf(format, args...) + } + os.Exit(1) +} + +func (logger *Logger) Panicf(format string, args ...interface{}) { + if logger.Level >= PanicLevel { + NewEntry(logger).Panicf(format, args...) + } +} + +func (logger *Logger) Debug(args ...interface{}) { + if logger.Level >= DebugLevel { + NewEntry(logger).Debug(args...) + } +} + +func (logger *Logger) Info(args ...interface{}) { + if logger.Level >= InfoLevel { + NewEntry(logger).Info(args...) + } +} + +func (logger *Logger) Print(args ...interface{}) { + NewEntry(logger).Info(args...) +} + +func (logger *Logger) Warn(args ...interface{}) { + if logger.Level >= WarnLevel { + NewEntry(logger).Warn(args...) + } +} + +func (logger *Logger) Warning(args ...interface{}) { + if logger.Level >= WarnLevel { + NewEntry(logger).Warn(args...) + } +} + +func (logger *Logger) Error(args ...interface{}) { + if logger.Level >= ErrorLevel { + NewEntry(logger).Error(args...) + } +} + +func (logger *Logger) Fatal(args ...interface{}) { + if logger.Level >= FatalLevel { + NewEntry(logger).Fatal(args...) + } + os.Exit(1) +} + +func (logger *Logger) Panic(args ...interface{}) { + if logger.Level >= PanicLevel { + NewEntry(logger).Panic(args...) + } +} + +func (logger *Logger) Debugln(args ...interface{}) { + if logger.Level >= DebugLevel { + NewEntry(logger).Debugln(args...) + } +} + +func (logger *Logger) Infoln(args ...interface{}) { + if logger.Level >= InfoLevel { + NewEntry(logger).Infoln(args...) + } +} + +func (logger *Logger) Println(args ...interface{}) { + NewEntry(logger).Println(args...) +} + +func (logger *Logger) Warnln(args ...interface{}) { + if logger.Level >= WarnLevel { + NewEntry(logger).Warnln(args...) + } +} + +func (logger *Logger) Warningln(args ...interface{}) { + if logger.Level >= WarnLevel { + NewEntry(logger).Warnln(args...) + } +} + +func (logger *Logger) Errorln(args ...interface{}) { + if logger.Level >= ErrorLevel { + NewEntry(logger).Errorln(args...) + } +} + +func (logger *Logger) Fatalln(args ...interface{}) { + if logger.Level >= FatalLevel { + NewEntry(logger).Fatalln(args...) + } + os.Exit(1) +} + +func (logger *Logger) Panicln(args ...interface{}) { + if logger.Level >= PanicLevel { + NewEntry(logger).Panicln(args...) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logrus.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logrus.go new file mode 100644 index 0000000000000..43ee12e90e7b4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logrus.go @@ -0,0 +1,94 @@ +package logrus + +import ( + "fmt" + "log" +) + +// Fields type, used to pass to `WithFields`. +type Fields map[string]interface{} + +// Level type +type Level uint8 + +// Convert the Level to a string. E.g. PanicLevel becomes "panic". +func (level Level) String() string { + switch level { + case DebugLevel: + return "debug" + case InfoLevel: + return "info" + case WarnLevel: + return "warning" + case ErrorLevel: + return "error" + case FatalLevel: + return "fatal" + case PanicLevel: + return "panic" + } + + return "unknown" +} + +// ParseLevel takes a string level and returns the Logrus log level constant. +func ParseLevel(lvl string) (Level, error) { + switch lvl { + case "panic": + return PanicLevel, nil + case "fatal": + return FatalLevel, nil + case "error": + return ErrorLevel, nil + case "warn", "warning": + return WarnLevel, nil + case "info": + return InfoLevel, nil + case "debug": + return DebugLevel, nil + } + + var l Level + return l, fmt.Errorf("not a valid logrus Level: %q", lvl) +} + +// These are the different logging levels. You can set the logging level to log +// on your instance of logger, obtained with `logrus.New()`. +const ( + // PanicLevel level, highest level of severity. Logs and then calls panic with the + // message passed to Debug, Info, ... + PanicLevel Level = iota + // FatalLevel level. Logs and then calls `os.Exit(1)`. It will exit even if the + // logging level is set to Panic. + FatalLevel + // ErrorLevel level. Logs. Used for errors that should definitely be noted. + // Commonly used for hooks to send errors to an error tracking service. + ErrorLevel + // WarnLevel level. Non-critical entries that deserve eyes. + WarnLevel + // InfoLevel level. General operational entries about what's going on inside the + // application. + InfoLevel + // DebugLevel level. Usually only enabled when debugging. Very verbose logging. + DebugLevel +) + +// Won't compile if StdLogger can't be realized by a log.Logger +var _ StdLogger = &log.Logger{} + +// StdLogger is what your logrus-enabled library should take, that way +// it'll accept a stdlib logger and a logrus logger. There's no standard +// interface, this is the closest we get, unfortunately. +type StdLogger interface { + Print(...interface{}) + Printf(string, ...interface{}) + Println(...interface{}) + + Fatal(...interface{}) + Fatalf(string, ...interface{}) + Fatalln(...interface{}) + + Panic(...interface{}) + Panicf(string, ...interface{}) + Panicln(...interface{}) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logrus_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logrus_test.go new file mode 100644 index 0000000000000..e8719b090db40 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/logrus_test.go @@ -0,0 +1,301 @@ +package logrus + +import ( + "bytes" + "encoding/json" + "strconv" + "strings" + "sync" + "testing" + + "github.com/fsouza/go-dockerclient/external/github.com/stretchr/testify/assert" +) + +func LogAndAssertJSON(t *testing.T, log func(*Logger), assertions func(fields Fields)) { + var buffer bytes.Buffer + var fields Fields + + logger := New() + logger.Out = &buffer + logger.Formatter = new(JSONFormatter) + + log(logger) + + err := json.Unmarshal(buffer.Bytes(), &fields) + assert.Nil(t, err) + + assertions(fields) +} + +func LogAndAssertText(t *testing.T, log func(*Logger), assertions func(fields map[string]string)) { + var buffer bytes.Buffer + + logger := New() + logger.Out = &buffer + logger.Formatter = &TextFormatter{ + DisableColors: true, + } + + log(logger) + + fields := make(map[string]string) + for _, kv := range strings.Split(buffer.String(), " ") { + if !strings.Contains(kv, "=") { + continue + } + kvArr := strings.Split(kv, "=") + key := strings.TrimSpace(kvArr[0]) + val := kvArr[1] + if kvArr[1][0] == '"' { + var err error + val, err = strconv.Unquote(val) + assert.NoError(t, err) + } + fields[key] = val + } + assertions(fields) +} + +func TestPrint(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Print("test") + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test") + assert.Equal(t, fields["level"], "info") + }) +} + +func TestInfo(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Info("test") + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test") + assert.Equal(t, fields["level"], "info") + }) +} + +func TestWarn(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Warn("test") + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test") + assert.Equal(t, fields["level"], "warning") + }) +} + +func TestInfolnShouldAddSpacesBetweenStrings(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Infoln("test", "test") + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test test") + }) +} + +func TestInfolnShouldAddSpacesBetweenStringAndNonstring(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Infoln("test", 10) + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test 10") + }) +} + +func TestInfolnShouldAddSpacesBetweenTwoNonStrings(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Infoln(10, 10) + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "10 10") + }) +} + +func TestInfoShouldAddSpacesBetweenTwoNonStrings(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Infoln(10, 10) + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "10 10") + }) +} + +func TestInfoShouldNotAddSpacesBetweenStringAndNonstring(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Info("test", 10) + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test10") + }) +} + +func TestInfoShouldNotAddSpacesBetweenStrings(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.Info("test", "test") + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "testtest") + }) +} + +func TestWithFieldsShouldAllowAssignments(t *testing.T) { + var buffer bytes.Buffer + var fields Fields + + logger := New() + logger.Out = &buffer + logger.Formatter = new(JSONFormatter) + + localLog := logger.WithFields(Fields{ + "key1": "value1", + }) + + localLog.WithField("key2", "value2").Info("test") + err := json.Unmarshal(buffer.Bytes(), &fields) + assert.Nil(t, err) + + assert.Equal(t, "value2", fields["key2"]) + assert.Equal(t, "value1", fields["key1"]) + + buffer = bytes.Buffer{} + fields = Fields{} + localLog.Info("test") + err = json.Unmarshal(buffer.Bytes(), &fields) + assert.Nil(t, err) + + _, ok := fields["key2"] + assert.Equal(t, false, ok) + assert.Equal(t, "value1", fields["key1"]) +} + +func TestUserSuppliedFieldDoesNotOverwriteDefaults(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.WithField("msg", "hello").Info("test") + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test") + }) +} + +func TestUserSuppliedMsgFieldHasPrefix(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.WithField("msg", "hello").Info("test") + }, func(fields Fields) { + assert.Equal(t, fields["msg"], "test") + assert.Equal(t, fields["fields.msg"], "hello") + }) +} + +func TestUserSuppliedTimeFieldHasPrefix(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.WithField("time", "hello").Info("test") + }, func(fields Fields) { + assert.Equal(t, fields["fields.time"], "hello") + }) +} + +func TestUserSuppliedLevelFieldHasPrefix(t *testing.T) { + LogAndAssertJSON(t, func(log *Logger) { + log.WithField("level", 1).Info("test") + }, func(fields Fields) { + assert.Equal(t, fields["level"], "info") + assert.Equal(t, fields["fields.level"], 1.0) // JSON has floats only + }) +} + +func TestDefaultFieldsAreNotPrefixed(t *testing.T) { + LogAndAssertText(t, func(log *Logger) { + ll := log.WithField("herp", "derp") + ll.Info("hello") + ll.Info("bye") + }, func(fields map[string]string) { + for _, fieldName := range []string{"fields.level", "fields.time", "fields.msg"} { + if _, ok := fields[fieldName]; ok { + t.Fatalf("should not have prefixed %q: %v", fieldName, fields) + } + } + }) +} + +func TestDoubleLoggingDoesntPrefixPreviousFields(t *testing.T) { + + var buffer bytes.Buffer + var fields Fields + + logger := New() + logger.Out = &buffer + logger.Formatter = new(JSONFormatter) + + llog := logger.WithField("context", "eating raw fish") + + llog.Info("looks delicious") + + err := json.Unmarshal(buffer.Bytes(), &fields) + assert.NoError(t, err, "should have decoded first message") + assert.Equal(t, len(fields), 4, "should only have msg/time/level/context fields") + assert.Equal(t, fields["msg"], "looks delicious") + assert.Equal(t, fields["context"], "eating raw fish") + + buffer.Reset() + + llog.Warn("omg it is!") + + err = json.Unmarshal(buffer.Bytes(), &fields) + assert.NoError(t, err, "should have decoded second message") + assert.Equal(t, len(fields), 4, "should only have msg/time/level/context fields") + assert.Equal(t, fields["msg"], "omg it is!") + assert.Equal(t, fields["context"], "eating raw fish") + assert.Nil(t, fields["fields.msg"], "should not have prefixed previous `msg` entry") + +} + +func TestConvertLevelToString(t *testing.T) { + assert.Equal(t, "debug", DebugLevel.String()) + assert.Equal(t, "info", InfoLevel.String()) + assert.Equal(t, "warning", WarnLevel.String()) + assert.Equal(t, "error", ErrorLevel.String()) + assert.Equal(t, "fatal", FatalLevel.String()) + assert.Equal(t, "panic", PanicLevel.String()) +} + +func TestParseLevel(t *testing.T) { + l, err := ParseLevel("panic") + assert.Nil(t, err) + assert.Equal(t, PanicLevel, l) + + l, err = ParseLevel("fatal") + assert.Nil(t, err) + assert.Equal(t, FatalLevel, l) + + l, err = ParseLevel("error") + assert.Nil(t, err) + assert.Equal(t, ErrorLevel, l) + + l, err = ParseLevel("warn") + assert.Nil(t, err) + assert.Equal(t, WarnLevel, l) + + l, err = ParseLevel("warning") + assert.Nil(t, err) + assert.Equal(t, WarnLevel, l) + + l, err = ParseLevel("info") + assert.Nil(t, err) + assert.Equal(t, InfoLevel, l) + + l, err = ParseLevel("debug") + assert.Nil(t, err) + assert.Equal(t, DebugLevel, l) + + l, err = ParseLevel("invalid") + assert.Equal(t, "not a valid logrus Level: \"invalid\"", err.Error()) +} + +func TestGetSetLevelRace(t *testing.T) { + wg := sync.WaitGroup{} + for i := 0; i < 100; i++ { + wg.Add(1) + go func(i int) { + defer wg.Done() + if i%2 == 0 { + SetLevel(InfoLevel) + } else { + GetLevel() + } + }(i) + + } + wg.Wait() +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_bsd.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_bsd.go new file mode 100644 index 0000000000000..71f8d67a55d77 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_bsd.go @@ -0,0 +1,9 @@ +// +build darwin freebsd openbsd netbsd dragonfly + +package logrus + +import "syscall" + +const ioctlReadTermios = syscall.TIOCGETA + +type Termios syscall.Termios diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_freebsd.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_freebsd.go new file mode 100644 index 0000000000000..0428ee5d52a6d --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_freebsd.go @@ -0,0 +1,20 @@ +/* + Go 1.2 doesn't include Termios for FreeBSD. This should be added in 1.3 and this could be merged with terminal_darwin. +*/ +package logrus + +import ( + "syscall" +) + +const ioctlReadTermios = syscall.TIOCGETA + +type Termios struct { + Iflag uint32 + Oflag uint32 + Cflag uint32 + Lflag uint32 + Cc [20]uint8 + Ispeed uint32 + Ospeed uint32 +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_linux.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_linux.go new file mode 100644 index 0000000000000..a2c0b40db612b --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_linux.go @@ -0,0 +1,12 @@ +// Based on ssh/terminal: +// Copyright 2013 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package logrus + +import "syscall" + +const ioctlReadTermios = syscall.TCGETS + +type Termios syscall.Termios diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_notwindows.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_notwindows.go new file mode 100644 index 0000000000000..b8bebc13eea65 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_notwindows.go @@ -0,0 +1,21 @@ +// Based on ssh/terminal: +// Copyright 2011 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build linux darwin freebsd openbsd + +package logrus + +import ( + "syscall" + "unsafe" +) + +// IsTerminal returns true if the given file descriptor is a terminal. +func IsTerminal() bool { + fd := syscall.Stdout + var termios Termios + _, _, err := syscall.Syscall6(syscall.SYS_IOCTL, uintptr(fd), ioctlReadTermios, uintptr(unsafe.Pointer(&termios)), 0, 0, 0) + return err == 0 +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_openbsd.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_openbsd.go new file mode 100644 index 0000000000000..af609a53d6492 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_openbsd.go @@ -0,0 +1,7 @@ +package logrus + +import "syscall" + +const ioctlReadTermios = syscall.TIOCGETA + +type Termios syscall.Termios diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_windows.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_windows.go new file mode 100644 index 0000000000000..2e09f6f7e31d2 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/terminal_windows.go @@ -0,0 +1,27 @@ +// Based on ssh/terminal: +// Copyright 2011 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// +build windows + +package logrus + +import ( + "syscall" + "unsafe" +) + +var kernel32 = syscall.NewLazyDLL("kernel32.dll") + +var ( + procGetConsoleMode = kernel32.NewProc("GetConsoleMode") +) + +// IsTerminal returns true if the given file descriptor is a terminal. +func IsTerminal() bool { + fd := syscall.Stdout + var st uint32 + r, _, e := syscall.Syscall(procGetConsoleMode.Addr(), 2, uintptr(fd), uintptr(unsafe.Pointer(&st)), 0) + return r != 0 && e == 0 +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/text_formatter.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/text_formatter.go new file mode 100644 index 0000000000000..2e6fe1bdd1862 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/text_formatter.go @@ -0,0 +1,158 @@ +package logrus + +import ( + "bytes" + "fmt" + "runtime" + "sort" + "strings" + "time" +) + +const ( + nocolor = 0 + red = 31 + green = 32 + yellow = 33 + blue = 34 + gray = 37 +) + +var ( + baseTimestamp time.Time + isTerminal bool +) + +func init() { + baseTimestamp = time.Now() + isTerminal = IsTerminal() +} + +func miniTS() int { + return int(time.Since(baseTimestamp) / time.Second) +} + +type TextFormatter struct { + // Set to true to bypass checking for a TTY before outputting colors. + ForceColors bool + + // Force disabling colors. + DisableColors bool + + // Disable timestamp logging. useful when output is redirected to logging + // system that already adds timestamps. + DisableTimestamp bool + + // Enable logging the full timestamp when a TTY is attached instead of just + // the time passed since beginning of execution. + FullTimestamp bool + + // TimestampFormat to use for display when a full timestamp is printed + TimestampFormat string + + // The fields are sorted by default for a consistent output. For applications + // that log extremely frequently and don't use the JSON formatter this may not + // be desired. + DisableSorting bool +} + +func (f *TextFormatter) Format(entry *Entry) ([]byte, error) { + var keys []string = make([]string, 0, len(entry.Data)) + for k := range entry.Data { + keys = append(keys, k) + } + + if !f.DisableSorting { + sort.Strings(keys) + } + + b := &bytes.Buffer{} + + prefixFieldClashes(entry.Data) + + isColorTerminal := isTerminal && (runtime.GOOS != "windows") + isColored := (f.ForceColors || isColorTerminal) && !f.DisableColors + + if f.TimestampFormat == "" { + f.TimestampFormat = DefaultTimestampFormat + } + if isColored { + f.printColored(b, entry, keys) + } else { + if !f.DisableTimestamp { + f.appendKeyValue(b, "time", entry.Time.Format(f.TimestampFormat)) + } + f.appendKeyValue(b, "level", entry.Level.String()) + f.appendKeyValue(b, "msg", entry.Message) + for _, key := range keys { + f.appendKeyValue(b, key, entry.Data[key]) + } + } + + b.WriteByte('\n') + return b.Bytes(), nil +} + +func (f *TextFormatter) printColored(b *bytes.Buffer, entry *Entry, keys []string) { + var levelColor int + switch entry.Level { + case DebugLevel: + levelColor = gray + case WarnLevel: + levelColor = yellow + case ErrorLevel, FatalLevel, PanicLevel: + levelColor = red + default: + levelColor = blue + } + + levelText := strings.ToUpper(entry.Level.String())[0:4] + + if !f.FullTimestamp { + fmt.Fprintf(b, "\x1b[%dm%s\x1b[0m[%04d] %-44s ", levelColor, levelText, miniTS(), entry.Message) + } else { + fmt.Fprintf(b, "\x1b[%dm%s\x1b[0m[%s] %-44s ", levelColor, levelText, entry.Time.Format(f.TimestampFormat), entry.Message) + } + for _, k := range keys { + v := entry.Data[k] + fmt.Fprintf(b, " \x1b[%dm%s\x1b[0m=%v", levelColor, k, v) + } +} + +func needsQuoting(text string) bool { + for _, ch := range text { + if !((ch >= 'a' && ch <= 'z') || + (ch >= 'A' && ch <= 'Z') || + (ch >= '0' && ch <= '9') || + ch == '-' || ch == '.') { + return false + } + } + return true +} + +func (f *TextFormatter) appendKeyValue(b *bytes.Buffer, key string, value interface{}) { + + b.WriteString(key) + b.WriteByte('=') + + switch value := value.(type) { + case string: + if needsQuoting(value) { + b.WriteString(value) + } else { + fmt.Fprintf(b, "%q", value) + } + case error: + errmsg := value.Error() + if needsQuoting(errmsg) { + b.WriteString(errmsg) + } else { + fmt.Fprintf(b, "%q", value) + } + default: + fmt.Fprint(b, value) + } + + b.WriteByte(' ') +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/text_formatter_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/text_formatter_test.go new file mode 100644 index 0000000000000..e25a44f67bfe3 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/text_formatter_test.go @@ -0,0 +1,61 @@ +package logrus + +import ( + "bytes" + "errors" + "testing" + "time" +) + +func TestQuoting(t *testing.T) { + tf := &TextFormatter{DisableColors: true} + + checkQuoting := func(q bool, value interface{}) { + b, _ := tf.Format(WithField("test", value)) + idx := bytes.Index(b, ([]byte)("test=")) + cont := bytes.Contains(b[idx+5:], []byte{'"'}) + if cont != q { + if q { + t.Errorf("quoting expected for: %#v", value) + } else { + t.Errorf("quoting not expected for: %#v", value) + } + } + } + + checkQuoting(false, "abcd") + checkQuoting(false, "v1.0") + checkQuoting(false, "1234567890") + checkQuoting(true, "/foobar") + checkQuoting(true, "x y") + checkQuoting(true, "x,y") + checkQuoting(false, errors.New("invalid")) + checkQuoting(true, errors.New("invalid argument")) +} + +func TestTimestampFormat(t *testing.T) { + checkTimeStr := func(format string) { + customFormatter := &TextFormatter{DisableColors: true, TimestampFormat: format} + customStr, _ := customFormatter.Format(WithField("test", "test")) + timeStart := bytes.Index(customStr, ([]byte)("time=")) + timeEnd := bytes.Index(customStr, ([]byte)("level=")) + timeStr := customStr[timeStart+5 : timeEnd-1] + if timeStr[0] == '"' && timeStr[len(timeStr)-1] == '"' { + timeStr = timeStr[1 : len(timeStr)-1] + } + if format == "" { + format = time.RFC3339 + } + _, e := time.Parse(format, (string)(timeStr)) + if e != nil { + t.Errorf("time string \"%s\" did not match provided time format \"%s\": %s", timeStr, format, e) + } + } + + checkTimeStr("2006-01-02T15:04:05.000000000Z07:00") + checkTimeStr("Mon Jan _2 15:04:05 2006") + checkTimeStr("") +} + +// TODO add tests for sorting etc., this requires a parser for the text +// formatter output. diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/writer.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/writer.go new file mode 100644 index 0000000000000..1e30b1c753a74 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus/writer.go @@ -0,0 +1,31 @@ +package logrus + +import ( + "bufio" + "io" + "runtime" +) + +func (logger *Logger) Writer() *io.PipeWriter { + reader, writer := io.Pipe() + + go logger.writerScanner(reader) + runtime.SetFinalizer(writer, writerFinalizer) + + return writer +} + +func (logger *Logger) writerScanner(reader *io.PipeReader) { + scanner := bufio.NewScanner(reader) + for scanner.Scan() { + logger.Print(scanner.Text()) + } + if err := scanner.Err(); err != nil { + logger.Errorf("Error while reading from Writer: %s", err) + } + reader.Close() +} + +func writerFinalizer(writer *io.PipeWriter) { + writer.Close() +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go new file mode 100644 index 0000000000000..b854227e86da0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile.go @@ -0,0 +1,62 @@ +package opts + +import ( + "bufio" + "fmt" + "os" + "regexp" + "strings" +) + +var ( + // EnvironmentVariableRegexp A regexp to validate correct environment variables + // Environment variables set by the user must have a name consisting solely of + // alphabetics, numerics, and underscores - the first of which must not be numeric. + EnvironmentVariableRegexp = regexp.MustCompile("^[[:alpha:]_][[:alpha:][:digit:]_]*$") +) + +// ParseEnvFile Read in a line delimited file with environment variables enumerated +func ParseEnvFile(filename string) ([]string, error) { + fh, err := os.Open(filename) + if err != nil { + return []string{}, err + } + defer fh.Close() + + lines := []string{} + scanner := bufio.NewScanner(fh) + for scanner.Scan() { + line := scanner.Text() + // line is not empty, and not starting with '#' + if len(line) > 0 && !strings.HasPrefix(line, "#") { + data := strings.SplitN(line, "=", 2) + + // trim the front of a variable, but nothing else + variable := strings.TrimLeft(data[0], whiteSpaces) + + if !EnvironmentVariableRegexp.MatchString(variable) { + return []string{}, ErrBadEnvVariable{fmt.Sprintf("variable '%s' is not a valid environment variable", variable)} + } + if len(data) > 1 { + + // pass the value through, no trimming + lines = append(lines, fmt.Sprintf("%s=%s", variable, data[1])) + } else { + // if only a pass-through variable is given, clean it up. + lines = append(lines, fmt.Sprintf("%s=%s", strings.TrimSpace(line), os.Getenv(line))) + } + } + } + return lines, scanner.Err() +} + +var whiteSpaces = " \t" + +// ErrBadEnvVariable typed error for bad environment variable +type ErrBadEnvVariable struct { + msg string +} + +func (e ErrBadEnvVariable) Error() string { + return fmt.Sprintf("poorly formatted environment: %s", e.msg) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile_test.go new file mode 100644 index 0000000000000..cd0ca8f325907 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/envfile_test.go @@ -0,0 +1,133 @@ +package opts + +import ( + "bufio" + "fmt" + "io/ioutil" + "os" + "reflect" + "strings" + "testing" +) + +func tmpFileWithContent(content string, t *testing.T) string { + tmpFile, err := ioutil.TempFile("", "envfile-test") + if err != nil { + t.Fatal(err) + } + defer tmpFile.Close() + + tmpFile.WriteString(content) + return tmpFile.Name() +} + +// Test ParseEnvFile for a file with a few well formatted lines +func TestParseEnvFileGoodFile(t *testing.T) { + content := `foo=bar + baz=quux +# comment + +_foobar=foobaz +` + + tmpFile := tmpFileWithContent(content, t) + defer os.Remove(tmpFile) + + lines, err := ParseEnvFile(tmpFile) + if err != nil { + t.Fatal(err) + } + + expectedLines := []string{ + "foo=bar", + "baz=quux", + "_foobar=foobaz", + } + + if !reflect.DeepEqual(lines, expectedLines) { + t.Fatal("lines not equal to expected_lines") + } +} + +// Test ParseEnvFile for an empty file +func TestParseEnvFileEmptyFile(t *testing.T) { + tmpFile := tmpFileWithContent("", t) + defer os.Remove(tmpFile) + + lines, err := ParseEnvFile(tmpFile) + if err != nil { + t.Fatal(err) + } + + if len(lines) != 0 { + t.Fatal("lines not empty; expected empty") + } +} + +// Test ParseEnvFile for a non existent file +func TestParseEnvFileNonExistentFile(t *testing.T) { + _, err := ParseEnvFile("foo_bar_baz") + if err == nil { + t.Fatal("ParseEnvFile succeeded; expected failure") + } + if _, ok := err.(*os.PathError); !ok { + t.Fatalf("Expected a PathError, got [%v]", err) + } +} + +// Test ParseEnvFile for a badly formatted file +func TestParseEnvFileBadlyFormattedFile(t *testing.T) { + content := `foo=bar + f =quux +` + + tmpFile := tmpFileWithContent(content, t) + defer os.Remove(tmpFile) + + _, err := ParseEnvFile(tmpFile) + if err == nil { + t.Fatalf("Expected a ErrBadEnvVariable, got nothing") + } + if _, ok := err.(ErrBadEnvVariable); !ok { + t.Fatalf("Expected a ErrBadEnvVariable, got [%v]", err) + } + expectedMessage := "poorly formatted environment: variable 'f ' is not a valid environment variable" + if err.Error() != expectedMessage { + t.Fatalf("Expected [%v], got [%v]", expectedMessage, err.Error()) + } +} + +// Test ParseEnvFile for a file with a line exeeding bufio.MaxScanTokenSize +func TestParseEnvFileLineTooLongFile(t *testing.T) { + content := strings.Repeat("a", bufio.MaxScanTokenSize+42) + content = fmt.Sprint("foo=", content) + + tmpFile := tmpFileWithContent(content, t) + defer os.Remove(tmpFile) + + _, err := ParseEnvFile(tmpFile) + if err == nil { + t.Fatal("ParseEnvFile succeeded; expected failure") + } +} + +// ParseEnvFile with a random file, pass through +func TestParseEnvFileRandomFile(t *testing.T) { + content := `first line +another invalid line` + tmpFile := tmpFileWithContent(content, t) + defer os.Remove(tmpFile) + + _, err := ParseEnvFile(tmpFile) + + if err == nil { + t.Fatalf("Expected a ErrBadEnvVariable, got nothing") + } + if _, ok := err.(ErrBadEnvVariable); !ok { + t.Fatalf("Expected a ErrBadEnvvariable, got [%v]", err) + } + expectedMessage := "poorly formatted environment: variable 'first line' is not a valid environment variable" + if err.Error() != expectedMessage { + t.Fatalf("Expected [%v], got [%v]", expectedMessage, err.Error()) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/hosts_unix.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/hosts_unix.go new file mode 100644 index 0000000000000..a29335e605a72 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/hosts_unix.go @@ -0,0 +1,7 @@ +// +build !windows + +package opts + +import "fmt" + +var DefaultHost = fmt.Sprintf("unix://%s", DefaultUnixSocket) diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/hosts_windows.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/hosts_windows.go new file mode 100644 index 0000000000000..55eac2aaca446 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/hosts_windows.go @@ -0,0 +1,7 @@ +// +build windows + +package opts + +import "fmt" + +var DefaultHost = fmt.Sprintf("tcp://%s:%d", DefaultHTTPHost, DefaultHTTPPort) diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ip.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ip.go new file mode 100644 index 0000000000000..b1f9587555eb0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ip.go @@ -0,0 +1,35 @@ +package opts + +import ( + "fmt" + "net" +) + +// IpOpt type that hold an IP +type IpOpt struct { + *net.IP +} + +func NewIpOpt(ref *net.IP, defaultVal string) *IpOpt { + o := &IpOpt{ + IP: ref, + } + o.Set(defaultVal) + return o +} + +func (o *IpOpt) Set(val string) error { + ip := net.ParseIP(val) + if ip == nil { + return fmt.Errorf("%s is not an ip address", val) + } + *o.IP = ip + return nil +} + +func (o *IpOpt) String() string { + if *o.IP == nil { + return "" + } + return o.IP.String() +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ip_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ip_test.go new file mode 100644 index 0000000000000..b6b526a578b3a --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ip_test.go @@ -0,0 +1,54 @@ +package opts + +import ( + "net" + "testing" +) + +func TestIpOptString(t *testing.T) { + addresses := []string{"", "0.0.0.0"} + var ip net.IP + + for _, address := range addresses { + stringAddress := NewIpOpt(&ip, address).String() + if stringAddress != address { + t.Fatalf("IpOpt string should be `%s`, not `%s`", address, stringAddress) + } + } +} + +func TestNewIpOptInvalidDefaultVal(t *testing.T) { + ip := net.IPv4(127, 0, 0, 1) + defaultVal := "Not an ip" + + ipOpt := NewIpOpt(&ip, defaultVal) + + expected := "127.0.0.1" + if ipOpt.String() != expected { + t.Fatalf("Expected [%v], got [%v]", expected, ipOpt.String()) + } +} + +func TestNewIpOptValidDefaultVal(t *testing.T) { + ip := net.IPv4(127, 0, 0, 1) + defaultVal := "192.168.1.1" + + ipOpt := NewIpOpt(&ip, defaultVal) + + expected := "192.168.1.1" + if ipOpt.String() != expected { + t.Fatalf("Expected [%v], got [%v]", expected, ipOpt.String()) + } +} + +func TestIpOptSetInvalidVal(t *testing.T) { + ip := net.IPv4(127, 0, 0, 1) + ipOpt := &IpOpt{IP: &ip} + + invalidIp := "invalid ip" + expectedError := "invalid ip is not an ip address" + err := ipOpt.Set(invalidIp) + if err == nil || err.Error() != expectedError { + t.Fatalf("Expected an Error with [%v], got [%v]", expectedError, err.Error()) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go new file mode 100644 index 0000000000000..aa409b99effbb --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts.go @@ -0,0 +1,323 @@ +package opts + +import ( + "fmt" + "net" + "os" + "path" + "regexp" + "strings" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers" + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume" +) + +var ( + alphaRegexp = regexp.MustCompile(`[a-zA-Z]`) + domainRegexp = regexp.MustCompile(`^(:?(:?[a-zA-Z0-9]|(:?[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9]))(:?\.(:?[a-zA-Z0-9]|(:?[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9])))*)\.?\s*$`) + // DefaultHTTPHost Default HTTP Host used if only port is provided to -H flag e.g. docker -d -H tcp://:8080 + DefaultHTTPHost = "127.0.0.1" + // DefaultHTTPPort Default HTTP Port used if only the protocol is provided to -H flag e.g. docker -d -H tcp:// + // TODO Windows. DefaultHTTPPort is only used on Windows if a -H parameter + // is not supplied. A better longer term solution would be to use a named + // pipe as the default on the Windows daemon. + DefaultHTTPPort = 2375 // Default HTTP Port + // DefaultUnixSocket Path for the unix socket. + // Docker daemon by default always listens on the default unix socket + DefaultUnixSocket = "/var/run/docker.sock" +) + +// ListOpts type that hold a list of values and a validation function. +type ListOpts struct { + values *[]string + validator ValidatorFctType +} + +// NewListOpts Create a new ListOpts with the specified validator. +func NewListOpts(validator ValidatorFctType) ListOpts { + var values []string + return *NewListOptsRef(&values, validator) +} + +func NewListOptsRef(values *[]string, validator ValidatorFctType) *ListOpts { + return &ListOpts{ + values: values, + validator: validator, + } +} + +func (opts *ListOpts) String() string { + return fmt.Sprintf("%v", []string((*opts.values))) +} + +// Set validates if needed the input value and add it to the +// internal slice. +func (opts *ListOpts) Set(value string) error { + if opts.validator != nil { + v, err := opts.validator(value) + if err != nil { + return err + } + value = v + } + (*opts.values) = append((*opts.values), value) + return nil +} + +// Delete remove the given element from the slice. +func (opts *ListOpts) Delete(key string) { + for i, k := range *opts.values { + if k == key { + (*opts.values) = append((*opts.values)[:i], (*opts.values)[i+1:]...) + return + } + } +} + +// GetMap returns the content of values in a map in order to avoid +// duplicates. +// FIXME: can we remove this? +func (opts *ListOpts) GetMap() map[string]struct{} { + ret := make(map[string]struct{}) + for _, k := range *opts.values { + ret[k] = struct{}{} + } + return ret +} + +// GetAll returns the values' slice. +// FIXME: Can we remove this? +func (opts *ListOpts) GetAll() []string { + return (*opts.values) +} + +// Get checks the existence of the given key. +func (opts *ListOpts) Get(key string) bool { + for _, k := range *opts.values { + if k == key { + return true + } + } + return false +} + +// Len returns the amount of element in the slice. +func (opts *ListOpts) Len() int { + return len((*opts.values)) +} + +//MapOpts type that holds a map of values and a validation function. +type MapOpts struct { + values map[string]string + validator ValidatorFctType +} + +// Set validates if needed the input value and add it to the +// internal map, by splitting on '='. +func (opts *MapOpts) Set(value string) error { + if opts.validator != nil { + v, err := opts.validator(value) + if err != nil { + return err + } + value = v + } + vals := strings.SplitN(value, "=", 2) + if len(vals) == 1 { + (opts.values)[vals[0]] = "" + } else { + (opts.values)[vals[0]] = vals[1] + } + return nil +} + +func (opts *MapOpts) String() string { + return fmt.Sprintf("%v", map[string]string((opts.values))) +} + +func NewMapOpts(values map[string]string, validator ValidatorFctType) *MapOpts { + if values == nil { + values = make(map[string]string) + } + return &MapOpts{ + values: values, + validator: validator, + } +} + +// ValidatorFctType validator that return a validate string and/or an error +type ValidatorFctType func(val string) (string, error) + +// ValidatorFctListType validator that return a validate list of string and/or an error +type ValidatorFctListType func(val string) ([]string, error) + +// ValidateAttach Validates that the specified string is a valid attach option. +func ValidateAttach(val string) (string, error) { + s := strings.ToLower(val) + for _, str := range []string{"stdin", "stdout", "stderr"} { + if s == str { + return s, nil + } + } + return val, fmt.Errorf("valid streams are STDIN, STDOUT and STDERR") +} + +// ValidateLink Validates that the specified string has a valid link format (containerName:alias). +func ValidateLink(val string) (string, error) { + if _, _, err := parsers.ParseLink(val); err != nil { + return val, err + } + return val, nil +} + +// ValidateDevice Validate a path for devices +// It will make sure 'val' is in the form: +// [host-dir:]container-path[:mode] +func ValidateDevice(val string) (string, error) { + return validatePath(val, false) +} + +// ValidatePath Validate a path for volumes +// It will make sure 'val' is in the form: +// [host-dir:]container-path[:rw|ro] +// It will also validate the mount mode. +func ValidatePath(val string) (string, error) { + return validatePath(val, true) +} + +func validatePath(val string, validateMountMode bool) (string, error) { + var containerPath string + var mode string + + if strings.Count(val, ":") > 2 { + return val, fmt.Errorf("bad format for volumes: %s", val) + } + + splited := strings.SplitN(val, ":", 3) + if splited[0] == "" { + return val, fmt.Errorf("bad format for volumes: %s", val) + } + switch len(splited) { + case 1: + containerPath = splited[0] + val = path.Clean(containerPath) + case 2: + if isValid, _ := volume.ValidateMountMode(splited[1]); validateMountMode && isValid { + containerPath = splited[0] + mode = splited[1] + val = fmt.Sprintf("%s:%s", path.Clean(containerPath), mode) + } else { + containerPath = splited[1] + val = fmt.Sprintf("%s:%s", splited[0], path.Clean(containerPath)) + } + case 3: + containerPath = splited[1] + mode = splited[2] + if isValid, _ := volume.ValidateMountMode(splited[2]); validateMountMode && !isValid { + return val, fmt.Errorf("bad mount mode specified : %s", mode) + } + val = fmt.Sprintf("%s:%s:%s", splited[0], containerPath, mode) + } + + if !path.IsAbs(containerPath) { + return val, fmt.Errorf("%s is not an absolute path", containerPath) + } + return val, nil +} + +// ValidateEnv Validate an environment variable and returns it +// It will use EnvironmentVariableRegexp to ensure the name of the environment variable is valid. +// If no value is specified, it returns the current value using os.Getenv. +func ValidateEnv(val string) (string, error) { + arr := strings.Split(val, "=") + if len(arr) > 1 { + return val, nil + } + if !EnvironmentVariableRegexp.MatchString(arr[0]) { + return val, ErrBadEnvVariable{fmt.Sprintf("variable '%s' is not a valid environment variable", val)} + } + if !doesEnvExist(val) { + return val, nil + } + return fmt.Sprintf("%s=%s", val, os.Getenv(val)), nil +} + +// ValidateIPAddress Validates an Ip address +func ValidateIPAddress(val string) (string, error) { + var ip = net.ParseIP(strings.TrimSpace(val)) + if ip != nil { + return ip.String(), nil + } + return "", fmt.Errorf("%s is not an ip address", val) +} + +// ValidateMACAddress Validates a MAC address +func ValidateMACAddress(val string) (string, error) { + _, err := net.ParseMAC(strings.TrimSpace(val)) + if err != nil { + return "", err + } + return val, nil +} + +// ValidateDNSSearch Validates domain for resolvconf search configuration. +// A zero length domain is represented by . +func ValidateDNSSearch(val string) (string, error) { + if val = strings.Trim(val, " "); val == "." { + return val, nil + } + return validateDomain(val) +} + +func validateDomain(val string) (string, error) { + if alphaRegexp.FindString(val) == "" { + return "", fmt.Errorf("%s is not a valid domain", val) + } + ns := domainRegexp.FindSubmatch([]byte(val)) + if len(ns) > 0 && len(ns[1]) < 255 { + return string(ns[1]), nil + } + return "", fmt.Errorf("%s is not a valid domain", val) +} + +// ValidateExtraHost Validate that the given string is a valid extrahost and returns it +// ExtraHost are in the form of name:ip where the ip has to be a valid ip (ipv4 or ipv6) +func ValidateExtraHost(val string) (string, error) { + // allow for IPv6 addresses in extra hosts by only splitting on first ":" + arr := strings.SplitN(val, ":", 2) + if len(arr) != 2 || len(arr[0]) == 0 { + return "", fmt.Errorf("bad format for add-host: %q", val) + } + if _, err := ValidateIPAddress(arr[1]); err != nil { + return "", fmt.Errorf("invalid IP address in add-host: %q", arr[1]) + } + return val, nil +} + +// ValidateLabel Validate that the given string is a valid label, and returns it +// Labels are in the form on key=value +func ValidateLabel(val string) (string, error) { + if strings.Count(val, "=") < 1 { + return "", fmt.Errorf("bad attribute format: %s", val) + } + return val, nil +} + +// ValidateHost Validate that the given string is a valid host and returns it +func ValidateHost(val string) (string, error) { + host, err := parsers.ParseHost(DefaultHTTPHost, DefaultUnixSocket, val) + if err != nil { + return val, err + } + return host, nil +} + +func doesEnvExist(name string) bool { + for _, entry := range os.Environ() { + parts := strings.SplitN(entry, "=", 2) + if parts[0] == name { + return true + } + } + return false +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts_test.go new file mode 100644 index 0000000000000..f08df30be6338 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/opts_test.go @@ -0,0 +1,479 @@ +package opts + +import ( + "fmt" + "os" + "strings" + "testing" +) + +func TestValidateIPAddress(t *testing.T) { + if ret, err := ValidateIPAddress(`1.2.3.4`); err != nil || ret == "" { + t.Fatalf("ValidateIPAddress(`1.2.3.4`) got %s %s", ret, err) + } + + if ret, err := ValidateIPAddress(`127.0.0.1`); err != nil || ret == "" { + t.Fatalf("ValidateIPAddress(`127.0.0.1`) got %s %s", ret, err) + } + + if ret, err := ValidateIPAddress(`::1`); err != nil || ret == "" { + t.Fatalf("ValidateIPAddress(`::1`) got %s %s", ret, err) + } + + if ret, err := ValidateIPAddress(`127`); err == nil || ret != "" { + t.Fatalf("ValidateIPAddress(`127`) got %s %s", ret, err) + } + + if ret, err := ValidateIPAddress(`random invalid string`); err == nil || ret != "" { + t.Fatalf("ValidateIPAddress(`random invalid string`) got %s %s", ret, err) + } + +} + +func TestMapOpts(t *testing.T) { + tmpMap := make(map[string]string) + o := NewMapOpts(tmpMap, logOptsValidator) + o.Set("max-size=1") + if o.String() != "map[max-size:1]" { + t.Errorf("%s != [map[max-size:1]", o.String()) + } + + o.Set("max-file=2") + if len(tmpMap) != 2 { + t.Errorf("map length %d != 2", len(tmpMap)) + } + + if tmpMap["max-file"] != "2" { + t.Errorf("max-file = %s != 2", tmpMap["max-file"]) + } + + if tmpMap["max-size"] != "1" { + t.Errorf("max-size = %s != 1", tmpMap["max-size"]) + } + if o.Set("dummy-val=3") == nil { + t.Errorf("validator is not being called") + } +} + +func TestValidateMACAddress(t *testing.T) { + if _, err := ValidateMACAddress(`92:d0:c6:0a:29:33`); err != nil { + t.Fatalf("ValidateMACAddress(`92:d0:c6:0a:29:33`) got %s", err) + } + + if _, err := ValidateMACAddress(`92:d0:c6:0a:33`); err == nil { + t.Fatalf("ValidateMACAddress(`92:d0:c6:0a:33`) succeeded; expected failure on invalid MAC") + } + + if _, err := ValidateMACAddress(`random invalid string`); err == nil { + t.Fatalf("ValidateMACAddress(`random invalid string`) succeeded; expected failure on invalid MAC") + } +} + +func TestListOptsWithoutValidator(t *testing.T) { + o := NewListOpts(nil) + o.Set("foo") + if o.String() != "[foo]" { + t.Errorf("%s != [foo]", o.String()) + } + o.Set("bar") + if o.Len() != 2 { + t.Errorf("%d != 2", o.Len()) + } + o.Set("bar") + if o.Len() != 3 { + t.Errorf("%d != 3", o.Len()) + } + if !o.Get("bar") { + t.Error("o.Get(\"bar\") == false") + } + if o.Get("baz") { + t.Error("o.Get(\"baz\") == true") + } + o.Delete("foo") + if o.String() != "[bar bar]" { + t.Errorf("%s != [bar bar]", o.String()) + } + listOpts := o.GetAll() + if len(listOpts) != 2 || listOpts[0] != "bar" || listOpts[1] != "bar" { + t.Errorf("Expected [[bar bar]], got [%v]", listOpts) + } + mapListOpts := o.GetMap() + if len(mapListOpts) != 1 { + t.Errorf("Expected [map[bar:{}]], got [%v]", mapListOpts) + } + +} + +func TestListOptsWithValidator(t *testing.T) { + // Re-using logOptsvalidator (used by MapOpts) + o := NewListOpts(logOptsValidator) + o.Set("foo") + if o.String() != "[]" { + t.Errorf("%s != []", o.String()) + } + o.Set("foo=bar") + if o.String() != "[]" { + t.Errorf("%s != []", o.String()) + } + o.Set("max-file=2") + if o.Len() != 1 { + t.Errorf("%d != 1", o.Len()) + } + if !o.Get("max-file=2") { + t.Error("o.Get(\"max-file=2\") == false") + } + if o.Get("baz") { + t.Error("o.Get(\"baz\") == true") + } + o.Delete("max-file=2") + if o.String() != "[]" { + t.Errorf("%s != []", o.String()) + } +} + +func TestValidateDNSSearch(t *testing.T) { + valid := []string{ + `.`, + `a`, + `a.`, + `1.foo`, + `17.foo`, + `foo.bar`, + `foo.bar.baz`, + `foo.bar.`, + `foo.bar.baz`, + `foo1.bar2`, + `foo1.bar2.baz`, + `1foo.2bar.`, + `1foo.2bar.baz`, + `foo-1.bar-2`, + `foo-1.bar-2.baz`, + `foo-1.bar-2.`, + `foo-1.bar-2.baz`, + `1-foo.2-bar`, + `1-foo.2-bar.baz`, + `1-foo.2-bar.`, + `1-foo.2-bar.baz`, + } + + invalid := []string{ + ``, + ` `, + ` `, + `17`, + `17.`, + `.17`, + `17-.`, + `17-.foo`, + `.foo`, + `foo-.bar`, + `-foo.bar`, + `foo.bar-`, + `foo.bar-.baz`, + `foo.-bar`, + `foo.-bar.baz`, + `foo.bar.baz.this.should.fail.on.long.name.beause.it.is.longer.thanisshouldbethis.should.fail.on.long.name.beause.it.is.longer.thanisshouldbethis.should.fail.on.long.name.beause.it.is.longer.thanisshouldbethis.should.fail.on.long.name.beause.it.is.longer.thanisshouldbe`, + } + + for _, domain := range valid { + if ret, err := ValidateDNSSearch(domain); err != nil || ret == "" { + t.Fatalf("ValidateDNSSearch(`"+domain+"`) got %s %s", ret, err) + } + } + + for _, domain := range invalid { + if ret, err := ValidateDNSSearch(domain); err == nil || ret != "" { + t.Fatalf("ValidateDNSSearch(`"+domain+"`) got %s %s", ret, err) + } + } +} + +func TestValidateExtraHosts(t *testing.T) { + valid := []string{ + `myhost:192.168.0.1`, + `thathost:10.0.2.1`, + `anipv6host:2003:ab34:e::1`, + `ipv6local:::1`, + } + + invalid := map[string]string{ + `myhost:192.notanipaddress.1`: `invalid IP`, + `thathost-nosemicolon10.0.0.1`: `bad format`, + `anipv6host:::::1`: `invalid IP`, + `ipv6local:::0::`: `invalid IP`, + } + + for _, extrahost := range valid { + if _, err := ValidateExtraHost(extrahost); err != nil { + t.Fatalf("ValidateExtraHost(`"+extrahost+"`) should succeed: error %v", err) + } + } + + for extraHost, expectedError := range invalid { + if _, err := ValidateExtraHost(extraHost); err == nil { + t.Fatalf("ValidateExtraHost(`%q`) should have failed validation", extraHost) + } else { + if !strings.Contains(err.Error(), expectedError) { + t.Fatalf("ValidateExtraHost(`%q`) error should contain %q", extraHost, expectedError) + } + } + } +} + +func TestValidateAttach(t *testing.T) { + valid := []string{ + "stdin", + "stdout", + "stderr", + "STDIN", + "STDOUT", + "STDERR", + } + if _, err := ValidateAttach("invalid"); err == nil { + t.Fatalf("Expected error with [valid streams are STDIN, STDOUT and STDERR], got nothing") + } + + for _, attach := range valid { + value, err := ValidateAttach(attach) + if err != nil { + t.Fatal(err) + } + if value != strings.ToLower(attach) { + t.Fatalf("Expected [%v], got [%v]", attach, value) + } + } +} + +func TestValidateLink(t *testing.T) { + valid := []string{ + "name", + "dcdfbe62ecd0:alias", + "7a67485460b7642516a4ad82ecefe7f57d0c4916f530561b71a50a3f9c4e33da", + "angry_torvalds:linus", + } + invalid := map[string]string{ + "": "empty string specified for links", + "too:much:of:it": "bad format for links: too:much:of:it", + } + + for _, link := range valid { + if _, err := ValidateLink(link); err != nil { + t.Fatalf("ValidateLink(`%q`) should succeed: error %q", link, err) + } + } + + for link, expectedError := range invalid { + if _, err := ValidateLink(link); err == nil { + t.Fatalf("ValidateLink(`%q`) should have failed validation", link) + } else { + if !strings.Contains(err.Error(), expectedError) { + t.Fatalf("ValidateLink(`%q`) error should contain %q", link, expectedError) + } + } + } +} + +func TestValidatePath(t *testing.T) { + valid := []string{ + "/home", + "/home:/home", + "/home:/something/else", + "/with space", + "/home:/with space", + "relative:/absolute-path", + "hostPath:/containerPath:ro", + "/hostPath:/containerPath:rw", + "/rw:/ro", + "/path:rw", + "/path:ro", + "/rw:rw", + } + invalid := map[string]string{ + "": "bad format for volumes: ", + "./": "./ is not an absolute path", + "../": "../ is not an absolute path", + "/:../": "../ is not an absolute path", + "/:path": "path is not an absolute path", + ":": "bad format for volumes: :", + "/tmp:": " is not an absolute path", + ":test": "bad format for volumes: :test", + ":/test": "bad format for volumes: :/test", + "tmp:": " is not an absolute path", + ":test:": "bad format for volumes: :test:", + "::": "bad format for volumes: ::", + ":::": "bad format for volumes: :::", + "/tmp:::": "bad format for volumes: /tmp:::", + ":/tmp::": "bad format for volumes: :/tmp::", + "path:ro": "path is not an absolute path", + "/path:/path:sw": "bad mount mode specified : sw", + "/path:/path:rwz": "bad mount mode specified : rwz", + } + + for _, path := range valid { + if _, err := ValidatePath(path); err != nil { + t.Fatalf("ValidatePath(`%q`) should succeed: error %q", path, err) + } + } + + for path, expectedError := range invalid { + if _, err := ValidatePath(path); err == nil { + t.Fatalf("ValidatePath(`%q`) should have failed validation", path) + } else { + if err.Error() != expectedError { + t.Fatalf("ValidatePath(`%q`) error should contain %q, got %q", path, expectedError, err.Error()) + } + } + } +} +func TestValidateDevice(t *testing.T) { + valid := []string{ + "/home", + "/home:/home", + "/home:/something/else", + "/with space", + "/home:/with space", + "relative:/absolute-path", + "hostPath:/containerPath:ro", + "/hostPath:/containerPath:rw", + "/hostPath:/containerPath:mrw", + } + invalid := map[string]string{ + "": "bad format for volumes: ", + "./": "./ is not an absolute path", + "../": "../ is not an absolute path", + "/:../": "../ is not an absolute path", + "/:path": "path is not an absolute path", + ":": "bad format for volumes: :", + "/tmp:": " is not an absolute path", + ":test": "bad format for volumes: :test", + ":/test": "bad format for volumes: :/test", + "tmp:": " is not an absolute path", + ":test:": "bad format for volumes: :test:", + "::": "bad format for volumes: ::", + ":::": "bad format for volumes: :::", + "/tmp:::": "bad format for volumes: /tmp:::", + ":/tmp::": "bad format for volumes: :/tmp::", + "path:ro": "ro is not an absolute path", + } + + for _, path := range valid { + if _, err := ValidateDevice(path); err != nil { + t.Fatalf("ValidateDevice(`%q`) should succeed: error %q", path, err) + } + } + + for path, expectedError := range invalid { + if _, err := ValidateDevice(path); err == nil { + t.Fatalf("ValidateDevice(`%q`) should have failed validation", path) + } else { + if err.Error() != expectedError { + t.Fatalf("ValidateDevice(`%q`) error should contain %q, got %q", path, expectedError, err.Error()) + } + } + } +} + +func TestValidateEnv(t *testing.T) { + invalids := map[string]string{ + "some spaces": "poorly formatted environment: variable 'some spaces' is not a valid environment variable", + "asd!qwe": "poorly formatted environment: variable 'asd!qwe' is not a valid environment variable", + "1asd": "poorly formatted environment: variable '1asd' is not a valid environment variable", + "123": "poorly formatted environment: variable '123' is not a valid environment variable", + } + valids := map[string]string{ + "a": "a", + "something": "something", + "_=a": "_=a", + "env1=value1": "env1=value1", + "_env1=value1": "_env1=value1", + "env2=value2=value3": "env2=value2=value3", + "env3=abc!qwe": "env3=abc!qwe", + "env_4=value 4": "env_4=value 4", + "PATH": fmt.Sprintf("PATH=%v", os.Getenv("PATH")), + "PATH=something": "PATH=something", + } + for value, expectedError := range invalids { + _, err := ValidateEnv(value) + if err == nil { + t.Fatalf("Expected ErrBadEnvVariable, got nothing") + } + if _, ok := err.(ErrBadEnvVariable); !ok { + t.Fatalf("Expected ErrBadEnvVariable, got [%s]", err) + } + if err.Error() != expectedError { + t.Fatalf("Expected ErrBadEnvVariable with message [%s], got [%s]", expectedError, err.Error()) + } + } + for value, expected := range valids { + actual, err := ValidateEnv(value) + if err != nil { + t.Fatal(err) + } + if actual != expected { + t.Fatalf("Expected [%v], got [%v]", expected, actual) + } + } +} + +func TestValidateLabel(t *testing.T) { + if _, err := ValidateLabel("label"); err == nil || err.Error() != "bad attribute format: label" { + t.Fatalf("Expected an error [bad attribute format: label], go %v", err) + } + if actual, err := ValidateLabel("key1=value1"); err != nil || actual != "key1=value1" { + t.Fatalf("Expected [key1=value1], got [%v,%v]", actual, err) + } + // Validate it's working with more than one = + if actual, err := ValidateLabel("key1=value1=value2"); err != nil { + t.Fatalf("Expected [key1=value1=value2], got [%v,%v]", actual, err) + } + // Validate it's working with one more + if actual, err := ValidateLabel("key1=value1=value2=value3"); err != nil { + t.Fatalf("Expected [key1=value1=value2=value2], got [%v,%v]", actual, err) + } +} + +func TestValidateHost(t *testing.T) { + invalid := map[string]string{ + "anything": "Invalid bind address format: anything", + "something with spaces": "Invalid bind address format: something with spaces", + "://": "Invalid bind address format: ://", + "unknown://": "Invalid bind address format: unknown://", + "tcp://": "Invalid proto, expected tcp: ", + "tcp://:port": "Invalid bind address format: :port", + "tcp://invalid": "Invalid bind address format: invalid", + "tcp://invalid:port": "Invalid bind address format: invalid:port", + } + valid := map[string]string{ + "fd://": "fd://", + "fd://something": "fd://something", + "tcp://:2375": "tcp://127.0.0.1:2375", // default ip address + "tcp://:2376": "tcp://127.0.0.1:2376", // default ip address + "tcp://0.0.0.0:8080": "tcp://0.0.0.0:8080", + "tcp://192.168.0.0:12000": "tcp://192.168.0.0:12000", + "tcp://192.168:8080": "tcp://192.168:8080", + "tcp://0.0.0.0:1234567890": "tcp://0.0.0.0:1234567890", // yeah it's valid :P + "tcp://docker.com:2375": "tcp://docker.com:2375", + "unix://": "unix:///var/run/docker.sock", // default unix:// value + "unix://path/to/socket": "unix://path/to/socket", + } + + for value, errorMessage := range invalid { + if _, err := ValidateHost(value); err == nil || err.Error() != errorMessage { + t.Fatalf("Expected an error for %v with [%v], got [%v]", value, errorMessage, err) + } + } + for value, expected := range valid { + if actual, err := ValidateHost(value); err != nil || actual != expected { + t.Fatalf("Expected for %v [%v], got [%v, %v]", value, expected, actual, err) + } + } +} + +func logOptsValidator(val string) (string, error) { + allowedKeys := map[string]string{"max-size": "1", "max-file": "2"} + vals := strings.Split(val, "=") + if allowedKeys[vals[0]] != "" { + return val, nil + } + return "", fmt.Errorf("invalid key %s", vals[0]) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ulimit.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ulimit.go new file mode 100644 index 0000000000000..54f6c4e3f7d3d --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ulimit.go @@ -0,0 +1,47 @@ +package opts + +import ( + "fmt" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit" +) + +type UlimitOpt struct { + values *map[string]*ulimit.Ulimit +} + +func NewUlimitOpt(ref *map[string]*ulimit.Ulimit) *UlimitOpt { + if ref == nil { + ref = &map[string]*ulimit.Ulimit{} + } + return &UlimitOpt{ref} +} + +func (o *UlimitOpt) Set(val string) error { + l, err := ulimit.Parse(val) + if err != nil { + return err + } + + (*o.values)[l.Name] = l + + return nil +} + +func (o *UlimitOpt) String() string { + var out []string + for _, v := range *o.values { + out = append(out, v.String()) + } + + return fmt.Sprintf("%v", out) +} + +func (o *UlimitOpt) GetList() []*ulimit.Ulimit { + var ulimits []*ulimit.Ulimit + for _, v := range *o.values { + ulimits = append(ulimits, v) + } + + return ulimits +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ulimit_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ulimit_test.go new file mode 100644 index 0000000000000..ad284e7545b77 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/opts/ulimit_test.go @@ -0,0 +1,42 @@ +package opts + +import ( + "testing" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit" +) + +func TestUlimitOpt(t *testing.T) { + ulimitMap := map[string]*ulimit.Ulimit{ + "nofile": {"nofile", 1024, 512}, + } + + ulimitOpt := NewUlimitOpt(&ulimitMap) + + expected := "[nofile=512:1024]" + if ulimitOpt.String() != expected { + t.Fatalf("Expected %v, got %v", expected, ulimitOpt) + } + + // Valid ulimit append to opts + if err := ulimitOpt.Set("core=1024:1024"); err != nil { + t.Fatal(err) + } + + // Invalid ulimit type returns an error and do not append to opts + if err := ulimitOpt.Set("notavalidtype=1024:1024"); err == nil { + t.Fatalf("Expected error on invalid ulimit type") + } + expected = "[nofile=512:1024 core=1024:1024]" + expected2 := "[core=1024:1024 nofile=512:1024]" + result := ulimitOpt.String() + if result != expected && result != expected2 { + t.Fatalf("Expected %v or %v, got %v", expected, expected2, ulimitOpt) + } + + // And test GetList + ulimits := ulimitOpt.GetList() + if len(ulimits) != 2 { + t.Fatalf("Expected a ulimit list of 2, got %v", ulimits) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/README.md b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/README.md new file mode 100644 index 0000000000000..7307d9694f667 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/README.md @@ -0,0 +1 @@ +This code provides helper functions for dealing with archive files. diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive.go new file mode 100644 index 0000000000000..7306840b66b9f --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive.go @@ -0,0 +1,902 @@ +package archive + +import ( + "archive/tar" + "bufio" + "bytes" + "compress/bzip2" + "compress/gzip" + "errors" + "fmt" + "io" + "io/ioutil" + "os" + "os/exec" + "path/filepath" + "runtime" + "strings" + "syscall" + + "github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus" + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils" + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/pools" + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/promise" + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system" +) + +type ( + Archive io.ReadCloser + ArchiveReader io.Reader + Compression int + TarChownOptions struct { + UID, GID int + } + TarOptions struct { + IncludeFiles []string + ExcludePatterns []string + Compression Compression + NoLchown bool + ChownOpts *TarChownOptions + Name string + IncludeSourceDir bool + // When unpacking, specifies whether overwriting a directory with a + // non-directory is allowed and vice versa. + NoOverwriteDirNonDir bool + } + + // Archiver allows the reuse of most utility functions of this package + // with a pluggable Untar function. + Archiver struct { + Untar func(io.Reader, string, *TarOptions) error + } + + // breakoutError is used to differentiate errors related to breaking out + // When testing archive breakout in the unit tests, this error is expected + // in order for the test to pass. + breakoutError error +) + +var ( + ErrNotImplemented = errors.New("Function not implemented") + defaultArchiver = &Archiver{Untar} +) + +const ( + Uncompressed Compression = iota + Bzip2 + Gzip + Xz +) + +func IsArchive(header []byte) bool { + compression := DetectCompression(header) + if compression != Uncompressed { + return true + } + r := tar.NewReader(bytes.NewBuffer(header)) + _, err := r.Next() + return err == nil +} + +func DetectCompression(source []byte) Compression { + for compression, m := range map[Compression][]byte{ + Bzip2: {0x42, 0x5A, 0x68}, + Gzip: {0x1F, 0x8B, 0x08}, + Xz: {0xFD, 0x37, 0x7A, 0x58, 0x5A, 0x00}, + } { + if len(source) < len(m) { + logrus.Debugf("Len too short") + continue + } + if bytes.Compare(m, source[:len(m)]) == 0 { + return compression + } + } + return Uncompressed +} + +func xzDecompress(archive io.Reader) (io.ReadCloser, error) { + args := []string{"xz", "-d", "-c", "-q"} + + return CmdStream(exec.Command(args[0], args[1:]...), archive) +} + +func DecompressStream(archive io.Reader) (io.ReadCloser, error) { + p := pools.BufioReader32KPool + buf := p.Get(archive) + bs, err := buf.Peek(10) + if err != nil { + return nil, err + } + + compression := DetectCompression(bs) + switch compression { + case Uncompressed: + readBufWrapper := p.NewReadCloserWrapper(buf, buf) + return readBufWrapper, nil + case Gzip: + gzReader, err := gzip.NewReader(buf) + if err != nil { + return nil, err + } + readBufWrapper := p.NewReadCloserWrapper(buf, gzReader) + return readBufWrapper, nil + case Bzip2: + bz2Reader := bzip2.NewReader(buf) + readBufWrapper := p.NewReadCloserWrapper(buf, bz2Reader) + return readBufWrapper, nil + case Xz: + xzReader, err := xzDecompress(buf) + if err != nil { + return nil, err + } + readBufWrapper := p.NewReadCloserWrapper(buf, xzReader) + return readBufWrapper, nil + default: + return nil, fmt.Errorf("Unsupported compression format %s", (&compression).Extension()) + } +} + +func CompressStream(dest io.WriteCloser, compression Compression) (io.WriteCloser, error) { + p := pools.BufioWriter32KPool + buf := p.Get(dest) + switch compression { + case Uncompressed: + writeBufWrapper := p.NewWriteCloserWrapper(buf, buf) + return writeBufWrapper, nil + case Gzip: + gzWriter := gzip.NewWriter(dest) + writeBufWrapper := p.NewWriteCloserWrapper(buf, gzWriter) + return writeBufWrapper, nil + case Bzip2, Xz: + // archive/bzip2 does not support writing, and there is no xz support at all + // However, this is not a problem as docker only currently generates gzipped tars + return nil, fmt.Errorf("Unsupported compression format %s", (&compression).Extension()) + default: + return nil, fmt.Errorf("Unsupported compression format %s", (&compression).Extension()) + } +} + +func (compression *Compression) Extension() string { + switch *compression { + case Uncompressed: + return "tar" + case Bzip2: + return "tar.bz2" + case Gzip: + return "tar.gz" + case Xz: + return "tar.xz" + } + return "" +} + +type tarAppender struct { + TarWriter *tar.Writer + Buffer *bufio.Writer + + // for hardlink mapping + SeenFiles map[uint64]string +} + +// canonicalTarName provides a platform-independent and consistent posix-style +//path for files and directories to be archived regardless of the platform. +func canonicalTarName(name string, isDir bool) (string, error) { + name, err := CanonicalTarNameForPath(name) + if err != nil { + return "", err + } + + // suffix with '/' for directories + if isDir && !strings.HasSuffix(name, "/") { + name += "/" + } + return name, nil +} + +func (ta *tarAppender) addTarFile(path, name string) error { + fi, err := os.Lstat(path) + if err != nil { + return err + } + + link := "" + if fi.Mode()&os.ModeSymlink != 0 { + if link, err = os.Readlink(path); err != nil { + return err + } + } + + hdr, err := tar.FileInfoHeader(fi, link) + if err != nil { + return err + } + hdr.Mode = int64(chmodTarEntry(os.FileMode(hdr.Mode))) + + name, err = canonicalTarName(name, fi.IsDir()) + if err != nil { + return fmt.Errorf("tar: cannot canonicalize path: %v", err) + } + hdr.Name = name + + nlink, inode, err := setHeaderForSpecialDevice(hdr, ta, name, fi.Sys()) + if err != nil { + return err + } + + // if it's a regular file and has more than 1 link, + // it's hardlinked, so set the type flag accordingly + if fi.Mode().IsRegular() && nlink > 1 { + // a link should have a name that it links too + // and that linked name should be first in the tar archive + if oldpath, ok := ta.SeenFiles[inode]; ok { + hdr.Typeflag = tar.TypeLink + hdr.Linkname = oldpath + hdr.Size = 0 // This Must be here for the writer math to add up! + } else { + ta.SeenFiles[inode] = name + } + } + + capability, _ := system.Lgetxattr(path, "security.capability") + if capability != nil { + hdr.Xattrs = make(map[string]string) + hdr.Xattrs["security.capability"] = string(capability) + } + + if err := ta.TarWriter.WriteHeader(hdr); err != nil { + return err + } + + if hdr.Typeflag == tar.TypeReg { + file, err := os.Open(path) + if err != nil { + return err + } + + ta.Buffer.Reset(ta.TarWriter) + defer ta.Buffer.Reset(nil) + _, err = io.Copy(ta.Buffer, file) + file.Close() + if err != nil { + return err + } + err = ta.Buffer.Flush() + if err != nil { + return err + } + } + + return nil +} + +func createTarFile(path, extractDir string, hdr *tar.Header, reader io.Reader, Lchown bool, chownOpts *TarChownOptions) error { + // hdr.Mode is in linux format, which we can use for sycalls, + // but for os.Foo() calls we need the mode converted to os.FileMode, + // so use hdrInfo.Mode() (they differ for e.g. setuid bits) + hdrInfo := hdr.FileInfo() + + switch hdr.Typeflag { + case tar.TypeDir: + // Create directory unless it exists as a directory already. + // In that case we just want to merge the two + if fi, err := os.Lstat(path); !(err == nil && fi.IsDir()) { + if err := os.Mkdir(path, hdrInfo.Mode()); err != nil { + return err + } + } + + case tar.TypeReg, tar.TypeRegA: + // Source is regular file + file, err := os.OpenFile(path, os.O_CREATE|os.O_WRONLY, hdrInfo.Mode()) + if err != nil { + return err + } + if _, err := io.Copy(file, reader); err != nil { + file.Close() + return err + } + file.Close() + + case tar.TypeBlock, tar.TypeChar, tar.TypeFifo: + // Handle this is an OS-specific way + if err := handleTarTypeBlockCharFifo(hdr, path); err != nil { + return err + } + + case tar.TypeLink: + targetPath := filepath.Join(extractDir, hdr.Linkname) + // check for hardlink breakout + if !strings.HasPrefix(targetPath, extractDir) { + return breakoutError(fmt.Errorf("invalid hardlink %q -> %q", targetPath, hdr.Linkname)) + } + if err := os.Link(targetPath, path); err != nil { + return err + } + + case tar.TypeSymlink: + // path -> hdr.Linkname = targetPath + // e.g. /extractDir/path/to/symlink -> ../2/file = /extractDir/path/2/file + targetPath := filepath.Join(filepath.Dir(path), hdr.Linkname) + + // the reason we don't need to check symlinks in the path (with FollowSymlinkInScope) is because + // that symlink would first have to be created, which would be caught earlier, at this very check: + if !strings.HasPrefix(targetPath, extractDir) { + return breakoutError(fmt.Errorf("invalid symlink %q -> %q", path, hdr.Linkname)) + } + if err := os.Symlink(hdr.Linkname, path); err != nil { + return err + } + + case tar.TypeXGlobalHeader: + logrus.Debugf("PAX Global Extended Headers found and ignored") + return nil + + default: + return fmt.Errorf("Unhandled tar header type %d\n", hdr.Typeflag) + } + + // Lchown is not supported on Windows. + if Lchown && runtime.GOOS != "windows" { + if chownOpts == nil { + chownOpts = &TarChownOptions{UID: hdr.Uid, GID: hdr.Gid} + } + if err := os.Lchown(path, chownOpts.UID, chownOpts.GID); err != nil { + return err + } + } + + for key, value := range hdr.Xattrs { + if err := system.Lsetxattr(path, key, []byte(value), 0); err != nil { + return err + } + } + + // There is no LChmod, so ignore mode for symlink. Also, this + // must happen after chown, as that can modify the file mode + if err := handleLChmod(hdr, path, hdrInfo); err != nil { + return err + } + + ts := []syscall.Timespec{timeToTimespec(hdr.AccessTime), timeToTimespec(hdr.ModTime)} + // syscall.UtimesNano doesn't support a NOFOLLOW flag atm + if hdr.Typeflag == tar.TypeLink { + if fi, err := os.Lstat(hdr.Linkname); err == nil && (fi.Mode()&os.ModeSymlink == 0) { + if err := system.UtimesNano(path, ts); err != nil && err != system.ErrNotSupportedPlatform { + return err + } + } + } else if hdr.Typeflag != tar.TypeSymlink { + if err := system.UtimesNano(path, ts); err != nil && err != system.ErrNotSupportedPlatform { + return err + } + } else { + if err := system.LUtimesNano(path, ts); err != nil && err != system.ErrNotSupportedPlatform { + return err + } + } + return nil +} + +// Tar creates an archive from the directory at `path`, and returns it as a +// stream of bytes. +func Tar(path string, compression Compression) (io.ReadCloser, error) { + return TarWithOptions(path, &TarOptions{Compression: compression}) +} + +// TarWithOptions creates an archive from the directory at `path`, only including files whose relative +// paths are included in `options.IncludeFiles` (if non-nil) or not in `options.ExcludePatterns`. +func TarWithOptions(srcPath string, options *TarOptions) (io.ReadCloser, error) { + + patterns, patDirs, exceptions, err := fileutils.CleanPatterns(options.ExcludePatterns) + + if err != nil { + return nil, err + } + + pipeReader, pipeWriter := io.Pipe() + + compressWriter, err := CompressStream(pipeWriter, options.Compression) + if err != nil { + return nil, err + } + + go func() { + ta := &tarAppender{ + TarWriter: tar.NewWriter(compressWriter), + Buffer: pools.BufioWriter32KPool.Get(nil), + SeenFiles: make(map[uint64]string), + } + + defer func() { + // Make sure to check the error on Close. + if err := ta.TarWriter.Close(); err != nil { + logrus.Debugf("Can't close tar writer: %s", err) + } + if err := compressWriter.Close(); err != nil { + logrus.Debugf("Can't close compress writer: %s", err) + } + if err := pipeWriter.Close(); err != nil { + logrus.Debugf("Can't close pipe writer: %s", err) + } + }() + + // this buffer is needed for the duration of this piped stream + defer pools.BufioWriter32KPool.Put(ta.Buffer) + + // In general we log errors here but ignore them because + // during e.g. a diff operation the container can continue + // mutating the filesystem and we can see transient errors + // from this + + stat, err := os.Lstat(srcPath) + if err != nil { + return + } + + if !stat.IsDir() { + // We can't later join a non-dir with any includes because the + // 'walk' will error if "file/." is stat-ed and "file" is not a + // directory. So, we must split the source path and use the + // basename as the include. + if len(options.IncludeFiles) > 0 { + logrus.Warn("Tar: Can't archive a file with includes") + } + + dir, base := SplitPathDirEntry(srcPath) + srcPath = dir + options.IncludeFiles = []string{base} + } + + if len(options.IncludeFiles) == 0 { + options.IncludeFiles = []string{"."} + } + + seen := make(map[string]bool) + + var renamedRelFilePath string // For when tar.Options.Name is set + for _, include := range options.IncludeFiles { + // We can't use filepath.Join(srcPath, include) because this will + // clean away a trailing "." or "/" which may be important. + walkRoot := strings.Join([]string{srcPath, include}, string(filepath.Separator)) + filepath.Walk(walkRoot, func(filePath string, f os.FileInfo, err error) error { + if err != nil { + logrus.Debugf("Tar: Can't stat file %s to tar: %s", srcPath, err) + return nil + } + + relFilePath, err := filepath.Rel(srcPath, filePath) + if err != nil || (!options.IncludeSourceDir && relFilePath == "." && f.IsDir()) { + // Error getting relative path OR we are looking + // at the source directory path. Skip in both situations. + return nil + } + + if options.IncludeSourceDir && include == "." && relFilePath != "." { + relFilePath = strings.Join([]string{".", relFilePath}, string(filepath.Separator)) + } + + skip := false + + // If "include" is an exact match for the current file + // then even if there's an "excludePatterns" pattern that + // matches it, don't skip it. IOW, assume an explicit 'include' + // is asking for that file no matter what - which is true + // for some files, like .dockerignore and Dockerfile (sometimes) + if include != relFilePath { + skip, err = fileutils.OptimizedMatches(relFilePath, patterns, patDirs) + if err != nil { + logrus.Debugf("Error matching %s: %v", relFilePath, err) + return err + } + } + + if skip { + if !exceptions && f.IsDir() { + return filepath.SkipDir + } + return nil + } + + if seen[relFilePath] { + return nil + } + seen[relFilePath] = true + + // TODO Windows: Verify if this needs to be os.Pathseparator + // Rename the base resource + if options.Name != "" && filePath == srcPath+"/"+filepath.Base(relFilePath) { + renamedRelFilePath = relFilePath + } + // Set this to make sure the items underneath also get renamed + if options.Name != "" { + relFilePath = strings.Replace(relFilePath, renamedRelFilePath, options.Name, 1) + } + + if err := ta.addTarFile(filePath, relFilePath); err != nil { + logrus.Debugf("Can't add file %s to tar: %s", filePath, err) + } + return nil + }) + } + }() + + return pipeReader, nil +} + +func Unpack(decompressedArchive io.Reader, dest string, options *TarOptions) error { + tr := tar.NewReader(decompressedArchive) + trBuf := pools.BufioReader32KPool.Get(nil) + defer pools.BufioReader32KPool.Put(trBuf) + + var dirs []*tar.Header + + // Iterate through the files in the archive. +loop: + for { + hdr, err := tr.Next() + if err == io.EOF { + // end of tar archive + break + } + if err != nil { + return err + } + + // Normalize name, for safety and for a simple is-root check + // This keeps "../" as-is, but normalizes "/../" to "/". Or Windows: + // This keeps "..\" as-is, but normalizes "\..\" to "\". + hdr.Name = filepath.Clean(hdr.Name) + + for _, exclude := range options.ExcludePatterns { + if strings.HasPrefix(hdr.Name, exclude) { + continue loop + } + } + + // After calling filepath.Clean(hdr.Name) above, hdr.Name will now be in + // the filepath format for the OS on which the daemon is running. Hence + // the check for a slash-suffix MUST be done in an OS-agnostic way. + if !strings.HasSuffix(hdr.Name, string(os.PathSeparator)) { + // Not the root directory, ensure that the parent directory exists + parent := filepath.Dir(hdr.Name) + parentPath := filepath.Join(dest, parent) + if _, err := os.Lstat(parentPath); err != nil && os.IsNotExist(err) { + err = system.MkdirAll(parentPath, 0777) + if err != nil { + return err + } + } + } + + path := filepath.Join(dest, hdr.Name) + rel, err := filepath.Rel(dest, path) + if err != nil { + return err + } + if strings.HasPrefix(rel, ".."+string(os.PathSeparator)) { + return breakoutError(fmt.Errorf("%q is outside of %q", hdr.Name, dest)) + } + + // If path exits we almost always just want to remove and replace it + // The only exception is when it is a directory *and* the file from + // the layer is also a directory. Then we want to merge them (i.e. + // just apply the metadata from the layer). + if fi, err := os.Lstat(path); err == nil { + if options.NoOverwriteDirNonDir && fi.IsDir() && hdr.Typeflag != tar.TypeDir { + // If NoOverwriteDirNonDir is true then we cannot replace + // an existing directory with a non-directory from the archive. + return fmt.Errorf("cannot overwrite directory %q with non-directory %q", path, dest) + } + + if options.NoOverwriteDirNonDir && !fi.IsDir() && hdr.Typeflag == tar.TypeDir { + // If NoOverwriteDirNonDir is true then we cannot replace + // an existing non-directory with a directory from the archive. + return fmt.Errorf("cannot overwrite non-directory %q with directory %q", path, dest) + } + + if fi.IsDir() && hdr.Name == "." { + continue + } + + if !(fi.IsDir() && hdr.Typeflag == tar.TypeDir) { + if err := os.RemoveAll(path); err != nil { + return err + } + } + } + trBuf.Reset(tr) + + if err := createTarFile(path, dest, hdr, trBuf, !options.NoLchown, options.ChownOpts); err != nil { + return err + } + + // Directory mtimes must be handled at the end to avoid further + // file creation in them to modify the directory mtime + if hdr.Typeflag == tar.TypeDir { + dirs = append(dirs, hdr) + } + } + + for _, hdr := range dirs { + path := filepath.Join(dest, hdr.Name) + ts := []syscall.Timespec{timeToTimespec(hdr.AccessTime), timeToTimespec(hdr.ModTime)} + if err := syscall.UtimesNano(path, ts); err != nil { + return err + } + } + return nil +} + +// Untar reads a stream of bytes from `archive`, parses it as a tar archive, +// and unpacks it into the directory at `dest`. +// The archive may be compressed with one of the following algorithms: +// identity (uncompressed), gzip, bzip2, xz. +// FIXME: specify behavior when target path exists vs. doesn't exist. +func Untar(tarArchive io.Reader, dest string, options *TarOptions) error { + return untarHandler(tarArchive, dest, options, true) +} + +// Untar reads a stream of bytes from `archive`, parses it as a tar archive, +// and unpacks it into the directory at `dest`. +// The archive must be an uncompressed stream. +func UntarUncompressed(tarArchive io.Reader, dest string, options *TarOptions) error { + return untarHandler(tarArchive, dest, options, false) +} + +// Handler for teasing out the automatic decompression +func untarHandler(tarArchive io.Reader, dest string, options *TarOptions, decompress bool) error { + if tarArchive == nil { + return fmt.Errorf("Empty archive") + } + dest = filepath.Clean(dest) + if options == nil { + options = &TarOptions{} + } + if options.ExcludePatterns == nil { + options.ExcludePatterns = []string{} + } + + var r io.Reader = tarArchive + if decompress { + decompressedArchive, err := DecompressStream(tarArchive) + if err != nil { + return err + } + defer decompressedArchive.Close() + r = decompressedArchive + } + + return Unpack(r, dest, options) +} + +func (archiver *Archiver) TarUntar(src, dst string) error { + logrus.Debugf("TarUntar(%s %s)", src, dst) + archive, err := TarWithOptions(src, &TarOptions{Compression: Uncompressed}) + if err != nil { + return err + } + defer archive.Close() + return archiver.Untar(archive, dst, nil) +} + +// TarUntar is a convenience function which calls Tar and Untar, with the output of one piped into the other. +// If either Tar or Untar fails, TarUntar aborts and returns the error. +func TarUntar(src, dst string) error { + return defaultArchiver.TarUntar(src, dst) +} + +func (archiver *Archiver) UntarPath(src, dst string) error { + archive, err := os.Open(src) + if err != nil { + return err + } + defer archive.Close() + if err := archiver.Untar(archive, dst, nil); err != nil { + return err + } + return nil +} + +// UntarPath is a convenience function which looks for an archive +// at filesystem path `src`, and unpacks it at `dst`. +func UntarPath(src, dst string) error { + return defaultArchiver.UntarPath(src, dst) +} + +func (archiver *Archiver) CopyWithTar(src, dst string) error { + srcSt, err := os.Stat(src) + if err != nil { + return err + } + if !srcSt.IsDir() { + return archiver.CopyFileWithTar(src, dst) + } + // Create dst, copy src's content into it + logrus.Debugf("Creating dest directory: %s", dst) + if err := system.MkdirAll(dst, 0755); err != nil && !os.IsExist(err) { + return err + } + logrus.Debugf("Calling TarUntar(%s, %s)", src, dst) + return archiver.TarUntar(src, dst) +} + +// CopyWithTar creates a tar archive of filesystem path `src`, and +// unpacks it at filesystem path `dst`. +// The archive is streamed directly with fixed buffering and no +// intermediary disk IO. +func CopyWithTar(src, dst string) error { + return defaultArchiver.CopyWithTar(src, dst) +} + +func (archiver *Archiver) CopyFileWithTar(src, dst string) (err error) { + logrus.Debugf("CopyFileWithTar(%s, %s)", src, dst) + srcSt, err := os.Stat(src) + if err != nil { + return err + } + + if srcSt.IsDir() { + return fmt.Errorf("Can't copy a directory") + } + + // Clean up the trailing slash. This must be done in an operating + // system specific manner. + if dst[len(dst)-1] == os.PathSeparator { + dst = filepath.Join(dst, filepath.Base(src)) + } + // Create the holding directory if necessary + if err := system.MkdirAll(filepath.Dir(dst), 0700); err != nil && !os.IsExist(err) { + return err + } + + r, w := io.Pipe() + errC := promise.Go(func() error { + defer w.Close() + + srcF, err := os.Open(src) + if err != nil { + return err + } + defer srcF.Close() + + hdr, err := tar.FileInfoHeader(srcSt, "") + if err != nil { + return err + } + hdr.Name = filepath.Base(dst) + hdr.Mode = int64(chmodTarEntry(os.FileMode(hdr.Mode))) + + tw := tar.NewWriter(w) + defer tw.Close() + if err := tw.WriteHeader(hdr); err != nil { + return err + } + if _, err := io.Copy(tw, srcF); err != nil { + return err + } + return nil + }) + defer func() { + if er := <-errC; err != nil { + err = er + } + }() + return archiver.Untar(r, filepath.Dir(dst), nil) +} + +// CopyFileWithTar emulates the behavior of the 'cp' command-line +// for a single file. It copies a regular file from path `src` to +// path `dst`, and preserves all its metadata. +// +// Destination handling is in an operating specific manner depending +// where the daemon is running. If `dst` ends with a trailing slash +// the final destination path will be `dst/base(src)` (Linux) or +// `dst\base(src)` (Windows). +func CopyFileWithTar(src, dst string) (err error) { + return defaultArchiver.CopyFileWithTar(src, dst) +} + +// CmdStream executes a command, and returns its stdout as a stream. +// If the command fails to run or doesn't complete successfully, an error +// will be returned, including anything written on stderr. +func CmdStream(cmd *exec.Cmd, input io.Reader) (io.ReadCloser, error) { + if input != nil { + stdin, err := cmd.StdinPipe() + if err != nil { + return nil, err + } + // Write stdin if any + go func() { + io.Copy(stdin, input) + stdin.Close() + }() + } + stdout, err := cmd.StdoutPipe() + if err != nil { + return nil, err + } + stderr, err := cmd.StderrPipe() + if err != nil { + return nil, err + } + pipeR, pipeW := io.Pipe() + errChan := make(chan []byte) + // Collect stderr, we will use it in case of an error + go func() { + errText, e := ioutil.ReadAll(stderr) + if e != nil { + errText = []byte("(...couldn't fetch stderr: " + e.Error() + ")") + } + errChan <- errText + }() + // Copy stdout to the returned pipe + go func() { + _, err := io.Copy(pipeW, stdout) + if err != nil { + pipeW.CloseWithError(err) + } + errText := <-errChan + if err := cmd.Wait(); err != nil { + pipeW.CloseWithError(fmt.Errorf("%s: %s", err, errText)) + } else { + pipeW.Close() + } + }() + // Run the command and return the pipe + if err := cmd.Start(); err != nil { + return nil, err + } + return pipeR, nil +} + +// NewTempArchive reads the content of src into a temporary file, and returns the contents +// of that file as an archive. The archive can only be read once - as soon as reading completes, +// the file will be deleted. +func NewTempArchive(src Archive, dir string) (*TempArchive, error) { + f, err := ioutil.TempFile(dir, "") + if err != nil { + return nil, err + } + if _, err := io.Copy(f, src); err != nil { + return nil, err + } + if _, err := f.Seek(0, 0); err != nil { + return nil, err + } + st, err := f.Stat() + if err != nil { + return nil, err + } + size := st.Size() + return &TempArchive{File: f, Size: size}, nil +} + +type TempArchive struct { + *os.File + Size int64 // Pre-computed from Stat().Size() as a convenience + read int64 + closed bool +} + +// Close closes the underlying file if it's still open, or does a no-op +// to allow callers to try to close the TempArchive multiple times safely. +func (archive *TempArchive) Close() error { + if archive.closed { + return nil + } + + archive.closed = true + + return archive.File.Close() +} + +func (archive *TempArchive) Read(data []byte) (int, error) { + n, err := archive.File.Read(data) + archive.read += int64(n) + if err != nil || archive.read == archive.Size { + archive.Close() + os.Remove(archive.File.Name()) + } + return n, err +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_test.go new file mode 100644 index 0000000000000..4bb4f6ff5d1a8 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_test.go @@ -0,0 +1,1204 @@ +package archive + +import ( + "archive/tar" + "bytes" + "fmt" + "io" + "io/ioutil" + "os" + "os/exec" + "path" + "path/filepath" + "strings" + "syscall" + "testing" + "time" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system" +) + +func TestIsArchiveNilHeader(t *testing.T) { + out := IsArchive(nil) + if out { + t.Fatalf("isArchive should return false as nil is not a valid archive header") + } +} + +func TestIsArchiveInvalidHeader(t *testing.T) { + header := []byte{0x00, 0x01, 0x02} + out := IsArchive(header) + if out { + t.Fatalf("isArchive should return false as %s is not a valid archive header", header) + } +} + +func TestIsArchiveBzip2(t *testing.T) { + header := []byte{0x42, 0x5A, 0x68} + out := IsArchive(header) + if !out { + t.Fatalf("isArchive should return true as %s is a bz2 header", header) + } +} + +func TestIsArchive7zip(t *testing.T) { + header := []byte{0x50, 0x4b, 0x03, 0x04} + out := IsArchive(header) + if out { + t.Fatalf("isArchive should return false as %s is a 7z header and it is not supported", header) + } +} + +func TestDecompressStreamGzip(t *testing.T) { + cmd := exec.Command("/bin/sh", "-c", "touch /tmp/archive && gzip -f /tmp/archive") + output, err := cmd.CombinedOutput() + if err != nil { + t.Fatalf("Fail to create an archive file for test : %s.", output) + } + archive, err := os.Open("/tmp/archive.gz") + _, err = DecompressStream(archive) + if err != nil { + t.Fatalf("Failed to decompress a gzip file.") + } +} + +func TestDecompressStreamBzip2(t *testing.T) { + cmd := exec.Command("/bin/sh", "-c", "touch /tmp/archive && bzip2 -f /tmp/archive") + output, err := cmd.CombinedOutput() + if err != nil { + t.Fatalf("Fail to create an archive file for test : %s.", output) + } + archive, err := os.Open("/tmp/archive.bz2") + _, err = DecompressStream(archive) + if err != nil { + t.Fatalf("Failed to decompress a bzip2 file.") + } +} + +func TestDecompressStreamXz(t *testing.T) { + cmd := exec.Command("/bin/sh", "-c", "touch /tmp/archive && xz -f /tmp/archive") + output, err := cmd.CombinedOutput() + if err != nil { + t.Fatalf("Fail to create an archive file for test : %s.", output) + } + archive, err := os.Open("/tmp/archive.xz") + _, err = DecompressStream(archive) + if err != nil { + t.Fatalf("Failed to decompress a xz file.") + } +} + +func TestCompressStreamXzUnsuported(t *testing.T) { + dest, err := os.Create("/tmp/dest") + if err != nil { + t.Fatalf("Fail to create the destination file") + } + _, err = CompressStream(dest, Xz) + if err == nil { + t.Fatalf("Should fail as xz is unsupported for compression format.") + } +} + +func TestCompressStreamBzip2Unsupported(t *testing.T) { + dest, err := os.Create("/tmp/dest") + if err != nil { + t.Fatalf("Fail to create the destination file") + } + _, err = CompressStream(dest, Xz) + if err == nil { + t.Fatalf("Should fail as xz is unsupported for compression format.") + } +} + +func TestCompressStreamInvalid(t *testing.T) { + dest, err := os.Create("/tmp/dest") + if err != nil { + t.Fatalf("Fail to create the destination file") + } + _, err = CompressStream(dest, -1) + if err == nil { + t.Fatalf("Should fail as xz is unsupported for compression format.") + } +} + +func TestExtensionInvalid(t *testing.T) { + compression := Compression(-1) + output := compression.Extension() + if output != "" { + t.Fatalf("The extension of an invalid compression should be an empty string.") + } +} + +func TestExtensionUncompressed(t *testing.T) { + compression := Uncompressed + output := compression.Extension() + if output != "tar" { + t.Fatalf("The extension of a uncompressed archive should be 'tar'.") + } +} +func TestExtensionBzip2(t *testing.T) { + compression := Bzip2 + output := compression.Extension() + if output != "tar.bz2" { + t.Fatalf("The extension of a bzip2 archive should be 'tar.bz2'") + } +} +func TestExtensionGzip(t *testing.T) { + compression := Gzip + output := compression.Extension() + if output != "tar.gz" { + t.Fatalf("The extension of a bzip2 archive should be 'tar.gz'") + } +} +func TestExtensionXz(t *testing.T) { + compression := Xz + output := compression.Extension() + if output != "tar.xz" { + t.Fatalf("The extension of a bzip2 archive should be 'tar.xz'") + } +} + +func TestCmdStreamLargeStderr(t *testing.T) { + cmd := exec.Command("/bin/sh", "-c", "dd if=/dev/zero bs=1k count=1000 of=/dev/stderr; echo hello") + out, err := CmdStream(cmd, nil) + if err != nil { + t.Fatalf("Failed to start command: %s", err) + } + errCh := make(chan error) + go func() { + _, err := io.Copy(ioutil.Discard, out) + errCh <- err + }() + select { + case err := <-errCh: + if err != nil { + t.Fatalf("Command should not have failed (err=%.100s...)", err) + } + case <-time.After(5 * time.Second): + t.Fatalf("Command did not complete in 5 seconds; probable deadlock") + } +} + +func TestCmdStreamBad(t *testing.T) { + badCmd := exec.Command("/bin/sh", "-c", "echo hello; echo >&2 error couldn\\'t reverse the phase pulser; exit 1") + out, err := CmdStream(badCmd, nil) + if err != nil { + t.Fatalf("Failed to start command: %s", err) + } + if output, err := ioutil.ReadAll(out); err == nil { + t.Fatalf("Command should have failed") + } else if err.Error() != "exit status 1: error couldn't reverse the phase pulser\n" { + t.Fatalf("Wrong error value (%s)", err) + } else if s := string(output); s != "hello\n" { + t.Fatalf("Command output should be '%s', not '%s'", "hello\\n", output) + } +} + +func TestCmdStreamGood(t *testing.T) { + cmd := exec.Command("/bin/sh", "-c", "echo hello; exit 0") + out, err := CmdStream(cmd, nil) + if err != nil { + t.Fatal(err) + } + if output, err := ioutil.ReadAll(out); err != nil { + t.Fatalf("Command should not have failed (err=%s)", err) + } else if s := string(output); s != "hello\n" { + t.Fatalf("Command output should be '%s', not '%s'", "hello\\n", output) + } +} + +func TestUntarPathWithInvalidDest(t *testing.T) { + tempFolder, err := ioutil.TempDir("", "docker-archive-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(tempFolder) + invalidDestFolder := path.Join(tempFolder, "invalidDest") + // Create a src file + srcFile := path.Join(tempFolder, "src") + _, err = os.Create(srcFile) + if err != nil { + t.Fatalf("Fail to create the source file") + } + err = UntarPath(srcFile, invalidDestFolder) + if err == nil { + t.Fatalf("UntarPath with invalid destination path should throw an error.") + } +} + +func TestUntarPathWithInvalidSrc(t *testing.T) { + dest, err := ioutil.TempDir("", "docker-archive-test") + if err != nil { + t.Fatalf("Fail to create the destination file") + } + defer os.RemoveAll(dest) + err = UntarPath("/invalid/path", dest) + if err == nil { + t.Fatalf("UntarPath with invalid src path should throw an error.") + } +} + +func TestUntarPath(t *testing.T) { + tmpFolder, err := ioutil.TempDir("", "docker-archive-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(tmpFolder) + srcFile := path.Join(tmpFolder, "src") + tarFile := path.Join(tmpFolder, "src.tar") + os.Create(path.Join(tmpFolder, "src")) + cmd := exec.Command("/bin/sh", "-c", "tar cf "+tarFile+" "+srcFile) + _, err = cmd.CombinedOutput() + if err != nil { + t.Fatal(err) + } + destFolder := path.Join(tmpFolder, "dest") + err = os.MkdirAll(destFolder, 0740) + if err != nil { + t.Fatalf("Fail to create the destination file") + } + err = UntarPath(tarFile, destFolder) + if err != nil { + t.Fatalf("UntarPath shouldn't throw an error, %s.", err) + } + expectedFile := path.Join(destFolder, srcFile) + _, err = os.Stat(expectedFile) + if err != nil { + t.Fatalf("Destination folder should contain the source file but did not.") + } +} + +// Do the same test as above but with the destination as file, it should fail +func TestUntarPathWithDestinationFile(t *testing.T) { + tmpFolder, err := ioutil.TempDir("", "docker-archive-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(tmpFolder) + srcFile := path.Join(tmpFolder, "src") + tarFile := path.Join(tmpFolder, "src.tar") + os.Create(path.Join(tmpFolder, "src")) + cmd := exec.Command("/bin/sh", "-c", "tar cf "+tarFile+" "+srcFile) + _, err = cmd.CombinedOutput() + if err != nil { + t.Fatal(err) + } + destFile := path.Join(tmpFolder, "dest") + _, err = os.Create(destFile) + if err != nil { + t.Fatalf("Fail to create the destination file") + } + err = UntarPath(tarFile, destFile) + if err == nil { + t.Fatalf("UntarPath should throw an error if the destination if a file") + } +} + +// Do the same test as above but with the destination folder already exists +// and the destination file is a directory +// It's working, see https://github.com/docker/docker/issues/10040 +func TestUntarPathWithDestinationSrcFileAsFolder(t *testing.T) { + tmpFolder, err := ioutil.TempDir("", "docker-archive-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(tmpFolder) + srcFile := path.Join(tmpFolder, "src") + tarFile := path.Join(tmpFolder, "src.tar") + os.Create(srcFile) + cmd := exec.Command("/bin/sh", "-c", "tar cf "+tarFile+" "+srcFile) + _, err = cmd.CombinedOutput() + if err != nil { + t.Fatal(err) + } + destFolder := path.Join(tmpFolder, "dest") + err = os.MkdirAll(destFolder, 0740) + if err != nil { + t.Fatalf("Fail to create the destination folder") + } + // Let's create a folder that will has the same path as the extracted file (from tar) + destSrcFileAsFolder := path.Join(destFolder, srcFile) + err = os.MkdirAll(destSrcFileAsFolder, 0740) + if err != nil { + t.Fatal(err) + } + err = UntarPath(tarFile, destFolder) + if err != nil { + t.Fatalf("UntarPath should throw not throw an error if the extracted file already exists and is a folder") + } +} + +func TestCopyWithTarInvalidSrc(t *testing.T) { + tempFolder, err := ioutil.TempDir("", "docker-archive-test") + if err != nil { + t.Fatal(nil) + } + destFolder := path.Join(tempFolder, "dest") + invalidSrc := path.Join(tempFolder, "doesnotexists") + err = os.MkdirAll(destFolder, 0740) + if err != nil { + t.Fatal(err) + } + err = CopyWithTar(invalidSrc, destFolder) + if err == nil { + t.Fatalf("archiver.CopyWithTar with invalid src path should throw an error.") + } +} + +func TestCopyWithTarInexistentDestWillCreateIt(t *testing.T) { + tempFolder, err := ioutil.TempDir("", "docker-archive-test") + if err != nil { + t.Fatal(nil) + } + srcFolder := path.Join(tempFolder, "src") + inexistentDestFolder := path.Join(tempFolder, "doesnotexists") + err = os.MkdirAll(srcFolder, 0740) + if err != nil { + t.Fatal(err) + } + err = CopyWithTar(srcFolder, inexistentDestFolder) + if err != nil { + t.Fatalf("CopyWithTar with an inexistent folder shouldn't fail.") + } + _, err = os.Stat(inexistentDestFolder) + if err != nil { + t.Fatalf("CopyWithTar with an inexistent folder should create it.") + } +} + +// Test CopyWithTar with a file as src +func TestCopyWithTarSrcFile(t *testing.T) { + folder, err := ioutil.TempDir("", "docker-archive-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(folder) + dest := path.Join(folder, "dest") + srcFolder := path.Join(folder, "src") + src := path.Join(folder, path.Join("src", "src")) + err = os.MkdirAll(srcFolder, 0740) + if err != nil { + t.Fatal(err) + } + err = os.MkdirAll(dest, 0740) + if err != nil { + t.Fatal(err) + } + ioutil.WriteFile(src, []byte("content"), 0777) + err = CopyWithTar(src, dest) + if err != nil { + t.Fatalf("archiver.CopyWithTar shouldn't throw an error, %s.", err) + } + _, err = os.Stat(dest) + // FIXME Check the content + if err != nil { + t.Fatalf("Destination file should be the same as the source.") + } +} + +// Test CopyWithTar with a folder as src +func TestCopyWithTarSrcFolder(t *testing.T) { + folder, err := ioutil.TempDir("", "docker-archive-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(folder) + dest := path.Join(folder, "dest") + src := path.Join(folder, path.Join("src", "folder")) + err = os.MkdirAll(src, 0740) + if err != nil { + t.Fatal(err) + } + err = os.MkdirAll(dest, 0740) + if err != nil { + t.Fatal(err) + } + ioutil.WriteFile(path.Join(src, "file"), []byte("content"), 0777) + err = CopyWithTar(src, dest) + if err != nil { + t.Fatalf("archiver.CopyWithTar shouldn't throw an error, %s.", err) + } + _, err = os.Stat(dest) + // FIXME Check the content (the file inside) + if err != nil { + t.Fatalf("Destination folder should contain the source file but did not.") + } +} + +func TestCopyFileWithTarInvalidSrc(t *testing.T) { + tempFolder, err := ioutil.TempDir("", "docker-archive-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(tempFolder) + destFolder := path.Join(tempFolder, "dest") + err = os.MkdirAll(destFolder, 0740) + if err != nil { + t.Fatal(err) + } + invalidFile := path.Join(tempFolder, "doesnotexists") + err = CopyFileWithTar(invalidFile, destFolder) + if err == nil { + t.Fatalf("archiver.CopyWithTar with invalid src path should throw an error.") + } +} + +func TestCopyFileWithTarInexistentDestWillCreateIt(t *testing.T) { + tempFolder, err := ioutil.TempDir("", "docker-archive-test") + if err != nil { + t.Fatal(nil) + } + defer os.RemoveAll(tempFolder) + srcFile := path.Join(tempFolder, "src") + inexistentDestFolder := path.Join(tempFolder, "doesnotexists") + _, err = os.Create(srcFile) + if err != nil { + t.Fatal(err) + } + err = CopyFileWithTar(srcFile, inexistentDestFolder) + if err != nil { + t.Fatalf("CopyWithTar with an inexistent folder shouldn't fail.") + } + _, err = os.Stat(inexistentDestFolder) + if err != nil { + t.Fatalf("CopyWithTar with an inexistent folder should create it.") + } + // FIXME Test the src file and content +} + +func TestCopyFileWithTarSrcFolder(t *testing.T) { + folder, err := ioutil.TempDir("", "docker-archive-copyfilewithtar-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(folder) + dest := path.Join(folder, "dest") + src := path.Join(folder, "srcfolder") + err = os.MkdirAll(src, 0740) + if err != nil { + t.Fatal(err) + } + err = os.MkdirAll(dest, 0740) + if err != nil { + t.Fatal(err) + } + err = CopyFileWithTar(src, dest) + if err == nil { + t.Fatalf("CopyFileWithTar should throw an error with a folder.") + } +} + +func TestCopyFileWithTarSrcFile(t *testing.T) { + folder, err := ioutil.TempDir("", "docker-archive-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(folder) + dest := path.Join(folder, "dest") + srcFolder := path.Join(folder, "src") + src := path.Join(folder, path.Join("src", "src")) + err = os.MkdirAll(srcFolder, 0740) + if err != nil { + t.Fatal(err) + } + err = os.MkdirAll(dest, 0740) + if err != nil { + t.Fatal(err) + } + ioutil.WriteFile(src, []byte("content"), 0777) + err = CopyWithTar(src, dest+"/") + if err != nil { + t.Fatalf("archiver.CopyFileWithTar shouldn't throw an error, %s.", err) + } + _, err = os.Stat(dest) + if err != nil { + t.Fatalf("Destination folder should contain the source file but did not.") + } +} + +func TestTarFiles(t *testing.T) { + // try without hardlinks + if err := checkNoChanges(1000, false); err != nil { + t.Fatal(err) + } + // try with hardlinks + if err := checkNoChanges(1000, true); err != nil { + t.Fatal(err) + } +} + +func checkNoChanges(fileNum int, hardlinks bool) error { + srcDir, err := ioutil.TempDir("", "docker-test-srcDir") + if err != nil { + return err + } + defer os.RemoveAll(srcDir) + + destDir, err := ioutil.TempDir("", "docker-test-destDir") + if err != nil { + return err + } + defer os.RemoveAll(destDir) + + _, err = prepareUntarSourceDirectory(fileNum, srcDir, hardlinks) + if err != nil { + return err + } + + err = TarUntar(srcDir, destDir) + if err != nil { + return err + } + + changes, err := ChangesDirs(destDir, srcDir) + if err != nil { + return err + } + if len(changes) > 0 { + return fmt.Errorf("with %d files and %v hardlinks: expected 0 changes, got %d", fileNum, hardlinks, len(changes)) + } + return nil +} + +func tarUntar(t *testing.T, origin string, options *TarOptions) ([]Change, error) { + archive, err := TarWithOptions(origin, options) + if err != nil { + t.Fatal(err) + } + defer archive.Close() + + buf := make([]byte, 10) + if _, err := archive.Read(buf); err != nil { + return nil, err + } + wrap := io.MultiReader(bytes.NewReader(buf), archive) + + detectedCompression := DetectCompression(buf) + compression := options.Compression + if detectedCompression.Extension() != compression.Extension() { + return nil, fmt.Errorf("Wrong compression detected. Actual compression: %s, found %s", compression.Extension(), detectedCompression.Extension()) + } + + tmp, err := ioutil.TempDir("", "docker-test-untar") + if err != nil { + return nil, err + } + defer os.RemoveAll(tmp) + if err := Untar(wrap, tmp, nil); err != nil { + return nil, err + } + if _, err := os.Stat(tmp); err != nil { + return nil, err + } + + return ChangesDirs(origin, tmp) +} + +func TestTarUntar(t *testing.T) { + origin, err := ioutil.TempDir("", "docker-test-untar-origin") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(origin) + if err := ioutil.WriteFile(path.Join(origin, "1"), []byte("hello world"), 0700); err != nil { + t.Fatal(err) + } + if err := ioutil.WriteFile(path.Join(origin, "2"), []byte("welcome!"), 0700); err != nil { + t.Fatal(err) + } + if err := ioutil.WriteFile(path.Join(origin, "3"), []byte("will be ignored"), 0700); err != nil { + t.Fatal(err) + } + + for _, c := range []Compression{ + Uncompressed, + Gzip, + } { + changes, err := tarUntar(t, origin, &TarOptions{ + Compression: c, + ExcludePatterns: []string{"3"}, + }) + + if err != nil { + t.Fatalf("Error tar/untar for compression %s: %s", c.Extension(), err) + } + + if len(changes) != 1 || changes[0].Path != "/3" { + t.Fatalf("Unexpected differences after tarUntar: %v", changes) + } + } +} + +func TestTarUntarWithXattr(t *testing.T) { + origin, err := ioutil.TempDir("", "docker-test-untar-origin") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(origin) + if err := ioutil.WriteFile(path.Join(origin, "1"), []byte("hello world"), 0700); err != nil { + t.Fatal(err) + } + if err := ioutil.WriteFile(path.Join(origin, "2"), []byte("welcome!"), 0700); err != nil { + t.Fatal(err) + } + if err := ioutil.WriteFile(path.Join(origin, "3"), []byte("will be ignored"), 0700); err != nil { + t.Fatal(err) + } + if err := system.Lsetxattr(path.Join(origin, "2"), "security.capability", []byte{0x00}, 0); err != nil { + t.Fatal(err) + } + + for _, c := range []Compression{ + Uncompressed, + Gzip, + } { + changes, err := tarUntar(t, origin, &TarOptions{ + Compression: c, + ExcludePatterns: []string{"3"}, + }) + + if err != nil { + t.Fatalf("Error tar/untar for compression %s: %s", c.Extension(), err) + } + + if len(changes) != 1 || changes[0].Path != "/3" { + t.Fatalf("Unexpected differences after tarUntar: %v", changes) + } + capability, _ := system.Lgetxattr(path.Join(origin, "2"), "security.capability") + if capability == nil && capability[0] != 0x00 { + t.Fatalf("Untar should have kept the 'security.capability' xattr.") + } + } +} + +func TestTarWithOptions(t *testing.T) { + origin, err := ioutil.TempDir("", "docker-test-untar-origin") + if err != nil { + t.Fatal(err) + } + if _, err := ioutil.TempDir(origin, "folder"); err != nil { + t.Fatal(err) + } + defer os.RemoveAll(origin) + if err := ioutil.WriteFile(path.Join(origin, "1"), []byte("hello world"), 0700); err != nil { + t.Fatal(err) + } + if err := ioutil.WriteFile(path.Join(origin, "2"), []byte("welcome!"), 0700); err != nil { + t.Fatal(err) + } + + cases := []struct { + opts *TarOptions + numChanges int + }{ + {&TarOptions{IncludeFiles: []string{"1"}}, 2}, + {&TarOptions{ExcludePatterns: []string{"2"}}, 1}, + {&TarOptions{ExcludePatterns: []string{"1", "folder*"}}, 2}, + {&TarOptions{IncludeFiles: []string{"1", "1"}}, 2}, + {&TarOptions{Name: "test", IncludeFiles: []string{"1"}}, 4}, + } + for _, testCase := range cases { + changes, err := tarUntar(t, origin, testCase.opts) + if err != nil { + t.Fatalf("Error tar/untar when testing inclusion/exclusion: %s", err) + } + if len(changes) != testCase.numChanges { + t.Errorf("Expected %d changes, got %d for %+v:", + testCase.numChanges, len(changes), testCase.opts) + } + } +} + +// Some tar archives such as http://haproxy.1wt.eu/download/1.5/src/devel/haproxy-1.5-dev21.tar.gz +// use PAX Global Extended Headers. +// Failing prevents the archives from being uncompressed during ADD +func TestTypeXGlobalHeaderDoesNotFail(t *testing.T) { + hdr := tar.Header{Typeflag: tar.TypeXGlobalHeader} + tmpDir, err := ioutil.TempDir("", "docker-test-archive-pax-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(tmpDir) + err = createTarFile(filepath.Join(tmpDir, "pax_global_header"), tmpDir, &hdr, nil, true, nil) + if err != nil { + t.Fatal(err) + } +} + +// Some tar have both GNU specific (huge uid) and Ustar specific (long name) things. +// Not supposed to happen (should use PAX instead of Ustar for long name) but it does and it should still work. +func TestUntarUstarGnuConflict(t *testing.T) { + f, err := os.Open("testdata/broken.tar") + if err != nil { + t.Fatal(err) + } + found := false + tr := tar.NewReader(f) + // Iterate through the files in the archive. + for { + hdr, err := tr.Next() + if err == io.EOF { + // end of tar archive + break + } + if err != nil { + t.Fatal(err) + } + if hdr.Name == "root/.cpanm/work/1395823785.24209/Plack-1.0030/blib/man3/Plack::Middleware::LighttpdScriptNameFix.3pm" { + found = true + break + } + } + if !found { + t.Fatalf("%s not found in the archive", "root/.cpanm/work/1395823785.24209/Plack-1.0030/blib/man3/Plack::Middleware::LighttpdScriptNameFix.3pm") + } +} + +func TestTarWithBlockCharFifo(t *testing.T) { + origin, err := ioutil.TempDir("", "docker-test-tar-hardlink") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(origin) + if err := ioutil.WriteFile(path.Join(origin, "1"), []byte("hello world"), 0700); err != nil { + t.Fatal(err) + } + if err := system.Mknod(path.Join(origin, "2"), syscall.S_IFBLK, int(system.Mkdev(int64(12), int64(5)))); err != nil { + t.Fatal(err) + } + if err := system.Mknod(path.Join(origin, "3"), syscall.S_IFCHR, int(system.Mkdev(int64(12), int64(5)))); err != nil { + t.Fatal(err) + } + if err := system.Mknod(path.Join(origin, "4"), syscall.S_IFIFO, int(system.Mkdev(int64(12), int64(5)))); err != nil { + t.Fatal(err) + } + + dest, err := ioutil.TempDir("", "docker-test-tar-hardlink-dest") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(dest) + + // we'll do this in two steps to separate failure + fh, err := Tar(origin, Uncompressed) + if err != nil { + t.Fatal(err) + } + + // ensure we can read the whole thing with no error, before writing back out + buf, err := ioutil.ReadAll(fh) + if err != nil { + t.Fatal(err) + } + + bRdr := bytes.NewReader(buf) + err = Untar(bRdr, dest, &TarOptions{Compression: Uncompressed}) + if err != nil { + t.Fatal(err) + } + + changes, err := ChangesDirs(origin, dest) + if err != nil { + t.Fatal(err) + } + if len(changes) > 0 { + t.Fatalf("Tar with special device (block, char, fifo) should keep them (recreate them when untar) : %v", changes) + } +} + +func TestTarWithHardLink(t *testing.T) { + origin, err := ioutil.TempDir("", "docker-test-tar-hardlink") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(origin) + if err := ioutil.WriteFile(path.Join(origin, "1"), []byte("hello world"), 0700); err != nil { + t.Fatal(err) + } + if err := os.Link(path.Join(origin, "1"), path.Join(origin, "2")); err != nil { + t.Fatal(err) + } + + var i1, i2 uint64 + if i1, err = getNlink(path.Join(origin, "1")); err != nil { + t.Fatal(err) + } + // sanity check that we can hardlink + if i1 != 2 { + t.Skipf("skipping since hardlinks don't work here; expected 2 links, got %d", i1) + } + + dest, err := ioutil.TempDir("", "docker-test-tar-hardlink-dest") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(dest) + + // we'll do this in two steps to separate failure + fh, err := Tar(origin, Uncompressed) + if err != nil { + t.Fatal(err) + } + + // ensure we can read the whole thing with no error, before writing back out + buf, err := ioutil.ReadAll(fh) + if err != nil { + t.Fatal(err) + } + + bRdr := bytes.NewReader(buf) + err = Untar(bRdr, dest, &TarOptions{Compression: Uncompressed}) + if err != nil { + t.Fatal(err) + } + + if i1, err = getInode(path.Join(dest, "1")); err != nil { + t.Fatal(err) + } + if i2, err = getInode(path.Join(dest, "2")); err != nil { + t.Fatal(err) + } + + if i1 != i2 { + t.Errorf("expected matching inodes, but got %d and %d", i1, i2) + } +} + +func getNlink(path string) (uint64, error) { + stat, err := os.Stat(path) + if err != nil { + return 0, err + } + statT, ok := stat.Sys().(*syscall.Stat_t) + if !ok { + return 0, fmt.Errorf("expected type *syscall.Stat_t, got %t", stat.Sys()) + } + // We need this conversion on ARM64 + return uint64(statT.Nlink), nil +} + +func getInode(path string) (uint64, error) { + stat, err := os.Stat(path) + if err != nil { + return 0, err + } + statT, ok := stat.Sys().(*syscall.Stat_t) + if !ok { + return 0, fmt.Errorf("expected type *syscall.Stat_t, got %t", stat.Sys()) + } + return statT.Ino, nil +} + +func prepareUntarSourceDirectory(numberOfFiles int, targetPath string, makeLinks bool) (int, error) { + fileData := []byte("fooo") + for n := 0; n < numberOfFiles; n++ { + fileName := fmt.Sprintf("file-%d", n) + if err := ioutil.WriteFile(path.Join(targetPath, fileName), fileData, 0700); err != nil { + return 0, err + } + if makeLinks { + if err := os.Link(path.Join(targetPath, fileName), path.Join(targetPath, fileName+"-link")); err != nil { + return 0, err + } + } + } + totalSize := numberOfFiles * len(fileData) + return totalSize, nil +} + +func BenchmarkTarUntar(b *testing.B) { + origin, err := ioutil.TempDir("", "docker-test-untar-origin") + if err != nil { + b.Fatal(err) + } + tempDir, err := ioutil.TempDir("", "docker-test-untar-destination") + if err != nil { + b.Fatal(err) + } + target := path.Join(tempDir, "dest") + n, err := prepareUntarSourceDirectory(100, origin, false) + if err != nil { + b.Fatal(err) + } + defer os.RemoveAll(origin) + defer os.RemoveAll(tempDir) + + b.ResetTimer() + b.SetBytes(int64(n)) + for n := 0; n < b.N; n++ { + err := TarUntar(origin, target) + if err != nil { + b.Fatal(err) + } + os.RemoveAll(target) + } +} + +func BenchmarkTarUntarWithLinks(b *testing.B) { + origin, err := ioutil.TempDir("", "docker-test-untar-origin") + if err != nil { + b.Fatal(err) + } + tempDir, err := ioutil.TempDir("", "docker-test-untar-destination") + if err != nil { + b.Fatal(err) + } + target := path.Join(tempDir, "dest") + n, err := prepareUntarSourceDirectory(100, origin, true) + if err != nil { + b.Fatal(err) + } + defer os.RemoveAll(origin) + defer os.RemoveAll(tempDir) + + b.ResetTimer() + b.SetBytes(int64(n)) + for n := 0; n < b.N; n++ { + err := TarUntar(origin, target) + if err != nil { + b.Fatal(err) + } + os.RemoveAll(target) + } +} + +func TestUntarInvalidFilenames(t *testing.T) { + for i, headers := range [][]*tar.Header{ + { + { + Name: "../victim/dotdot", + Typeflag: tar.TypeReg, + Mode: 0644, + }, + }, + { + { + // Note the leading slash + Name: "/../victim/slash-dotdot", + Typeflag: tar.TypeReg, + Mode: 0644, + }, + }, + } { + if err := testBreakout("untar", "docker-TestUntarInvalidFilenames", headers); err != nil { + t.Fatalf("i=%d. %v", i, err) + } + } +} + +func TestUntarHardlinkToSymlink(t *testing.T) { + for i, headers := range [][]*tar.Header{ + { + { + Name: "symlink1", + Typeflag: tar.TypeSymlink, + Linkname: "regfile", + Mode: 0644, + }, + { + Name: "symlink2", + Typeflag: tar.TypeLink, + Linkname: "symlink1", + Mode: 0644, + }, + { + Name: "regfile", + Typeflag: tar.TypeReg, + Mode: 0644, + }, + }, + } { + if err := testBreakout("untar", "docker-TestUntarHardlinkToSymlink", headers); err != nil { + t.Fatalf("i=%d. %v", i, err) + } + } +} + +func TestUntarInvalidHardlink(t *testing.T) { + for i, headers := range [][]*tar.Header{ + { // try reading victim/hello (../) + { + Name: "dotdot", + Typeflag: tar.TypeLink, + Linkname: "../victim/hello", + Mode: 0644, + }, + }, + { // try reading victim/hello (/../) + { + Name: "slash-dotdot", + Typeflag: tar.TypeLink, + // Note the leading slash + Linkname: "/../victim/hello", + Mode: 0644, + }, + }, + { // try writing victim/file + { + Name: "loophole-victim", + Typeflag: tar.TypeLink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "loophole-victim/file", + Typeflag: tar.TypeReg, + Mode: 0644, + }, + }, + { // try reading victim/hello (hardlink, symlink) + { + Name: "loophole-victim", + Typeflag: tar.TypeLink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "symlink", + Typeflag: tar.TypeSymlink, + Linkname: "loophole-victim/hello", + Mode: 0644, + }, + }, + { // Try reading victim/hello (hardlink, hardlink) + { + Name: "loophole-victim", + Typeflag: tar.TypeLink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "hardlink", + Typeflag: tar.TypeLink, + Linkname: "loophole-victim/hello", + Mode: 0644, + }, + }, + { // Try removing victim directory (hardlink) + { + Name: "loophole-victim", + Typeflag: tar.TypeLink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "loophole-victim", + Typeflag: tar.TypeReg, + Mode: 0644, + }, + }, + } { + if err := testBreakout("untar", "docker-TestUntarInvalidHardlink", headers); err != nil { + t.Fatalf("i=%d. %v", i, err) + } + } +} + +func TestUntarInvalidSymlink(t *testing.T) { + for i, headers := range [][]*tar.Header{ + { // try reading victim/hello (../) + { + Name: "dotdot", + Typeflag: tar.TypeSymlink, + Linkname: "../victim/hello", + Mode: 0644, + }, + }, + { // try reading victim/hello (/../) + { + Name: "slash-dotdot", + Typeflag: tar.TypeSymlink, + // Note the leading slash + Linkname: "/../victim/hello", + Mode: 0644, + }, + }, + { // try writing victim/file + { + Name: "loophole-victim", + Typeflag: tar.TypeSymlink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "loophole-victim/file", + Typeflag: tar.TypeReg, + Mode: 0644, + }, + }, + { // try reading victim/hello (symlink, symlink) + { + Name: "loophole-victim", + Typeflag: tar.TypeSymlink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "symlink", + Typeflag: tar.TypeSymlink, + Linkname: "loophole-victim/hello", + Mode: 0644, + }, + }, + { // try reading victim/hello (symlink, hardlink) + { + Name: "loophole-victim", + Typeflag: tar.TypeSymlink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "hardlink", + Typeflag: tar.TypeLink, + Linkname: "loophole-victim/hello", + Mode: 0644, + }, + }, + { // try removing victim directory (symlink) + { + Name: "loophole-victim", + Typeflag: tar.TypeSymlink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "loophole-victim", + Typeflag: tar.TypeReg, + Mode: 0644, + }, + }, + { // try writing to victim/newdir/newfile with a symlink in the path + { + // this header needs to be before the next one, or else there is an error + Name: "dir/loophole", + Typeflag: tar.TypeSymlink, + Linkname: "../../victim", + Mode: 0755, + }, + { + Name: "dir/loophole/newdir/newfile", + Typeflag: tar.TypeReg, + Mode: 0644, + }, + }, + } { + if err := testBreakout("untar", "docker-TestUntarInvalidSymlink", headers); err != nil { + t.Fatalf("i=%d. %v", i, err) + } + } +} + +func TestTempArchiveCloseMultipleTimes(t *testing.T) { + reader := ioutil.NopCloser(strings.NewReader("hello")) + tempArchive, err := NewTempArchive(reader, "") + buf := make([]byte, 10) + n, err := tempArchive.Read(buf) + if n != 5 { + t.Fatalf("Expected to read 5 bytes. Read %d instead", n) + } + for i := 0; i < 3; i++ { + if err = tempArchive.Close(); err != nil { + t.Fatalf("i=%d. Unexpected error closing temp archive: %v", i, err) + } + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_unix.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_unix.go new file mode 100644 index 0000000000000..5c754373fab47 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_unix.go @@ -0,0 +1,89 @@ +// +build !windows + +package archive + +import ( + "archive/tar" + "errors" + "os" + "syscall" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system" +) + +// CanonicalTarNameForPath returns platform-specific filepath +// to canonical posix-style path for tar archival. p is relative +// path. +func CanonicalTarNameForPath(p string) (string, error) { + return p, nil // already unix-style +} + +// chmodTarEntry is used to adjust the file permissions used in tar header based +// on the platform the archival is done. + +func chmodTarEntry(perm os.FileMode) os.FileMode { + return perm // noop for unix as golang APIs provide perm bits correctly +} + +func setHeaderForSpecialDevice(hdr *tar.Header, ta *tarAppender, name string, stat interface{}) (nlink uint32, inode uint64, err error) { + s, ok := stat.(*syscall.Stat_t) + + if !ok { + err = errors.New("cannot convert stat value to syscall.Stat_t") + return + } + + nlink = uint32(s.Nlink) + inode = uint64(s.Ino) + + // Currently go does not fil in the major/minors + if s.Mode&syscall.S_IFBLK != 0 || + s.Mode&syscall.S_IFCHR != 0 { + hdr.Devmajor = int64(major(uint64(s.Rdev))) + hdr.Devminor = int64(minor(uint64(s.Rdev))) + } + + return +} + +func major(device uint64) uint64 { + return (device >> 8) & 0xfff +} + +func minor(device uint64) uint64 { + return (device & 0xff) | ((device >> 12) & 0xfff00) +} + +// handleTarTypeBlockCharFifo is an OS-specific helper function used by +// createTarFile to handle the following types of header: Block; Char; Fifo +func handleTarTypeBlockCharFifo(hdr *tar.Header, path string) error { + mode := uint32(hdr.Mode & 07777) + switch hdr.Typeflag { + case tar.TypeBlock: + mode |= syscall.S_IFBLK + case tar.TypeChar: + mode |= syscall.S_IFCHR + case tar.TypeFifo: + mode |= syscall.S_IFIFO + } + + if err := system.Mknod(path, mode, int(system.Mkdev(hdr.Devmajor, hdr.Devminor))); err != nil { + return err + } + return nil +} + +func handleLChmod(hdr *tar.Header, path string, hdrInfo os.FileInfo) error { + if hdr.Typeflag == tar.TypeLink { + if fi, err := os.Lstat(hdr.Linkname); err == nil && (fi.Mode()&os.ModeSymlink == 0) { + if err := os.Chmod(path, hdrInfo.Mode()); err != nil { + return err + } + } + } else if hdr.Typeflag != tar.TypeSymlink { + if err := os.Chmod(path, hdrInfo.Mode()); err != nil { + return err + } + } + return nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_unix_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_unix_test.go new file mode 100644 index 0000000000000..18f45c480f10b --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_unix_test.go @@ -0,0 +1,60 @@ +// +build !windows + +package archive + +import ( + "os" + "testing" +) + +func TestCanonicalTarNameForPath(t *testing.T) { + cases := []struct{ in, expected string }{ + {"foo", "foo"}, + {"foo/bar", "foo/bar"}, + {"foo/dir/", "foo/dir/"}, + } + for _, v := range cases { + if out, err := CanonicalTarNameForPath(v.in); err != nil { + t.Fatalf("cannot get canonical name for path: %s: %v", v.in, err) + } else if out != v.expected { + t.Fatalf("wrong canonical tar name. expected:%s got:%s", v.expected, out) + } + } +} + +func TestCanonicalTarName(t *testing.T) { + cases := []struct { + in string + isDir bool + expected string + }{ + {"foo", false, "foo"}, + {"foo", true, "foo/"}, + {"foo/bar", false, "foo/bar"}, + {"foo/bar", true, "foo/bar/"}, + } + for _, v := range cases { + if out, err := canonicalTarName(v.in, v.isDir); err != nil { + t.Fatalf("cannot get canonical name for path: %s: %v", v.in, err) + } else if out != v.expected { + t.Fatalf("wrong canonical tar name. expected:%s got:%s", v.expected, out) + } + } +} + +func TestChmodTarEntry(t *testing.T) { + cases := []struct { + in, expected os.FileMode + }{ + {0000, 0000}, + {0777, 0777}, + {0644, 0644}, + {0755, 0755}, + {0444, 0444}, + } + for _, v := range cases { + if out := chmodTarEntry(v.in); out != v.expected { + t.Fatalf("wrong chmod. expected:%v got:%v", v.expected, out) + } + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_windows.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_windows.go new file mode 100644 index 0000000000000..10db4bd00ee39 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_windows.go @@ -0,0 +1,50 @@ +// +build windows + +package archive + +import ( + "archive/tar" + "fmt" + "os" + "strings" +) + +// canonicalTarNameForPath returns platform-specific filepath +// to canonical posix-style path for tar archival. p is relative +// path. +func CanonicalTarNameForPath(p string) (string, error) { + // windows: convert windows style relative path with backslashes + // into forward slashes. Since windows does not allow '/' or '\' + // in file names, it is mostly safe to replace however we must + // check just in case + if strings.Contains(p, "/") { + return "", fmt.Errorf("Windows path contains forward slash: %s", p) + } + return strings.Replace(p, string(os.PathSeparator), "/", -1), nil + +} + +// chmodTarEntry is used to adjust the file permissions used in tar header based +// on the platform the archival is done. +func chmodTarEntry(perm os.FileMode) os.FileMode { + perm &= 0755 + // Add the x bit: make everything +x from windows + perm |= 0111 + + return perm +} + +func setHeaderForSpecialDevice(hdr *tar.Header, ta *tarAppender, name string, stat interface{}) (nlink uint32, inode uint64, err error) { + // do nothing. no notion of Rdev, Inode, Nlink in stat on Windows + return +} + +// handleTarTypeBlockCharFifo is an OS-specific helper function used by +// createTarFile to handle the following types of header: Block; Char; Fifo +func handleTarTypeBlockCharFifo(hdr *tar.Header, path string) error { + return nil +} + +func handleLChmod(hdr *tar.Header, path string, hdrInfo os.FileInfo) error { + return nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_windows_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_windows_test.go new file mode 100644 index 0000000000000..72bc71e06b413 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/archive_windows_test.go @@ -0,0 +1,65 @@ +// +build windows + +package archive + +import ( + "os" + "testing" +) + +func TestCanonicalTarNameForPath(t *testing.T) { + cases := []struct { + in, expected string + shouldFail bool + }{ + {"foo", "foo", false}, + {"foo/bar", "___", true}, // unix-styled windows path must fail + {`foo\bar`, "foo/bar", false}, + } + for _, v := range cases { + if out, err := CanonicalTarNameForPath(v.in); err != nil && !v.shouldFail { + t.Fatalf("cannot get canonical name for path: %s: %v", v.in, err) + } else if v.shouldFail && err == nil { + t.Fatalf("canonical path call should have failed with error. in=%s out=%s", v.in, out) + } else if !v.shouldFail && out != v.expected { + t.Fatalf("wrong canonical tar name. expected:%s got:%s", v.expected, out) + } + } +} + +func TestCanonicalTarName(t *testing.T) { + cases := []struct { + in string + isDir bool + expected string + }{ + {"foo", false, "foo"}, + {"foo", true, "foo/"}, + {`foo\bar`, false, "foo/bar"}, + {`foo\bar`, true, "foo/bar/"}, + } + for _, v := range cases { + if out, err := canonicalTarName(v.in, v.isDir); err != nil { + t.Fatalf("cannot get canonical name for path: %s: %v", v.in, err) + } else if out != v.expected { + t.Fatalf("wrong canonical tar name. expected:%s got:%s", v.expected, out) + } + } +} + +func TestChmodTarEntry(t *testing.T) { + cases := []struct { + in, expected os.FileMode + }{ + {0000, 0111}, + {0777, 0755}, + {0644, 0755}, + {0755, 0755}, + {0444, 0555}, + } + for _, v := range cases { + if out := chmodTarEntry(v.in); out != v.expected { + t.Fatalf("wrong chmod. expected:%v got:%v", v.expected, out) + } + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes.go new file mode 100644 index 0000000000000..c7838e8599eb4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes.go @@ -0,0 +1,383 @@ +package archive + +import ( + "archive/tar" + "bytes" + "fmt" + "io" + "io/ioutil" + "os" + "path/filepath" + "sort" + "strings" + "syscall" + "time" + + "github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus" + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/pools" + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system" +) + +type ChangeType int + +const ( + ChangeModify = iota + ChangeAdd + ChangeDelete +) + +type Change struct { + Path string + Kind ChangeType +} + +func (change *Change) String() string { + var kind string + switch change.Kind { + case ChangeModify: + kind = "C" + case ChangeAdd: + kind = "A" + case ChangeDelete: + kind = "D" + } + return fmt.Sprintf("%s %s", kind, change.Path) +} + +// for sort.Sort +type changesByPath []Change + +func (c changesByPath) Less(i, j int) bool { return c[i].Path < c[j].Path } +func (c changesByPath) Len() int { return len(c) } +func (c changesByPath) Swap(i, j int) { c[j], c[i] = c[i], c[j] } + +// Gnu tar and the go tar writer don't have sub-second mtime +// precision, which is problematic when we apply changes via tar +// files, we handle this by comparing for exact times, *or* same +// second count and either a or b having exactly 0 nanoseconds +func sameFsTime(a, b time.Time) bool { + return a == b || + (a.Unix() == b.Unix() && + (a.Nanosecond() == 0 || b.Nanosecond() == 0)) +} + +func sameFsTimeSpec(a, b syscall.Timespec) bool { + return a.Sec == b.Sec && + (a.Nsec == b.Nsec || a.Nsec == 0 || b.Nsec == 0) +} + +// Changes walks the path rw and determines changes for the files in the path, +// with respect to the parent layers +func Changes(layers []string, rw string) ([]Change, error) { + var ( + changes []Change + changedDirs = make(map[string]struct{}) + ) + + err := filepath.Walk(rw, func(path string, f os.FileInfo, err error) error { + if err != nil { + return err + } + + // Rebase path + path, err = filepath.Rel(rw, path) + if err != nil { + return err + } + + // As this runs on the daemon side, file paths are OS specific. + path = filepath.Join(string(os.PathSeparator), path) + + // Skip root + if path == string(os.PathSeparator) { + return nil + } + + // Skip AUFS metadata + if matched, err := filepath.Match(string(os.PathSeparator)+".wh..wh.*", path); err != nil || matched { + return err + } + + change := Change{ + Path: path, + } + + // Find out what kind of modification happened + file := filepath.Base(path) + // If there is a whiteout, then the file was removed + if strings.HasPrefix(file, ".wh.") { + originalFile := file[len(".wh."):] + change.Path = filepath.Join(filepath.Dir(path), originalFile) + change.Kind = ChangeDelete + } else { + // Otherwise, the file was added + change.Kind = ChangeAdd + + // ...Unless it already existed in a top layer, in which case, it's a modification + for _, layer := range layers { + stat, err := os.Stat(filepath.Join(layer, path)) + if err != nil && !os.IsNotExist(err) { + return err + } + if err == nil { + // The file existed in the top layer, so that's a modification + + // However, if it's a directory, maybe it wasn't actually modified. + // If you modify /foo/bar/baz, then /foo will be part of the changed files only because it's the parent of bar + if stat.IsDir() && f.IsDir() { + if f.Size() == stat.Size() && f.Mode() == stat.Mode() && sameFsTime(f.ModTime(), stat.ModTime()) { + // Both directories are the same, don't record the change + return nil + } + } + change.Kind = ChangeModify + break + } + } + } + + // If /foo/bar/file.txt is modified, then /foo/bar must be part of the changed files. + // This block is here to ensure the change is recorded even if the + // modify time, mode and size of the parent directoriy in the rw and ro layers are all equal. + // Check https://github.com/docker/docker/pull/13590 for details. + if f.IsDir() { + changedDirs[path] = struct{}{} + } + if change.Kind == ChangeAdd || change.Kind == ChangeDelete { + parent := filepath.Dir(path) + if _, ok := changedDirs[parent]; !ok && parent != "/" { + changes = append(changes, Change{Path: parent, Kind: ChangeModify}) + changedDirs[parent] = struct{}{} + } + } + + // Record change + changes = append(changes, change) + return nil + }) + if err != nil && !os.IsNotExist(err) { + return nil, err + } + return changes, nil +} + +type FileInfo struct { + parent *FileInfo + name string + stat *system.Stat_t + children map[string]*FileInfo + capability []byte + added bool +} + +func (root *FileInfo) LookUp(path string) *FileInfo { + // As this runs on the daemon side, file paths are OS specific. + parent := root + if path == string(os.PathSeparator) { + return root + } + + pathElements := strings.Split(path, string(os.PathSeparator)) + for _, elem := range pathElements { + if elem != "" { + child := parent.children[elem] + if child == nil { + return nil + } + parent = child + } + } + return parent +} + +func (info *FileInfo) path() string { + if info.parent == nil { + // As this runs on the daemon side, file paths are OS specific. + return string(os.PathSeparator) + } + return filepath.Join(info.parent.path(), info.name) +} + +func (info *FileInfo) addChanges(oldInfo *FileInfo, changes *[]Change) { + + sizeAtEntry := len(*changes) + + if oldInfo == nil { + // add + change := Change{ + Path: info.path(), + Kind: ChangeAdd, + } + *changes = append(*changes, change) + info.added = true + } + + // We make a copy so we can modify it to detect additions + // also, we only recurse on the old dir if the new info is a directory + // otherwise any previous delete/change is considered recursive + oldChildren := make(map[string]*FileInfo) + if oldInfo != nil && info.isDir() { + for k, v := range oldInfo.children { + oldChildren[k] = v + } + } + + for name, newChild := range info.children { + oldChild, _ := oldChildren[name] + if oldChild != nil { + // change? + oldStat := oldChild.stat + newStat := newChild.stat + // Note: We can't compare inode or ctime or blocksize here, because these change + // when copying a file into a container. However, that is not generally a problem + // because any content change will change mtime, and any status change should + // be visible when actually comparing the stat fields. The only time this + // breaks down is if some code intentionally hides a change by setting + // back mtime + if statDifferent(oldStat, newStat) || + bytes.Compare(oldChild.capability, newChild.capability) != 0 { + change := Change{ + Path: newChild.path(), + Kind: ChangeModify, + } + *changes = append(*changes, change) + newChild.added = true + } + + // Remove from copy so we can detect deletions + delete(oldChildren, name) + } + + newChild.addChanges(oldChild, changes) + } + for _, oldChild := range oldChildren { + // delete + change := Change{ + Path: oldChild.path(), + Kind: ChangeDelete, + } + *changes = append(*changes, change) + } + + // If there were changes inside this directory, we need to add it, even if the directory + // itself wasn't changed. This is needed to properly save and restore filesystem permissions. + // As this runs on the daemon side, file paths are OS specific. + if len(*changes) > sizeAtEntry && info.isDir() && !info.added && info.path() != string(os.PathSeparator) { + change := Change{ + Path: info.path(), + Kind: ChangeModify, + } + // Let's insert the directory entry before the recently added entries located inside this dir + *changes = append(*changes, change) // just to resize the slice, will be overwritten + copy((*changes)[sizeAtEntry+1:], (*changes)[sizeAtEntry:]) + (*changes)[sizeAtEntry] = change + } + +} + +func (info *FileInfo) Changes(oldInfo *FileInfo) []Change { + var changes []Change + + info.addChanges(oldInfo, &changes) + + return changes +} + +func newRootFileInfo() *FileInfo { + // As this runs on the daemon side, file paths are OS specific. + root := &FileInfo{ + name: string(os.PathSeparator), + children: make(map[string]*FileInfo), + } + return root +} + +// ChangesDirs compares two directories and generates an array of Change objects describing the changes. +// If oldDir is "", then all files in newDir will be Add-Changes. +func ChangesDirs(newDir, oldDir string) ([]Change, error) { + var ( + oldRoot, newRoot *FileInfo + ) + if oldDir == "" { + emptyDir, err := ioutil.TempDir("", "empty") + if err != nil { + return nil, err + } + defer os.Remove(emptyDir) + oldDir = emptyDir + } + oldRoot, newRoot, err := collectFileInfoForChanges(oldDir, newDir) + if err != nil { + return nil, err + } + + return newRoot.Changes(oldRoot), nil +} + +// ChangesSize calculates the size in bytes of the provided changes, based on newDir. +func ChangesSize(newDir string, changes []Change) int64 { + var size int64 + for _, change := range changes { + if change.Kind == ChangeModify || change.Kind == ChangeAdd { + file := filepath.Join(newDir, change.Path) + fileInfo, _ := os.Lstat(file) + if fileInfo != nil && !fileInfo.IsDir() { + size += fileInfo.Size() + } + } + } + return size +} + +// ExportChanges produces an Archive from the provided changes, relative to dir. +func ExportChanges(dir string, changes []Change) (Archive, error) { + reader, writer := io.Pipe() + go func() { + ta := &tarAppender{ + TarWriter: tar.NewWriter(writer), + Buffer: pools.BufioWriter32KPool.Get(nil), + SeenFiles: make(map[uint64]string), + } + // this buffer is needed for the duration of this piped stream + defer pools.BufioWriter32KPool.Put(ta.Buffer) + + sort.Sort(changesByPath(changes)) + + // In general we log errors here but ignore them because + // during e.g. a diff operation the container can continue + // mutating the filesystem and we can see transient errors + // from this + for _, change := range changes { + if change.Kind == ChangeDelete { + whiteOutDir := filepath.Dir(change.Path) + whiteOutBase := filepath.Base(change.Path) + whiteOut := filepath.Join(whiteOutDir, ".wh."+whiteOutBase) + timestamp := time.Now() + hdr := &tar.Header{ + Name: whiteOut[1:], + Size: 0, + ModTime: timestamp, + AccessTime: timestamp, + ChangeTime: timestamp, + } + if err := ta.TarWriter.WriteHeader(hdr); err != nil { + logrus.Debugf("Can't write whiteout header: %s", err) + } + } else { + path := filepath.Join(dir, change.Path) + if err := ta.addTarFile(path, change.Path[1:]); err != nil { + logrus.Debugf("Can't add file %s to tar: %s", path, err) + } + } + } + + // Make sure to check the error on Close. + if err := ta.TarWriter.Close(); err != nil { + logrus.Debugf("Can't close layer: %s", err) + } + if err := writer.Close(); err != nil { + logrus.Debugf("failed close Changes writer: %s", err) + } + }() + return reader, nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_linux.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_linux.go new file mode 100644 index 0000000000000..378cc09c85960 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_linux.go @@ -0,0 +1,285 @@ +package archive + +import ( + "bytes" + "fmt" + "os" + "path/filepath" + "sort" + "syscall" + "unsafe" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system" +) + +// walker is used to implement collectFileInfoForChanges on linux. Where this +// method in general returns the entire contents of two directory trees, we +// optimize some FS calls out on linux. In particular, we take advantage of the +// fact that getdents(2) returns the inode of each file in the directory being +// walked, which, when walking two trees in parallel to generate a list of +// changes, can be used to prune subtrees without ever having to lstat(2) them +// directly. Eliminating stat calls in this way can save up to seconds on large +// images. +type walker struct { + dir1 string + dir2 string + root1 *FileInfo + root2 *FileInfo +} + +// collectFileInfoForChanges returns a complete representation of the trees +// rooted at dir1 and dir2, with one important exception: any subtree or +// leaf where the inode and device numbers are an exact match between dir1 +// and dir2 will be pruned from the results. This method is *only* to be used +// to generating a list of changes between the two directories, as it does not +// reflect the full contents. +func collectFileInfoForChanges(dir1, dir2 string) (*FileInfo, *FileInfo, error) { + w := &walker{ + dir1: dir1, + dir2: dir2, + root1: newRootFileInfo(), + root2: newRootFileInfo(), + } + + i1, err := os.Lstat(w.dir1) + if err != nil { + return nil, nil, err + } + i2, err := os.Lstat(w.dir2) + if err != nil { + return nil, nil, err + } + + if err := w.walk("/", i1, i2); err != nil { + return nil, nil, err + } + + return w.root1, w.root2, nil +} + +// Given a FileInfo, its path info, and a reference to the root of the tree +// being constructed, register this file with the tree. +func walkchunk(path string, fi os.FileInfo, dir string, root *FileInfo) error { + if fi == nil { + return nil + } + parent := root.LookUp(filepath.Dir(path)) + if parent == nil { + return fmt.Errorf("collectFileInfoForChanges: Unexpectedly no parent for %s", path) + } + info := &FileInfo{ + name: filepath.Base(path), + children: make(map[string]*FileInfo), + parent: parent, + } + cpath := filepath.Join(dir, path) + stat, err := system.FromStatT(fi.Sys().(*syscall.Stat_t)) + if err != nil { + return err + } + info.stat = stat + info.capability, _ = system.Lgetxattr(cpath, "security.capability") // lgetxattr(2): fs access + parent.children[info.name] = info + return nil +} + +// Walk a subtree rooted at the same path in both trees being iterated. For +// example, /docker/overlay/1234/a/b/c/d and /docker/overlay/8888/a/b/c/d +func (w *walker) walk(path string, i1, i2 os.FileInfo) (err error) { + // Register these nodes with the return trees, unless we're still at the + // (already-created) roots: + if path != "/" { + if err := walkchunk(path, i1, w.dir1, w.root1); err != nil { + return err + } + if err := walkchunk(path, i2, w.dir2, w.root2); err != nil { + return err + } + } + + is1Dir := i1 != nil && i1.IsDir() + is2Dir := i2 != nil && i2.IsDir() + + sameDevice := false + if i1 != nil && i2 != nil { + si1 := i1.Sys().(*syscall.Stat_t) + si2 := i2.Sys().(*syscall.Stat_t) + if si1.Dev == si2.Dev { + sameDevice = true + } + } + + // If these files are both non-existent, or leaves (non-dirs), we are done. + if !is1Dir && !is2Dir { + return nil + } + + // Fetch the names of all the files contained in both directories being walked: + var names1, names2 []nameIno + if is1Dir { + names1, err = readdirnames(filepath.Join(w.dir1, path)) // getdents(2): fs access + if err != nil { + return err + } + } + if is2Dir { + names2, err = readdirnames(filepath.Join(w.dir2, path)) // getdents(2): fs access + if err != nil { + return err + } + } + + // We have lists of the files contained in both parallel directories, sorted + // in the same order. Walk them in parallel, generating a unique merged list + // of all items present in either or both directories. + var names []string + ix1 := 0 + ix2 := 0 + + for { + if ix1 >= len(names1) { + break + } + if ix2 >= len(names2) { + break + } + + ni1 := names1[ix1] + ni2 := names2[ix2] + + switch bytes.Compare([]byte(ni1.name), []byte(ni2.name)) { + case -1: // ni1 < ni2 -- advance ni1 + // we will not encounter ni1 in names2 + names = append(names, ni1.name) + ix1++ + case 0: // ni1 == ni2 + if ni1.ino != ni2.ino || !sameDevice { + names = append(names, ni1.name) + } + ix1++ + ix2++ + case 1: // ni1 > ni2 -- advance ni2 + // we will not encounter ni2 in names1 + names = append(names, ni2.name) + ix2++ + } + } + for ix1 < len(names1) { + names = append(names, names1[ix1].name) + ix1++ + } + for ix2 < len(names2) { + names = append(names, names2[ix2].name) + ix2++ + } + + // For each of the names present in either or both of the directories being + // iterated, stat the name under each root, and recurse the pair of them: + for _, name := range names { + fname := filepath.Join(path, name) + var cInfo1, cInfo2 os.FileInfo + if is1Dir { + cInfo1, err = os.Lstat(filepath.Join(w.dir1, fname)) // lstat(2): fs access + if err != nil && !os.IsNotExist(err) { + return err + } + } + if is2Dir { + cInfo2, err = os.Lstat(filepath.Join(w.dir2, fname)) // lstat(2): fs access + if err != nil && !os.IsNotExist(err) { + return err + } + } + if err = w.walk(fname, cInfo1, cInfo2); err != nil { + return err + } + } + return nil +} + +// {name,inode} pairs used to support the early-pruning logic of the walker type +type nameIno struct { + name string + ino uint64 +} + +type nameInoSlice []nameIno + +func (s nameInoSlice) Len() int { return len(s) } +func (s nameInoSlice) Swap(i, j int) { s[i], s[j] = s[j], s[i] } +func (s nameInoSlice) Less(i, j int) bool { return s[i].name < s[j].name } + +// readdirnames is a hacked-apart version of the Go stdlib code, exposing inode +// numbers further up the stack when reading directory contents. Unlike +// os.Readdirnames, which returns a list of filenames, this function returns a +// list of {filename,inode} pairs. +func readdirnames(dirname string) (names []nameIno, err error) { + var ( + size = 100 + buf = make([]byte, 4096) + nbuf int + bufp int + nb int + ) + + f, err := os.Open(dirname) + if err != nil { + return nil, err + } + defer f.Close() + + names = make([]nameIno, 0, size) // Empty with room to grow. + for { + // Refill the buffer if necessary + if bufp >= nbuf { + bufp = 0 + nbuf, err = syscall.ReadDirent(int(f.Fd()), buf) // getdents on linux + if nbuf < 0 { + nbuf = 0 + } + if err != nil { + return nil, os.NewSyscallError("readdirent", err) + } + if nbuf <= 0 { + break // EOF + } + } + + // Drain the buffer + nb, names = parseDirent(buf[bufp:nbuf], names) + bufp += nb + } + + sl := nameInoSlice(names) + sort.Sort(sl) + return sl, nil +} + +// parseDirent is a minor modification of syscall.ParseDirent (linux version) +// which returns {name,inode} pairs instead of just names. +func parseDirent(buf []byte, names []nameIno) (consumed int, newnames []nameIno) { + origlen := len(buf) + for len(buf) > 0 { + dirent := (*syscall.Dirent)(unsafe.Pointer(&buf[0])) + buf = buf[dirent.Reclen:] + if dirent.Ino == 0 { // File absent in directory. + continue + } + bytes := (*[10000]byte)(unsafe.Pointer(&dirent.Name[0])) + var name = string(bytes[0:clen(bytes[:])]) + if name == "." || name == ".." { // Useless names + continue + } + names = append(names, nameIno{name, dirent.Ino}) + } + return origlen - len(buf), names +} + +func clen(n []byte) int { + for i := 0; i < len(n); i++ { + if n[i] == 0 { + return i + } + } + return len(n) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_other.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_other.go new file mode 100644 index 0000000000000..35832f087d0f0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_other.go @@ -0,0 +1,97 @@ +// +build !linux + +package archive + +import ( + "fmt" + "os" + "path/filepath" + "runtime" + "strings" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system" +) + +func collectFileInfoForChanges(oldDir, newDir string) (*FileInfo, *FileInfo, error) { + var ( + oldRoot, newRoot *FileInfo + err1, err2 error + errs = make(chan error, 2) + ) + go func() { + oldRoot, err1 = collectFileInfo(oldDir) + errs <- err1 + }() + go func() { + newRoot, err2 = collectFileInfo(newDir) + errs <- err2 + }() + + // block until both routines have returned + for i := 0; i < 2; i++ { + if err := <-errs; err != nil { + return nil, nil, err + } + } + + return oldRoot, newRoot, nil +} + +func collectFileInfo(sourceDir string) (*FileInfo, error) { + root := newRootFileInfo() + + err := filepath.Walk(sourceDir, func(path string, f os.FileInfo, err error) error { + if err != nil { + return err + } + + // Rebase path + relPath, err := filepath.Rel(sourceDir, path) + if err != nil { + return err + } + + // As this runs on the daemon side, file paths are OS specific. + relPath = filepath.Join(string(os.PathSeparator), relPath) + + // See https://github.com/golang/go/issues/9168 - bug in filepath.Join. + // Temporary workaround. If the returned path starts with two backslashes, + // trim it down to a single backslash. Only relevant on Windows. + if runtime.GOOS == "windows" { + if strings.HasPrefix(relPath, `\\`) { + relPath = relPath[1:] + } + } + + if relPath == string(os.PathSeparator) { + return nil + } + + parent := root.LookUp(filepath.Dir(relPath)) + if parent == nil { + return fmt.Errorf("collectFileInfo: Unexpectedly no parent for %s", relPath) + } + + info := &FileInfo{ + name: filepath.Base(relPath), + children: make(map[string]*FileInfo), + parent: parent, + } + + s, err := system.Lstat(path) + if err != nil { + return err + } + info.stat = s + + info.capability, _ = system.Lgetxattr(path, "security.capability") + + parent.children[info.name] = info + + return nil + }) + if err != nil { + return nil, err + } + return root, nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_posix_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_posix_test.go new file mode 100644 index 0000000000000..9d528e614c4ae --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_posix_test.go @@ -0,0 +1,127 @@ +package archive + +import ( + "archive/tar" + "fmt" + "io" + "io/ioutil" + "os" + "path" + "sort" + "testing" +) + +func TestHardLinkOrder(t *testing.T) { + names := []string{"file1.txt", "file2.txt", "file3.txt"} + msg := []byte("Hey y'all") + + // Create dir + src, err := ioutil.TempDir("", "docker-hardlink-test-src-") + if err != nil { + t.Fatal(err) + } + //defer os.RemoveAll(src) + for _, name := range names { + func() { + fh, err := os.Create(path.Join(src, name)) + if err != nil { + t.Fatal(err) + } + defer fh.Close() + if _, err = fh.Write(msg); err != nil { + t.Fatal(err) + } + }() + } + // Create dest, with changes that includes hardlinks + dest, err := ioutil.TempDir("", "docker-hardlink-test-dest-") + if err != nil { + t.Fatal(err) + } + os.RemoveAll(dest) // we just want the name, at first + if err := copyDir(src, dest); err != nil { + t.Fatal(err) + } + defer os.RemoveAll(dest) + for _, name := range names { + for i := 0; i < 5; i++ { + if err := os.Link(path.Join(dest, name), path.Join(dest, fmt.Sprintf("%s.link%d", name, i))); err != nil { + t.Fatal(err) + } + } + } + + // get changes + changes, err := ChangesDirs(dest, src) + if err != nil { + t.Fatal(err) + } + + // sort + sort.Sort(changesByPath(changes)) + + // ExportChanges + ar, err := ExportChanges(dest, changes) + if err != nil { + t.Fatal(err) + } + hdrs, err := walkHeaders(ar) + if err != nil { + t.Fatal(err) + } + + // reverse sort + sort.Sort(sort.Reverse(changesByPath(changes))) + // ExportChanges + arRev, err := ExportChanges(dest, changes) + if err != nil { + t.Fatal(err) + } + hdrsRev, err := walkHeaders(arRev) + if err != nil { + t.Fatal(err) + } + + // line up the two sets + sort.Sort(tarHeaders(hdrs)) + sort.Sort(tarHeaders(hdrsRev)) + + // compare Size and LinkName + for i := range hdrs { + if hdrs[i].Name != hdrsRev[i].Name { + t.Errorf("headers - expected name %q; but got %q", hdrs[i].Name, hdrsRev[i].Name) + } + if hdrs[i].Size != hdrsRev[i].Size { + t.Errorf("headers - %q expected size %d; but got %d", hdrs[i].Name, hdrs[i].Size, hdrsRev[i].Size) + } + if hdrs[i].Typeflag != hdrsRev[i].Typeflag { + t.Errorf("headers - %q expected type %d; but got %d", hdrs[i].Name, hdrs[i].Typeflag, hdrsRev[i].Typeflag) + } + if hdrs[i].Linkname != hdrsRev[i].Linkname { + t.Errorf("headers - %q expected linkname %q; but got %q", hdrs[i].Name, hdrs[i].Linkname, hdrsRev[i].Linkname) + } + } + +} + +type tarHeaders []tar.Header + +func (th tarHeaders) Len() int { return len(th) } +func (th tarHeaders) Swap(i, j int) { th[j], th[i] = th[i], th[j] } +func (th tarHeaders) Less(i, j int) bool { return th[i].Name < th[j].Name } + +func walkHeaders(r io.Reader) ([]tar.Header, error) { + t := tar.NewReader(r) + headers := []tar.Header{} + for { + hdr, err := t.Next() + if err != nil { + if err == io.EOF { + break + } + return headers, err + } + headers = append(headers, *hdr) + } + return headers, nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_test.go new file mode 100644 index 0000000000000..509bdb2e6d642 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_test.go @@ -0,0 +1,495 @@ +package archive + +import ( + "io/ioutil" + "os" + "os/exec" + "path" + "sort" + "testing" + "time" +) + +func max(x, y int) int { + if x >= y { + return x + } + return y +} + +func copyDir(src, dst string) error { + cmd := exec.Command("cp", "-a", src, dst) + if err := cmd.Run(); err != nil { + return err + } + return nil +} + +type FileType uint32 + +const ( + Regular FileType = iota + Dir + Symlink +) + +type FileData struct { + filetype FileType + path string + contents string + permissions os.FileMode +} + +func createSampleDir(t *testing.T, root string) { + files := []FileData{ + {Regular, "file1", "file1\n", 0600}, + {Regular, "file2", "file2\n", 0666}, + {Regular, "file3", "file3\n", 0404}, + {Regular, "file4", "file4\n", 0600}, + {Regular, "file5", "file5\n", 0600}, + {Regular, "file6", "file6\n", 0600}, + {Regular, "file7", "file7\n", 0600}, + {Dir, "dir1", "", 0740}, + {Regular, "dir1/file1-1", "file1-1\n", 01444}, + {Regular, "dir1/file1-2", "file1-2\n", 0666}, + {Dir, "dir2", "", 0700}, + {Regular, "dir2/file2-1", "file2-1\n", 0666}, + {Regular, "dir2/file2-2", "file2-2\n", 0666}, + {Dir, "dir3", "", 0700}, + {Regular, "dir3/file3-1", "file3-1\n", 0666}, + {Regular, "dir3/file3-2", "file3-2\n", 0666}, + {Dir, "dir4", "", 0700}, + {Regular, "dir4/file3-1", "file4-1\n", 0666}, + {Regular, "dir4/file3-2", "file4-2\n", 0666}, + {Symlink, "symlink1", "target1", 0666}, + {Symlink, "symlink2", "target2", 0666}, + } + + now := time.Now() + for _, info := range files { + p := path.Join(root, info.path) + if info.filetype == Dir { + if err := os.MkdirAll(p, info.permissions); err != nil { + t.Fatal(err) + } + } else if info.filetype == Regular { + if err := ioutil.WriteFile(p, []byte(info.contents), info.permissions); err != nil { + t.Fatal(err) + } + } else if info.filetype == Symlink { + if err := os.Symlink(info.contents, p); err != nil { + t.Fatal(err) + } + } + + if info.filetype != Symlink { + // Set a consistent ctime, atime for all files and dirs + if err := os.Chtimes(p, now, now); err != nil { + t.Fatal(err) + } + } + } +} + +func TestChangeString(t *testing.T) { + modifiyChange := Change{"change", ChangeModify} + toString := modifiyChange.String() + if toString != "C change" { + t.Fatalf("String() of a change with ChangeModifiy Kind should have been %s but was %s", "C change", toString) + } + addChange := Change{"change", ChangeAdd} + toString = addChange.String() + if toString != "A change" { + t.Fatalf("String() of a change with ChangeAdd Kind should have been %s but was %s", "A change", toString) + } + deleteChange := Change{"change", ChangeDelete} + toString = deleteChange.String() + if toString != "D change" { + t.Fatalf("String() of a change with ChangeDelete Kind should have been %s but was %s", "D change", toString) + } +} + +func TestChangesWithNoChanges(t *testing.T) { + rwLayer, err := ioutil.TempDir("", "docker-changes-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(rwLayer) + layer, err := ioutil.TempDir("", "docker-changes-test-layer") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(layer) + createSampleDir(t, layer) + changes, err := Changes([]string{layer}, rwLayer) + if err != nil { + t.Fatal(err) + } + if len(changes) != 0 { + t.Fatalf("Changes with no difference should have detect no changes, but detected %d", len(changes)) + } +} + +func TestChangesWithChanges(t *testing.T) { + // Mock the readonly layer + layer, err := ioutil.TempDir("", "docker-changes-test-layer") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(layer) + createSampleDir(t, layer) + os.MkdirAll(path.Join(layer, "dir1/subfolder"), 0740) + + // Mock the RW layer + rwLayer, err := ioutil.TempDir("", "docker-changes-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(rwLayer) + + // Create a folder in RW layer + dir1 := path.Join(rwLayer, "dir1") + os.MkdirAll(dir1, 0740) + deletedFile := path.Join(dir1, ".wh.file1-2") + ioutil.WriteFile(deletedFile, []byte{}, 0600) + modifiedFile := path.Join(dir1, "file1-1") + ioutil.WriteFile(modifiedFile, []byte{0x00}, 01444) + // Let's add a subfolder for a newFile + subfolder := path.Join(dir1, "subfolder") + os.MkdirAll(subfolder, 0740) + newFile := path.Join(subfolder, "newFile") + ioutil.WriteFile(newFile, []byte{}, 0740) + + changes, err := Changes([]string{layer}, rwLayer) + if err != nil { + t.Fatal(err) + } + + expectedChanges := []Change{ + {"/dir1", ChangeModify}, + {"/dir1/file1-1", ChangeModify}, + {"/dir1/file1-2", ChangeDelete}, + {"/dir1/subfolder", ChangeModify}, + {"/dir1/subfolder/newFile", ChangeAdd}, + } + checkChanges(expectedChanges, changes, t) +} + +// See https://github.com/docker/docker/pull/13590 +func TestChangesWithChangesGH13590(t *testing.T) { + baseLayer, err := ioutil.TempDir("", "docker-changes-test.") + defer os.RemoveAll(baseLayer) + + dir3 := path.Join(baseLayer, "dir1/dir2/dir3") + os.MkdirAll(dir3, 07400) + + file := path.Join(dir3, "file.txt") + ioutil.WriteFile(file, []byte("hello"), 0666) + + layer, err := ioutil.TempDir("", "docker-changes-test2.") + defer os.RemoveAll(layer) + + // Test creating a new file + if err := copyDir(baseLayer+"/dir1", layer+"/"); err != nil { + t.Fatalf("Cmd failed: %q", err) + } + + os.Remove(path.Join(layer, "dir1/dir2/dir3/file.txt")) + file = path.Join(layer, "dir1/dir2/dir3/file1.txt") + ioutil.WriteFile(file, []byte("bye"), 0666) + + changes, err := Changes([]string{baseLayer}, layer) + if err != nil { + t.Fatal(err) + } + + expectedChanges := []Change{ + {"/dir1/dir2/dir3", ChangeModify}, + {"/dir1/dir2/dir3/file1.txt", ChangeAdd}, + } + checkChanges(expectedChanges, changes, t) + + // Now test changing a file + layer, err = ioutil.TempDir("", "docker-changes-test3.") + defer os.RemoveAll(layer) + + if err := copyDir(baseLayer+"/dir1", layer+"/"); err != nil { + t.Fatalf("Cmd failed: %q", err) + } + + file = path.Join(layer, "dir1/dir2/dir3/file.txt") + ioutil.WriteFile(file, []byte("bye"), 0666) + + changes, err = Changes([]string{baseLayer}, layer) + if err != nil { + t.Fatal(err) + } + + expectedChanges = []Change{ + {"/dir1/dir2/dir3/file.txt", ChangeModify}, + } + checkChanges(expectedChanges, changes, t) +} + +// Create an directory, copy it, make sure we report no changes between the two +func TestChangesDirsEmpty(t *testing.T) { + src, err := ioutil.TempDir("", "docker-changes-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(src) + createSampleDir(t, src) + dst := src + "-copy" + if err := copyDir(src, dst); err != nil { + t.Fatal(err) + } + defer os.RemoveAll(dst) + changes, err := ChangesDirs(dst, src) + if err != nil { + t.Fatal(err) + } + + if len(changes) != 0 { + t.Fatalf("Reported changes for identical dirs: %v", changes) + } + os.RemoveAll(src) + os.RemoveAll(dst) +} + +func mutateSampleDir(t *testing.T, root string) { + // Remove a regular file + if err := os.RemoveAll(path.Join(root, "file1")); err != nil { + t.Fatal(err) + } + + // Remove a directory + if err := os.RemoveAll(path.Join(root, "dir1")); err != nil { + t.Fatal(err) + } + + // Remove a symlink + if err := os.RemoveAll(path.Join(root, "symlink1")); err != nil { + t.Fatal(err) + } + + // Rewrite a file + if err := ioutil.WriteFile(path.Join(root, "file2"), []byte("fileNN\n"), 0777); err != nil { + t.Fatal(err) + } + + // Replace a file + if err := os.RemoveAll(path.Join(root, "file3")); err != nil { + t.Fatal(err) + } + if err := ioutil.WriteFile(path.Join(root, "file3"), []byte("fileMM\n"), 0404); err != nil { + t.Fatal(err) + } + + // Touch file + if err := os.Chtimes(path.Join(root, "file4"), time.Now().Add(time.Second), time.Now().Add(time.Second)); err != nil { + t.Fatal(err) + } + + // Replace file with dir + if err := os.RemoveAll(path.Join(root, "file5")); err != nil { + t.Fatal(err) + } + if err := os.MkdirAll(path.Join(root, "file5"), 0666); err != nil { + t.Fatal(err) + } + + // Create new file + if err := ioutil.WriteFile(path.Join(root, "filenew"), []byte("filenew\n"), 0777); err != nil { + t.Fatal(err) + } + + // Create new dir + if err := os.MkdirAll(path.Join(root, "dirnew"), 0766); err != nil { + t.Fatal(err) + } + + // Create a new symlink + if err := os.Symlink("targetnew", path.Join(root, "symlinknew")); err != nil { + t.Fatal(err) + } + + // Change a symlink + if err := os.RemoveAll(path.Join(root, "symlink2")); err != nil { + t.Fatal(err) + } + if err := os.Symlink("target2change", path.Join(root, "symlink2")); err != nil { + t.Fatal(err) + } + + // Replace dir with file + if err := os.RemoveAll(path.Join(root, "dir2")); err != nil { + t.Fatal(err) + } + if err := ioutil.WriteFile(path.Join(root, "dir2"), []byte("dir2\n"), 0777); err != nil { + t.Fatal(err) + } + + // Touch dir + if err := os.Chtimes(path.Join(root, "dir3"), time.Now().Add(time.Second), time.Now().Add(time.Second)); err != nil { + t.Fatal(err) + } +} + +func TestChangesDirsMutated(t *testing.T) { + src, err := ioutil.TempDir("", "docker-changes-test") + if err != nil { + t.Fatal(err) + } + createSampleDir(t, src) + dst := src + "-copy" + if err := copyDir(src, dst); err != nil { + t.Fatal(err) + } + defer os.RemoveAll(src) + defer os.RemoveAll(dst) + + mutateSampleDir(t, dst) + + changes, err := ChangesDirs(dst, src) + if err != nil { + t.Fatal(err) + } + + sort.Sort(changesByPath(changes)) + + expectedChanges := []Change{ + {"/dir1", ChangeDelete}, + {"/dir2", ChangeModify}, + {"/dirnew", ChangeAdd}, + {"/file1", ChangeDelete}, + {"/file2", ChangeModify}, + {"/file3", ChangeModify}, + {"/file4", ChangeModify}, + {"/file5", ChangeModify}, + {"/filenew", ChangeAdd}, + {"/symlink1", ChangeDelete}, + {"/symlink2", ChangeModify}, + {"/symlinknew", ChangeAdd}, + } + + for i := 0; i < max(len(changes), len(expectedChanges)); i++ { + if i >= len(expectedChanges) { + t.Fatalf("unexpected change %s\n", changes[i].String()) + } + if i >= len(changes) { + t.Fatalf("no change for expected change %s\n", expectedChanges[i].String()) + } + if changes[i].Path == expectedChanges[i].Path { + if changes[i] != expectedChanges[i] { + t.Fatalf("Wrong change for %s, expected %s, got %s\n", changes[i].Path, changes[i].String(), expectedChanges[i].String()) + } + } else if changes[i].Path < expectedChanges[i].Path { + t.Fatalf("unexpected change %s\n", changes[i].String()) + } else { + t.Fatalf("no change for expected change %s != %s\n", expectedChanges[i].String(), changes[i].String()) + } + } +} + +func TestApplyLayer(t *testing.T) { + src, err := ioutil.TempDir("", "docker-changes-test") + if err != nil { + t.Fatal(err) + } + createSampleDir(t, src) + defer os.RemoveAll(src) + dst := src + "-copy" + if err := copyDir(src, dst); err != nil { + t.Fatal(err) + } + mutateSampleDir(t, dst) + defer os.RemoveAll(dst) + + changes, err := ChangesDirs(dst, src) + if err != nil { + t.Fatal(err) + } + + layer, err := ExportChanges(dst, changes) + if err != nil { + t.Fatal(err) + } + + layerCopy, err := NewTempArchive(layer, "") + if err != nil { + t.Fatal(err) + } + + if _, err := ApplyLayer(src, layerCopy); err != nil { + t.Fatal(err) + } + + changes2, err := ChangesDirs(src, dst) + if err != nil { + t.Fatal(err) + } + + if len(changes2) != 0 { + t.Fatalf("Unexpected differences after reapplying mutation: %v", changes2) + } +} + +func TestChangesSizeWithNoChanges(t *testing.T) { + size := ChangesSize("/tmp", nil) + if size != 0 { + t.Fatalf("ChangesSizes with no changes should be 0, was %d", size) + } +} + +func TestChangesSizeWithOnlyDeleteChanges(t *testing.T) { + changes := []Change{ + {Path: "deletedPath", Kind: ChangeDelete}, + } + size := ChangesSize("/tmp", changes) + if size != 0 { + t.Fatalf("ChangesSizes with only delete changes should be 0, was %d", size) + } +} + +func TestChangesSize(t *testing.T) { + parentPath, err := ioutil.TempDir("", "docker-changes-test") + defer os.RemoveAll(parentPath) + addition := path.Join(parentPath, "addition") + if err := ioutil.WriteFile(addition, []byte{0x01, 0x01, 0x01}, 0744); err != nil { + t.Fatal(err) + } + modification := path.Join(parentPath, "modification") + if err = ioutil.WriteFile(modification, []byte{0x01, 0x01, 0x01}, 0744); err != nil { + t.Fatal(err) + } + changes := []Change{ + {Path: "addition", Kind: ChangeAdd}, + {Path: "modification", Kind: ChangeModify}, + } + size := ChangesSize(parentPath, changes) + if size != 6 { + t.Fatalf("ChangesSizes with only delete changes should be 0, was %d", size) + } +} + +func checkChanges(expectedChanges, changes []Change, t *testing.T) { + sort.Sort(changesByPath(expectedChanges)) + sort.Sort(changesByPath(changes)) + for i := 0; i < max(len(changes), len(expectedChanges)); i++ { + if i >= len(expectedChanges) { + t.Fatalf("unexpected change %s\n", changes[i].String()) + } + if i >= len(changes) { + t.Fatalf("no change for expected change %s\n", expectedChanges[i].String()) + } + if changes[i].Path == expectedChanges[i].Path { + if changes[i] != expectedChanges[i] { + t.Fatalf("Wrong change for %s, expected %s, got %s\n", changes[i].Path, changes[i].String(), expectedChanges[i].String()) + } + } else if changes[i].Path < expectedChanges[i].Path { + t.Fatalf("unexpected change %s\n", changes[i].String()) + } else { + t.Fatalf("no change for expected change %s != %s\n", expectedChanges[i].String(), changes[i].String()) + } + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_unix.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_unix.go new file mode 100644 index 0000000000000..dc1ea608bef37 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_unix.go @@ -0,0 +1,27 @@ +// +build !windows + +package archive + +import ( + "syscall" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system" +) + +func statDifferent(oldStat *system.Stat_t, newStat *system.Stat_t) bool { + // Don't look at size for dirs, its not a good measure of change + if oldStat.Mode() != newStat.Mode() || + oldStat.Uid() != newStat.Uid() || + oldStat.Gid() != newStat.Gid() || + oldStat.Rdev() != newStat.Rdev() || + // Don't look at size for dirs, its not a good measure of change + (oldStat.Mode()&syscall.S_IFDIR != syscall.S_IFDIR && + (!sameFsTimeSpec(oldStat.Mtim(), newStat.Mtim()) || (oldStat.Size() != newStat.Size()))) { + return true + } + return false +} + +func (info *FileInfo) isDir() bool { + return info.parent == nil || info.stat.Mode()&syscall.S_IFDIR != 0 +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_windows.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_windows.go new file mode 100644 index 0000000000000..6026575e5c85a --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/changes_windows.go @@ -0,0 +1,20 @@ +package archive + +import ( + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system" +) + +func statDifferent(oldStat *system.Stat_t, newStat *system.Stat_t) bool { + + // Don't look at size for dirs, its not a good measure of change + if oldStat.ModTime() != newStat.ModTime() || + oldStat.Mode() != newStat.Mode() || + oldStat.Size() != newStat.Size() && !oldStat.IsDir() { + return true + } + return false +} + +func (info *FileInfo) isDir() bool { + return info.parent == nil || info.stat.IsDir() +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/copy.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/copy.go new file mode 100644 index 0000000000000..576f336ba15d7 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/copy.go @@ -0,0 +1,308 @@ +package archive + +import ( + "archive/tar" + "errors" + "io" + "io/ioutil" + "os" + "path" + "path/filepath" + "strings" + + "github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus" +) + +// Errors used or returned by this file. +var ( + ErrNotDirectory = errors.New("not a directory") + ErrDirNotExists = errors.New("no such directory") + ErrCannotCopyDir = errors.New("cannot copy directory") + ErrInvalidCopySource = errors.New("invalid copy source content") +) + +// PreserveTrailingDotOrSeparator returns the given cleaned path (after +// processing using any utility functions from the path or filepath stdlib +// packages) and appends a trailing `/.` or `/` if its corresponding original +// path (from before being processed by utility functions from the path or +// filepath stdlib packages) ends with a trailing `/.` or `/`. If the cleaned +// path already ends in a `.` path segment, then another is not added. If the +// clean path already ends in a path separator, then another is not added. +func PreserveTrailingDotOrSeparator(cleanedPath, originalPath string) string { + if !SpecifiesCurrentDir(cleanedPath) && SpecifiesCurrentDir(originalPath) { + if !HasTrailingPathSeparator(cleanedPath) { + // Add a separator if it doesn't already end with one (a cleaned + // path would only end in a separator if it is the root). + cleanedPath += string(filepath.Separator) + } + cleanedPath += "." + } + + if !HasTrailingPathSeparator(cleanedPath) && HasTrailingPathSeparator(originalPath) { + cleanedPath += string(filepath.Separator) + } + + return cleanedPath +} + +// AssertsDirectory returns whether the given path is +// asserted to be a directory, i.e., the path ends with +// a trailing '/' or `/.`, assuming a path separator of `/`. +func AssertsDirectory(path string) bool { + return HasTrailingPathSeparator(path) || SpecifiesCurrentDir(path) +} + +// HasTrailingPathSeparator returns whether the given +// path ends with the system's path separator character. +func HasTrailingPathSeparator(path string) bool { + return len(path) > 0 && os.IsPathSeparator(path[len(path)-1]) +} + +// SpecifiesCurrentDir returns whether the given path specifies +// a "current directory", i.e., the last path segment is `.`. +func SpecifiesCurrentDir(path string) bool { + return filepath.Base(path) == "." +} + +// SplitPathDirEntry splits the given path between its +// parent directory and its basename in that directory. +func SplitPathDirEntry(localizedPath string) (dir, base string) { + normalizedPath := filepath.ToSlash(localizedPath) + vol := filepath.VolumeName(normalizedPath) + normalizedPath = normalizedPath[len(vol):] + + if normalizedPath == "/" { + // Specifies the root path. + return filepath.FromSlash(vol + normalizedPath), "." + } + + trimmedPath := vol + strings.TrimRight(normalizedPath, "/") + + dir = filepath.FromSlash(path.Dir(trimmedPath)) + base = filepath.FromSlash(path.Base(trimmedPath)) + + return dir, base +} + +// TarResource archives the resource at the given sourcePath into a Tar +// archive. A non-nil error is returned if sourcePath does not exist or is +// asserted to be a directory but exists as another type of file. +// +// This function acts as a convenient wrapper around TarWithOptions, which +// requires a directory as the source path. TarResource accepts either a +// directory or a file path and correctly sets the Tar options. +func TarResource(sourcePath string) (content Archive, err error) { + if _, err = os.Lstat(sourcePath); err != nil { + // Catches the case where the source does not exist or is not a + // directory if asserted to be a directory, as this also causes an + // error. + return + } + + if len(sourcePath) > 1 && HasTrailingPathSeparator(sourcePath) { + // In the case where the source path is a symbolic link AND it ends + // with a path separator, we will want to evaluate the symbolic link. + trimmedPath := sourcePath[:len(sourcePath)-1] + stat, err := os.Lstat(trimmedPath) + if err != nil { + return nil, err + } + + if stat.Mode()&os.ModeSymlink != 0 { + if sourcePath, err = filepath.EvalSymlinks(trimmedPath); err != nil { + return nil, err + } + } + } + + // Separate the source path between it's directory and + // the entry in that directory which we are archiving. + sourceDir, sourceBase := SplitPathDirEntry(sourcePath) + + filter := []string{sourceBase} + + logrus.Debugf("copying %q from %q", sourceBase, sourceDir) + + return TarWithOptions(sourceDir, &TarOptions{ + Compression: Uncompressed, + IncludeFiles: filter, + IncludeSourceDir: true, + }) +} + +// CopyInfo holds basic info about the source +// or destination path of a copy operation. +type CopyInfo struct { + Path string + Exists bool + IsDir bool +} + +// CopyInfoStatPath stats the given path to create a CopyInfo +// struct representing that resource. If mustExist is true, then +// it is an error if there is no file or directory at the given path. +func CopyInfoStatPath(path string, mustExist bool) (CopyInfo, error) { + pathInfo := CopyInfo{Path: path} + + fileInfo, err := os.Lstat(path) + + if err == nil { + pathInfo.Exists, pathInfo.IsDir = true, fileInfo.IsDir() + } else if os.IsNotExist(err) && !mustExist { + err = nil + } + + return pathInfo, err +} + +// PrepareArchiveCopy prepares the given srcContent archive, which should +// contain the archived resource described by srcInfo, to the destination +// described by dstInfo. Returns the possibly modified content archive along +// with the path to the destination directory which it should be extracted to. +func PrepareArchiveCopy(srcContent ArchiveReader, srcInfo, dstInfo CopyInfo) (dstDir string, content Archive, err error) { + // Separate the destination path between its directory and base + // components in case the source archive contents need to be rebased. + dstDir, dstBase := SplitPathDirEntry(dstInfo.Path) + _, srcBase := SplitPathDirEntry(srcInfo.Path) + + switch { + case dstInfo.Exists && dstInfo.IsDir: + // The destination exists as a directory. No alteration + // to srcContent is needed as its contents can be + // simply extracted to the destination directory. + return dstInfo.Path, ioutil.NopCloser(srcContent), nil + case dstInfo.Exists && srcInfo.IsDir: + // The destination exists as some type of file and the source + // content is a directory. This is an error condition since + // you cannot copy a directory to an existing file location. + return "", nil, ErrCannotCopyDir + case dstInfo.Exists: + // The destination exists as some type of file and the source content + // is also a file. The source content entry will have to be renamed to + // have a basename which matches the destination path's basename. + return dstDir, rebaseArchiveEntries(srcContent, srcBase, dstBase), nil + case srcInfo.IsDir: + // The destination does not exist and the source content is an archive + // of a directory. The archive should be extracted to the parent of + // the destination path instead, and when it is, the directory that is + // created as a result should take the name of the destination path. + // The source content entries will have to be renamed to have a + // basename which matches the destination path's basename. + return dstDir, rebaseArchiveEntries(srcContent, srcBase, dstBase), nil + case AssertsDirectory(dstInfo.Path): + // The destination does not exist and is asserted to be created as a + // directory, but the source content is not a directory. This is an + // error condition since you cannot create a directory from a file + // source. + return "", nil, ErrDirNotExists + default: + // The last remaining case is when the destination does not exist, is + // not asserted to be a directory, and the source content is not an + // archive of a directory. It this case, the destination file will need + // to be created when the archive is extracted and the source content + // entry will have to be renamed to have a basename which matches the + // destination path's basename. + return dstDir, rebaseArchiveEntries(srcContent, srcBase, dstBase), nil + } + +} + +// rebaseArchiveEntries rewrites the given srcContent archive replacing +// an occurance of oldBase with newBase at the beginning of entry names. +func rebaseArchiveEntries(srcContent ArchiveReader, oldBase, newBase string) Archive { + rebased, w := io.Pipe() + + go func() { + srcTar := tar.NewReader(srcContent) + rebasedTar := tar.NewWriter(w) + + for { + hdr, err := srcTar.Next() + if err == io.EOF { + // Signals end of archive. + rebasedTar.Close() + w.Close() + return + } + if err != nil { + w.CloseWithError(err) + return + } + + hdr.Name = strings.Replace(hdr.Name, oldBase, newBase, 1) + + if err = rebasedTar.WriteHeader(hdr); err != nil { + w.CloseWithError(err) + return + } + + if _, err = io.Copy(rebasedTar, srcTar); err != nil { + w.CloseWithError(err) + return + } + } + }() + + return rebased +} + +// CopyResource performs an archive copy from the given source path to the +// given destination path. The source path MUST exist and the destination +// path's parent directory must exist. +func CopyResource(srcPath, dstPath string) error { + var ( + srcInfo CopyInfo + err error + ) + + // Clean the source and destination paths. + srcPath = PreserveTrailingDotOrSeparator(filepath.Clean(srcPath), srcPath) + dstPath = PreserveTrailingDotOrSeparator(filepath.Clean(dstPath), dstPath) + + if srcInfo, err = CopyInfoStatPath(srcPath, true); err != nil { + return err + } + + content, err := TarResource(srcPath) + if err != nil { + return err + } + defer content.Close() + + return CopyTo(content, srcInfo, dstPath) +} + +// CopyTo handles extracting the given content whose +// entries should be sourced from srcInfo to dstPath. +func CopyTo(content ArchiveReader, srcInfo CopyInfo, dstPath string) error { + dstInfo, err := CopyInfoStatPath(dstPath, false) + if err != nil { + return err + } + + if !dstInfo.Exists { + // Ensure destination parent dir exists. + dstParent, _ := SplitPathDirEntry(dstPath) + + dstStat, err := os.Lstat(dstParent) + if err != nil { + return err + } + if !dstStat.IsDir() { + return ErrNotDirectory + } + } + + dstDir, copyArchive, err := PrepareArchiveCopy(content, srcInfo, dstInfo) + if err != nil { + return err + } + defer copyArchive.Close() + + options := &TarOptions{ + NoLchown: true, + NoOverwriteDirNonDir: true, + } + + return Untar(copyArchive, dstDir, options) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/copy_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/copy_test.go new file mode 100644 index 0000000000000..dd0b323626643 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/copy_test.go @@ -0,0 +1,637 @@ +package archive + +import ( + "bytes" + "crypto/sha256" + "encoding/hex" + "fmt" + "io" + "io/ioutil" + "os" + "path/filepath" + "strings" + "testing" +) + +func removeAllPaths(paths ...string) { + for _, path := range paths { + os.RemoveAll(path) + } +} + +func getTestTempDirs(t *testing.T) (tmpDirA, tmpDirB string) { + var err error + + if tmpDirA, err = ioutil.TempDir("", "archive-copy-test"); err != nil { + t.Fatal(err) + } + + if tmpDirB, err = ioutil.TempDir("", "archive-copy-test"); err != nil { + t.Fatal(err) + } + + return +} + +func isNotDir(err error) bool { + return strings.Contains(err.Error(), "not a directory") +} + +func joinTrailingSep(pathElements ...string) string { + joined := filepath.Join(pathElements...) + + return fmt.Sprintf("%s%c", joined, filepath.Separator) +} + +func fileContentsEqual(t *testing.T, filenameA, filenameB string) (err error) { + t.Logf("checking for equal file contents: %q and %q\n", filenameA, filenameB) + + fileA, err := os.Open(filenameA) + if err != nil { + return + } + defer fileA.Close() + + fileB, err := os.Open(filenameB) + if err != nil { + return + } + defer fileB.Close() + + hasher := sha256.New() + + if _, err = io.Copy(hasher, fileA); err != nil { + return + } + + hashA := hasher.Sum(nil) + hasher.Reset() + + if _, err = io.Copy(hasher, fileB); err != nil { + return + } + + hashB := hasher.Sum(nil) + + if !bytes.Equal(hashA, hashB) { + err = fmt.Errorf("file content hashes not equal - expected %s, got %s", hex.EncodeToString(hashA), hex.EncodeToString(hashB)) + } + + return +} + +func dirContentsEqual(t *testing.T, newDir, oldDir string) (err error) { + t.Logf("checking for equal directory contents: %q and %q\n", newDir, oldDir) + + var changes []Change + + if changes, err = ChangesDirs(newDir, oldDir); err != nil { + return + } + + if len(changes) != 0 { + err = fmt.Errorf("expected no changes between directories, but got: %v", changes) + } + + return +} + +func logDirContents(t *testing.T, dirPath string) { + logWalkedPaths := filepath.WalkFunc(func(path string, info os.FileInfo, err error) error { + if err != nil { + t.Errorf("stat error for path %q: %s", path, err) + return nil + } + + if info.IsDir() { + path = joinTrailingSep(path) + } + + t.Logf("\t%s", path) + + return nil + }) + + t.Logf("logging directory contents: %q", dirPath) + + if err := filepath.Walk(dirPath, logWalkedPaths); err != nil { + t.Fatal(err) + } +} + +func testCopyHelper(t *testing.T, srcPath, dstPath string) (err error) { + t.Logf("copying from %q to %q", srcPath, dstPath) + + return CopyResource(srcPath, dstPath) +} + +// Basic assumptions about SRC and DST: +// 1. SRC must exist. +// 2. If SRC ends with a trailing separator, it must be a directory. +// 3. DST parent directory must exist. +// 4. If DST exists as a file, it must not end with a trailing separator. + +// First get these easy error cases out of the way. + +// Test for error when SRC does not exist. +func TestCopyErrSrcNotExists(t *testing.T) { + tmpDirA, tmpDirB := getTestTempDirs(t) + defer removeAllPaths(tmpDirA, tmpDirB) + + content, err := TarResource(filepath.Join(tmpDirA, "file1")) + if err == nil { + content.Close() + t.Fatal("expected IsNotExist error, but got nil instead") + } + + if !os.IsNotExist(err) { + t.Fatalf("expected IsNotExist error, but got %T: %s", err, err) + } +} + +// Test for error when SRC ends in a trailing +// path separator but it exists as a file. +func TestCopyErrSrcNotDir(t *testing.T) { + tmpDirA, tmpDirB := getTestTempDirs(t) + defer removeAllPaths(tmpDirA, tmpDirB) + + // Load A with some sample files and directories. + createSampleDir(t, tmpDirA) + + content, err := TarResource(joinTrailingSep(tmpDirA, "file1")) + if err == nil { + content.Close() + t.Fatal("expected IsNotDir error, but got nil instead") + } + + if !isNotDir(err) { + t.Fatalf("expected IsNotDir error, but got %T: %s", err, err) + } +} + +// Test for error when SRC is a valid file or directory, +// but the DST parent directory does not exist. +func TestCopyErrDstParentNotExists(t *testing.T) { + tmpDirA, tmpDirB := getTestTempDirs(t) + defer removeAllPaths(tmpDirA, tmpDirB) + + // Load A with some sample files and directories. + createSampleDir(t, tmpDirA) + + srcInfo := CopyInfo{Path: filepath.Join(tmpDirA, "file1"), Exists: true, IsDir: false} + + // Try with a file source. + content, err := TarResource(srcInfo.Path) + if err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + defer content.Close() + + // Copy to a file whose parent does not exist. + if err = CopyTo(content, srcInfo, filepath.Join(tmpDirB, "fakeParentDir", "file1")); err == nil { + t.Fatal("expected IsNotExist error, but got nil instead") + } + + if !os.IsNotExist(err) { + t.Fatalf("expected IsNotExist error, but got %T: %s", err, err) + } + + // Try with a directory source. + srcInfo = CopyInfo{Path: filepath.Join(tmpDirA, "dir1"), Exists: true, IsDir: true} + + content, err = TarResource(srcInfo.Path) + if err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + defer content.Close() + + // Copy to a directory whose parent does not exist. + if err = CopyTo(content, srcInfo, joinTrailingSep(tmpDirB, "fakeParentDir", "fakeDstDir")); err == nil { + t.Fatal("expected IsNotExist error, but got nil instead") + } + + if !os.IsNotExist(err) { + t.Fatalf("expected IsNotExist error, but got %T: %s", err, err) + } +} + +// Test for error when DST ends in a trailing +// path separator but exists as a file. +func TestCopyErrDstNotDir(t *testing.T) { + tmpDirA, tmpDirB := getTestTempDirs(t) + defer removeAllPaths(tmpDirA, tmpDirB) + + // Load A and B with some sample files and directories. + createSampleDir(t, tmpDirA) + createSampleDir(t, tmpDirB) + + // Try with a file source. + srcInfo := CopyInfo{Path: filepath.Join(tmpDirA, "file1"), Exists: true, IsDir: false} + + content, err := TarResource(srcInfo.Path) + if err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + defer content.Close() + + if err = CopyTo(content, srcInfo, joinTrailingSep(tmpDirB, "file1")); err == nil { + t.Fatal("expected IsNotDir error, but got nil instead") + } + + if !isNotDir(err) { + t.Fatalf("expected IsNotDir error, but got %T: %s", err, err) + } + + // Try with a directory source. + srcInfo = CopyInfo{Path: filepath.Join(tmpDirA, "dir1"), Exists: true, IsDir: true} + + content, err = TarResource(srcInfo.Path) + if err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + defer content.Close() + + if err = CopyTo(content, srcInfo, joinTrailingSep(tmpDirB, "file1")); err == nil { + t.Fatal("expected IsNotDir error, but got nil instead") + } + + if !isNotDir(err) { + t.Fatalf("expected IsNotDir error, but got %T: %s", err, err) + } +} + +// Possibilities are reduced to the remaining 10 cases: +// +// case | srcIsDir | onlyDirContents | dstExists | dstIsDir | dstTrSep | action +// =================================================================================================== +// A | no | - | no | - | no | create file +// B | no | - | no | - | yes | error +// C | no | - | yes | no | - | overwrite file +// D | no | - | yes | yes | - | create file in dst dir +// E | yes | no | no | - | - | create dir, copy contents +// F | yes | no | yes | no | - | error +// G | yes | no | yes | yes | - | copy dir and contents +// H | yes | yes | no | - | - | create dir, copy contents +// I | yes | yes | yes | no | - | error +// J | yes | yes | yes | yes | - | copy dir contents +// + +// A. SRC specifies a file and DST (no trailing path separator) doesn't +// exist. This should create a file with the name DST and copy the +// contents of the source file into it. +func TestCopyCaseA(t *testing.T) { + tmpDirA, tmpDirB := getTestTempDirs(t) + defer removeAllPaths(tmpDirA, tmpDirB) + + // Load A with some sample files and directories. + createSampleDir(t, tmpDirA) + + srcPath := filepath.Join(tmpDirA, "file1") + dstPath := filepath.Join(tmpDirB, "itWorks.txt") + + var err error + + if err = testCopyHelper(t, srcPath, dstPath); err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + + if err = fileContentsEqual(t, srcPath, dstPath); err != nil { + t.Fatal(err) + } +} + +// B. SRC specifies a file and DST (with trailing path separator) doesn't +// exist. This should cause an error because the copy operation cannot +// create a directory when copying a single file. +func TestCopyCaseB(t *testing.T) { + tmpDirA, tmpDirB := getTestTempDirs(t) + defer removeAllPaths(tmpDirA, tmpDirB) + + // Load A with some sample files and directories. + createSampleDir(t, tmpDirA) + + srcPath := filepath.Join(tmpDirA, "file1") + dstDir := joinTrailingSep(tmpDirB, "testDir") + + var err error + + if err = testCopyHelper(t, srcPath, dstDir); err == nil { + t.Fatal("expected ErrDirNotExists error, but got nil instead") + } + + if err != ErrDirNotExists { + t.Fatalf("expected ErrDirNotExists error, but got %T: %s", err, err) + } +} + +// C. SRC specifies a file and DST exists as a file. This should overwrite +// the file at DST with the contents of the source file. +func TestCopyCaseC(t *testing.T) { + tmpDirA, tmpDirB := getTestTempDirs(t) + defer removeAllPaths(tmpDirA, tmpDirB) + + // Load A and B with some sample files and directories. + createSampleDir(t, tmpDirA) + createSampleDir(t, tmpDirB) + + srcPath := filepath.Join(tmpDirA, "file1") + dstPath := filepath.Join(tmpDirB, "file2") + + var err error + + // Ensure they start out different. + if err = fileContentsEqual(t, srcPath, dstPath); err == nil { + t.Fatal("expected different file contents") + } + + if err = testCopyHelper(t, srcPath, dstPath); err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + + if err = fileContentsEqual(t, srcPath, dstPath); err != nil { + t.Fatal(err) + } +} + +// D. SRC specifies a file and DST exists as a directory. This should place +// a copy of the source file inside it using the basename from SRC. Ensure +// this works whether DST has a trailing path separator or not. +func TestCopyCaseD(t *testing.T) { + tmpDirA, tmpDirB := getTestTempDirs(t) + defer removeAllPaths(tmpDirA, tmpDirB) + + // Load A and B with some sample files and directories. + createSampleDir(t, tmpDirA) + createSampleDir(t, tmpDirB) + + srcPath := filepath.Join(tmpDirA, "file1") + dstDir := filepath.Join(tmpDirB, "dir1") + dstPath := filepath.Join(dstDir, "file1") + + var err error + + // Ensure that dstPath doesn't exist. + if _, err = os.Stat(dstPath); !os.IsNotExist(err) { + t.Fatalf("did not expect dstPath %q to exist", dstPath) + } + + if err = testCopyHelper(t, srcPath, dstDir); err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + + if err = fileContentsEqual(t, srcPath, dstPath); err != nil { + t.Fatal(err) + } + + // Now try again but using a trailing path separator for dstDir. + + if err = os.RemoveAll(dstDir); err != nil { + t.Fatalf("unable to remove dstDir: %s", err) + } + + if err = os.MkdirAll(dstDir, os.FileMode(0755)); err != nil { + t.Fatalf("unable to make dstDir: %s", err) + } + + dstDir = joinTrailingSep(tmpDirB, "dir1") + + if err = testCopyHelper(t, srcPath, dstDir); err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + + if err = fileContentsEqual(t, srcPath, dstPath); err != nil { + t.Fatal(err) + } +} + +// E. SRC specifies a directory and DST does not exist. This should create a +// directory at DST and copy the contents of the SRC directory into the DST +// directory. Ensure this works whether DST has a trailing path separator or +// not. +func TestCopyCaseE(t *testing.T) { + tmpDirA, tmpDirB := getTestTempDirs(t) + defer removeAllPaths(tmpDirA, tmpDirB) + + // Load A with some sample files and directories. + createSampleDir(t, tmpDirA) + + srcDir := filepath.Join(tmpDirA, "dir1") + dstDir := filepath.Join(tmpDirB, "testDir") + + var err error + + if err = testCopyHelper(t, srcDir, dstDir); err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + + if err = dirContentsEqual(t, dstDir, srcDir); err != nil { + t.Log("dir contents not equal") + logDirContents(t, tmpDirA) + logDirContents(t, tmpDirB) + t.Fatal(err) + } + + // Now try again but using a trailing path separator for dstDir. + + if err = os.RemoveAll(dstDir); err != nil { + t.Fatalf("unable to remove dstDir: %s", err) + } + + dstDir = joinTrailingSep(tmpDirB, "testDir") + + if err = testCopyHelper(t, srcDir, dstDir); err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + + if err = dirContentsEqual(t, dstDir, srcDir); err != nil { + t.Fatal(err) + } +} + +// F. SRC specifies a directory and DST exists as a file. This should cause an +// error as it is not possible to overwrite a file with a directory. +func TestCopyCaseF(t *testing.T) { + tmpDirA, tmpDirB := getTestTempDirs(t) + defer removeAllPaths(tmpDirA, tmpDirB) + + // Load A and B with some sample files and directories. + createSampleDir(t, tmpDirA) + createSampleDir(t, tmpDirB) + + srcDir := filepath.Join(tmpDirA, "dir1") + dstFile := filepath.Join(tmpDirB, "file1") + + var err error + + if err = testCopyHelper(t, srcDir, dstFile); err == nil { + t.Fatal("expected ErrCannotCopyDir error, but got nil instead") + } + + if err != ErrCannotCopyDir { + t.Fatalf("expected ErrCannotCopyDir error, but got %T: %s", err, err) + } +} + +// G. SRC specifies a directory and DST exists as a directory. This should copy +// the SRC directory and all its contents to the DST directory. Ensure this +// works whether DST has a trailing path separator or not. +func TestCopyCaseG(t *testing.T) { + tmpDirA, tmpDirB := getTestTempDirs(t) + defer removeAllPaths(tmpDirA, tmpDirB) + + // Load A and B with some sample files and directories. + createSampleDir(t, tmpDirA) + createSampleDir(t, tmpDirB) + + srcDir := filepath.Join(tmpDirA, "dir1") + dstDir := filepath.Join(tmpDirB, "dir2") + resultDir := filepath.Join(dstDir, "dir1") + + var err error + + if err = testCopyHelper(t, srcDir, dstDir); err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + + if err = dirContentsEqual(t, resultDir, srcDir); err != nil { + t.Fatal(err) + } + + // Now try again but using a trailing path separator for dstDir. + + if err = os.RemoveAll(dstDir); err != nil { + t.Fatalf("unable to remove dstDir: %s", err) + } + + if err = os.MkdirAll(dstDir, os.FileMode(0755)); err != nil { + t.Fatalf("unable to make dstDir: %s", err) + } + + dstDir = joinTrailingSep(tmpDirB, "dir2") + + if err = testCopyHelper(t, srcDir, dstDir); err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + + if err = dirContentsEqual(t, resultDir, srcDir); err != nil { + t.Fatal(err) + } +} + +// H. SRC specifies a directory's contents only and DST does not exist. This +// should create a directory at DST and copy the contents of the SRC +// directory (but not the directory itself) into the DST directory. Ensure +// this works whether DST has a trailing path separator or not. +func TestCopyCaseH(t *testing.T) { + tmpDirA, tmpDirB := getTestTempDirs(t) + defer removeAllPaths(tmpDirA, tmpDirB) + + // Load A with some sample files and directories. + createSampleDir(t, tmpDirA) + + srcDir := joinTrailingSep(tmpDirA, "dir1") + "." + dstDir := filepath.Join(tmpDirB, "testDir") + + var err error + + if err = testCopyHelper(t, srcDir, dstDir); err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + + if err = dirContentsEqual(t, dstDir, srcDir); err != nil { + t.Log("dir contents not equal") + logDirContents(t, tmpDirA) + logDirContents(t, tmpDirB) + t.Fatal(err) + } + + // Now try again but using a trailing path separator for dstDir. + + if err = os.RemoveAll(dstDir); err != nil { + t.Fatalf("unable to remove dstDir: %s", err) + } + + dstDir = joinTrailingSep(tmpDirB, "testDir") + + if err = testCopyHelper(t, srcDir, dstDir); err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + + if err = dirContentsEqual(t, dstDir, srcDir); err != nil { + t.Log("dir contents not equal") + logDirContents(t, tmpDirA) + logDirContents(t, tmpDirB) + t.Fatal(err) + } +} + +// I. SRC specifies a directory's contents only and DST exists as a file. This +// should cause an error as it is not possible to overwrite a file with a +// directory. +func TestCopyCaseI(t *testing.T) { + tmpDirA, tmpDirB := getTestTempDirs(t) + defer removeAllPaths(tmpDirA, tmpDirB) + + // Load A and B with some sample files and directories. + createSampleDir(t, tmpDirA) + createSampleDir(t, tmpDirB) + + srcDir := joinTrailingSep(tmpDirA, "dir1") + "." + dstFile := filepath.Join(tmpDirB, "file1") + + var err error + + if err = testCopyHelper(t, srcDir, dstFile); err == nil { + t.Fatal("expected ErrCannotCopyDir error, but got nil instead") + } + + if err != ErrCannotCopyDir { + t.Fatalf("expected ErrCannotCopyDir error, but got %T: %s", err, err) + } +} + +// J. SRC specifies a directory's contents only and DST exists as a directory. +// This should copy the contents of the SRC directory (but not the directory +// itself) into the DST directory. Ensure this works whether DST has a +// trailing path separator or not. +func TestCopyCaseJ(t *testing.T) { + tmpDirA, tmpDirB := getTestTempDirs(t) + defer removeAllPaths(tmpDirA, tmpDirB) + + // Load A and B with some sample files and directories. + createSampleDir(t, tmpDirA) + createSampleDir(t, tmpDirB) + + srcDir := joinTrailingSep(tmpDirA, "dir1") + "." + dstDir := filepath.Join(tmpDirB, "dir5") + + var err error + + if err = testCopyHelper(t, srcDir, dstDir); err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + + if err = dirContentsEqual(t, dstDir, srcDir); err != nil { + t.Fatal(err) + } + + // Now try again but using a trailing path separator for dstDir. + + if err = os.RemoveAll(dstDir); err != nil { + t.Fatalf("unable to remove dstDir: %s", err) + } + + if err = os.MkdirAll(dstDir, os.FileMode(0755)); err != nil { + t.Fatalf("unable to make dstDir: %s", err) + } + + dstDir = joinTrailingSep(tmpDirB, "dir5") + + if err = testCopyHelper(t, srcDir, dstDir); err != nil { + t.Fatalf("unexpected error %T: %s", err, err) + } + + if err = dirContentsEqual(t, dstDir, srcDir); err != nil { + t.Fatal(err) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/copy_unix.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/copy_unix.go new file mode 100644 index 0000000000000..e305b5e4af911 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/copy_unix.go @@ -0,0 +1,11 @@ +// +build !windows + +package archive + +import ( + "path/filepath" +) + +func normalizePath(path string) string { + return filepath.ToSlash(path) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/copy_windows.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/copy_windows.go new file mode 100644 index 0000000000000..2b775b45c4f11 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/copy_windows.go @@ -0,0 +1,9 @@ +package archive + +import ( + "path/filepath" +) + +func normalizePath(path string) string { + return filepath.FromSlash(path) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/diff.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/diff.go new file mode 100644 index 0000000000000..10a63a051bbf9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/diff.go @@ -0,0 +1,210 @@ +package archive + +import ( + "archive/tar" + "fmt" + "io" + "io/ioutil" + "os" + "path/filepath" + "runtime" + "strings" + "syscall" + + "github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus" + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/pools" + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system" +) + +func UnpackLayer(dest string, layer ArchiveReader) (size int64, err error) { + tr := tar.NewReader(layer) + trBuf := pools.BufioReader32KPool.Get(tr) + defer pools.BufioReader32KPool.Put(trBuf) + + var dirs []*tar.Header + + aufsTempdir := "" + aufsHardlinks := make(map[string]*tar.Header) + + // Iterate through the files in the archive. + for { + hdr, err := tr.Next() + if err == io.EOF { + // end of tar archive + break + } + if err != nil { + return 0, err + } + + size += hdr.Size + + // Normalize name, for safety and for a simple is-root check + hdr.Name = filepath.Clean(hdr.Name) + + // Windows does not support filenames with colons in them. Ignore + // these files. This is not a problem though (although it might + // appear that it is). Let's suppose a client is running docker pull. + // The daemon it points to is Windows. Would it make sense for the + // client to be doing a docker pull Ubuntu for example (which has files + // with colons in the name under /usr/share/man/man3)? No, absolutely + // not as it would really only make sense that they were pulling a + // Windows image. However, for development, it is necessary to be able + // to pull Linux images which are in the repository. + // + // TODO Windows. Once the registry is aware of what images are Windows- + // specific or Linux-specific, this warning should be changed to an error + // to cater for the situation where someone does manage to upload a Linux + // image but have it tagged as Windows inadvertantly. + if runtime.GOOS == "windows" { + if strings.Contains(hdr.Name, ":") { + logrus.Warnf("Windows: Ignoring %s (is this a Linux image?)", hdr.Name) + continue + } + } + + // Note as these operations are platform specific, so must the slash be. + if !strings.HasSuffix(hdr.Name, string(os.PathSeparator)) { + // Not the root directory, ensure that the parent directory exists. + // This happened in some tests where an image had a tarfile without any + // parent directories. + parent := filepath.Dir(hdr.Name) + parentPath := filepath.Join(dest, parent) + + if _, err := os.Lstat(parentPath); err != nil && os.IsNotExist(err) { + err = system.MkdirAll(parentPath, 0600) + if err != nil { + return 0, err + } + } + } + + // Skip AUFS metadata dirs + if strings.HasPrefix(hdr.Name, ".wh..wh.") { + // Regular files inside /.wh..wh.plnk can be used as hardlink targets + // We don't want this directory, but we need the files in them so that + // such hardlinks can be resolved. + if strings.HasPrefix(hdr.Name, ".wh..wh.plnk") && hdr.Typeflag == tar.TypeReg { + basename := filepath.Base(hdr.Name) + aufsHardlinks[basename] = hdr + if aufsTempdir == "" { + if aufsTempdir, err = ioutil.TempDir("", "dockerplnk"); err != nil { + return 0, err + } + defer os.RemoveAll(aufsTempdir) + } + if err := createTarFile(filepath.Join(aufsTempdir, basename), dest, hdr, tr, true, nil); err != nil { + return 0, err + } + } + continue + } + path := filepath.Join(dest, hdr.Name) + rel, err := filepath.Rel(dest, path) + if err != nil { + return 0, err + } + + // Note as these operations are platform specific, so must the slash be. + if strings.HasPrefix(rel, ".."+string(os.PathSeparator)) { + return 0, breakoutError(fmt.Errorf("%q is outside of %q", hdr.Name, dest)) + } + base := filepath.Base(path) + + if strings.HasPrefix(base, ".wh.") { + originalBase := base[len(".wh."):] + originalPath := filepath.Join(filepath.Dir(path), originalBase) + if err := os.RemoveAll(originalPath); err != nil { + return 0, err + } + } else { + // If path exits we almost always just want to remove and replace it. + // The only exception is when it is a directory *and* the file from + // the layer is also a directory. Then we want to merge them (i.e. + // just apply the metadata from the layer). + if fi, err := os.Lstat(path); err == nil { + if !(fi.IsDir() && hdr.Typeflag == tar.TypeDir) { + if err := os.RemoveAll(path); err != nil { + return 0, err + } + } + } + + trBuf.Reset(tr) + srcData := io.Reader(trBuf) + srcHdr := hdr + + // Hard links into /.wh..wh.plnk don't work, as we don't extract that directory, so + // we manually retarget these into the temporary files we extracted them into + if hdr.Typeflag == tar.TypeLink && strings.HasPrefix(filepath.Clean(hdr.Linkname), ".wh..wh.plnk") { + linkBasename := filepath.Base(hdr.Linkname) + srcHdr = aufsHardlinks[linkBasename] + if srcHdr == nil { + return 0, fmt.Errorf("Invalid aufs hardlink") + } + tmpFile, err := os.Open(filepath.Join(aufsTempdir, linkBasename)) + if err != nil { + return 0, err + } + defer tmpFile.Close() + srcData = tmpFile + } + + if err := createTarFile(path, dest, srcHdr, srcData, true, nil); err != nil { + return 0, err + } + + // Directory mtimes must be handled at the end to avoid further + // file creation in them to modify the directory mtime + if hdr.Typeflag == tar.TypeDir { + dirs = append(dirs, hdr) + } + } + } + + for _, hdr := range dirs { + path := filepath.Join(dest, hdr.Name) + ts := []syscall.Timespec{timeToTimespec(hdr.AccessTime), timeToTimespec(hdr.ModTime)} + if err := syscall.UtimesNano(path, ts); err != nil { + return 0, err + } + } + + return size, nil +} + +// ApplyLayer parses a diff in the standard layer format from `layer`, +// and applies it to the directory `dest`. The stream `layer` can be +// compressed or uncompressed. +// Returns the size in bytes of the contents of the layer. +func ApplyLayer(dest string, layer ArchiveReader) (int64, error) { + return applyLayerHandler(dest, layer, true) +} + +// ApplyUncompressedLayer parses a diff in the standard layer format from +// `layer`, and applies it to the directory `dest`. The stream `layer` +// can only be uncompressed. +// Returns the size in bytes of the contents of the layer. +func ApplyUncompressedLayer(dest string, layer ArchiveReader) (int64, error) { + return applyLayerHandler(dest, layer, false) +} + +// do the bulk load of ApplyLayer, but allow for not calling DecompressStream +func applyLayerHandler(dest string, layer ArchiveReader, decompress bool) (int64, error) { + dest = filepath.Clean(dest) + + // We need to be able to set any perms + oldmask, err := system.Umask(0) + if err != nil { + return 0, err + } + defer system.Umask(oldmask) // ignore err, ErrNotSupportedPlatform + + if decompress { + layer, err = DecompressStream(layer) + if err != nil { + return 0, err + } + } + return UnpackLayer(dest, layer) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/diff_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/diff_test.go new file mode 100644 index 0000000000000..01ed4372805ba --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/diff_test.go @@ -0,0 +1,190 @@ +package archive + +import ( + "archive/tar" + "testing" +) + +func TestApplyLayerInvalidFilenames(t *testing.T) { + for i, headers := range [][]*tar.Header{ + { + { + Name: "../victim/dotdot", + Typeflag: tar.TypeReg, + Mode: 0644, + }, + }, + { + { + // Note the leading slash + Name: "/../victim/slash-dotdot", + Typeflag: tar.TypeReg, + Mode: 0644, + }, + }, + } { + if err := testBreakout("applylayer", "docker-TestApplyLayerInvalidFilenames", headers); err != nil { + t.Fatalf("i=%d. %v", i, err) + } + } +} + +func TestApplyLayerInvalidHardlink(t *testing.T) { + for i, headers := range [][]*tar.Header{ + { // try reading victim/hello (../) + { + Name: "dotdot", + Typeflag: tar.TypeLink, + Linkname: "../victim/hello", + Mode: 0644, + }, + }, + { // try reading victim/hello (/../) + { + Name: "slash-dotdot", + Typeflag: tar.TypeLink, + // Note the leading slash + Linkname: "/../victim/hello", + Mode: 0644, + }, + }, + { // try writing victim/file + { + Name: "loophole-victim", + Typeflag: tar.TypeLink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "loophole-victim/file", + Typeflag: tar.TypeReg, + Mode: 0644, + }, + }, + { // try reading victim/hello (hardlink, symlink) + { + Name: "loophole-victim", + Typeflag: tar.TypeLink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "symlink", + Typeflag: tar.TypeSymlink, + Linkname: "loophole-victim/hello", + Mode: 0644, + }, + }, + { // Try reading victim/hello (hardlink, hardlink) + { + Name: "loophole-victim", + Typeflag: tar.TypeLink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "hardlink", + Typeflag: tar.TypeLink, + Linkname: "loophole-victim/hello", + Mode: 0644, + }, + }, + { // Try removing victim directory (hardlink) + { + Name: "loophole-victim", + Typeflag: tar.TypeLink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "loophole-victim", + Typeflag: tar.TypeReg, + Mode: 0644, + }, + }, + } { + if err := testBreakout("applylayer", "docker-TestApplyLayerInvalidHardlink", headers); err != nil { + t.Fatalf("i=%d. %v", i, err) + } + } +} + +func TestApplyLayerInvalidSymlink(t *testing.T) { + for i, headers := range [][]*tar.Header{ + { // try reading victim/hello (../) + { + Name: "dotdot", + Typeflag: tar.TypeSymlink, + Linkname: "../victim/hello", + Mode: 0644, + }, + }, + { // try reading victim/hello (/../) + { + Name: "slash-dotdot", + Typeflag: tar.TypeSymlink, + // Note the leading slash + Linkname: "/../victim/hello", + Mode: 0644, + }, + }, + { // try writing victim/file + { + Name: "loophole-victim", + Typeflag: tar.TypeSymlink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "loophole-victim/file", + Typeflag: tar.TypeReg, + Mode: 0644, + }, + }, + { // try reading victim/hello (symlink, symlink) + { + Name: "loophole-victim", + Typeflag: tar.TypeSymlink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "symlink", + Typeflag: tar.TypeSymlink, + Linkname: "loophole-victim/hello", + Mode: 0644, + }, + }, + { // try reading victim/hello (symlink, hardlink) + { + Name: "loophole-victim", + Typeflag: tar.TypeSymlink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "hardlink", + Typeflag: tar.TypeLink, + Linkname: "loophole-victim/hello", + Mode: 0644, + }, + }, + { // try removing victim directory (symlink) + { + Name: "loophole-victim", + Typeflag: tar.TypeSymlink, + Linkname: "../victim", + Mode: 0755, + }, + { + Name: "loophole-victim", + Typeflag: tar.TypeReg, + Mode: 0644, + }, + }, + } { + if err := testBreakout("applylayer", "docker-TestApplyLayerInvalidSymlink", headers); err != nil { + t.Fatalf("i=%d. %v", i, err) + } + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/example_changes.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/example_changes.go new file mode 100644 index 0000000000000..a5e08e4ee96c7 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/example_changes.go @@ -0,0 +1,97 @@ +// +build ignore + +// Simple tool to create an archive stream from an old and new directory +// +// By default it will stream the comparison of two temporary directories with junk files +package main + +import ( + "flag" + "fmt" + "io" + "io/ioutil" + "os" + "path" + + "github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus" + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive" +) + +var ( + flDebug = flag.Bool("D", false, "debugging output") + flNewDir = flag.String("newdir", "", "") + flOldDir = flag.String("olddir", "", "") + log = logrus.New() +) + +func main() { + flag.Usage = func() { + fmt.Println("Produce a tar from comparing two directory paths. By default a demo tar is created of around 200 files (including hardlinks)") + fmt.Printf("%s [OPTIONS]\n", os.Args[0]) + flag.PrintDefaults() + } + flag.Parse() + log.Out = os.Stderr + if (len(os.Getenv("DEBUG")) > 0) || *flDebug { + logrus.SetLevel(logrus.DebugLevel) + } + var newDir, oldDir string + + if len(*flNewDir) == 0 { + var err error + newDir, err = ioutil.TempDir("", "docker-test-newDir") + if err != nil { + log.Fatal(err) + } + defer os.RemoveAll(newDir) + if _, err := prepareUntarSourceDirectory(100, newDir, true); err != nil { + log.Fatal(err) + } + } else { + newDir = *flNewDir + } + + if len(*flOldDir) == 0 { + oldDir, err := ioutil.TempDir("", "docker-test-oldDir") + if err != nil { + log.Fatal(err) + } + defer os.RemoveAll(oldDir) + } else { + oldDir = *flOldDir + } + + changes, err := archive.ChangesDirs(newDir, oldDir) + if err != nil { + log.Fatal(err) + } + + a, err := archive.ExportChanges(newDir, changes) + if err != nil { + log.Fatal(err) + } + defer a.Close() + + i, err := io.Copy(os.Stdout, a) + if err != nil && err != io.EOF { + log.Fatal(err) + } + fmt.Fprintf(os.Stderr, "wrote archive of %d bytes", i) +} + +func prepareUntarSourceDirectory(numberOfFiles int, targetPath string, makeLinks bool) (int, error) { + fileData := []byte("fooo") + for n := 0; n < numberOfFiles; n++ { + fileName := fmt.Sprintf("file-%d", n) + if err := ioutil.WriteFile(path.Join(targetPath, fileName), fileData, 0700); err != nil { + return 0, err + } + if makeLinks { + if err := os.Link(path.Join(targetPath, fileName), path.Join(targetPath, fileName+"-link")); err != nil { + return 0, err + } + } + } + totalSize := numberOfFiles * len(fileData) + return totalSize, nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/time_linux.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/time_linux.go new file mode 100644 index 0000000000000..3448569b1ebb2 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/time_linux.go @@ -0,0 +1,16 @@ +package archive + +import ( + "syscall" + "time" +) + +func timeToTimespec(time time.Time) (ts syscall.Timespec) { + if time.IsZero() { + // Return UTIME_OMIT special value + ts.Sec = 0 + ts.Nsec = ((1 << 30) - 2) + return + } + return syscall.NsecToTimespec(time.UnixNano()) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/time_unsupported.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/time_unsupported.go new file mode 100644 index 0000000000000..e85aac0540808 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/time_unsupported.go @@ -0,0 +1,16 @@ +// +build !linux + +package archive + +import ( + "syscall" + "time" +) + +func timeToTimespec(time time.Time) (ts syscall.Timespec) { + nsec := int64(0) + if !time.IsZero() { + nsec = time.UnixNano() + } + return syscall.NsecToTimespec(nsec) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/utils_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/utils_test.go new file mode 100644 index 0000000000000..f5cacea8f0e9f --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/utils_test.go @@ -0,0 +1,166 @@ +package archive + +import ( + "archive/tar" + "bytes" + "fmt" + "io" + "io/ioutil" + "os" + "path/filepath" + "time" +) + +var testUntarFns = map[string]func(string, io.Reader) error{ + "untar": func(dest string, r io.Reader) error { + return Untar(r, dest, nil) + }, + "applylayer": func(dest string, r io.Reader) error { + _, err := ApplyLayer(dest, ArchiveReader(r)) + return err + }, +} + +// testBreakout is a helper function that, within the provided `tmpdir` directory, +// creates a `victim` folder with a generated `hello` file in it. +// `untar` extracts to a directory named `dest`, the tar file created from `headers`. +// +// Here are the tested scenarios: +// - removed `victim` folder (write) +// - removed files from `victim` folder (write) +// - new files in `victim` folder (write) +// - modified files in `victim` folder (write) +// - file in `dest` with same content as `victim/hello` (read) +// +// When using testBreakout make sure you cover one of the scenarios listed above. +func testBreakout(untarFn string, tmpdir string, headers []*tar.Header) error { + tmpdir, err := ioutil.TempDir("", tmpdir) + if err != nil { + return err + } + defer os.RemoveAll(tmpdir) + + dest := filepath.Join(tmpdir, "dest") + if err := os.Mkdir(dest, 0755); err != nil { + return err + } + + victim := filepath.Join(tmpdir, "victim") + if err := os.Mkdir(victim, 0755); err != nil { + return err + } + hello := filepath.Join(victim, "hello") + helloData, err := time.Now().MarshalText() + if err != nil { + return err + } + if err := ioutil.WriteFile(hello, helloData, 0644); err != nil { + return err + } + helloStat, err := os.Stat(hello) + if err != nil { + return err + } + + reader, writer := io.Pipe() + go func() { + t := tar.NewWriter(writer) + for _, hdr := range headers { + t.WriteHeader(hdr) + } + t.Close() + }() + + untar := testUntarFns[untarFn] + if untar == nil { + return fmt.Errorf("could not find untar function %q in testUntarFns", untarFn) + } + if err := untar(dest, reader); err != nil { + if _, ok := err.(breakoutError); !ok { + // If untar returns an error unrelated to an archive breakout, + // then consider this an unexpected error and abort. + return err + } + // Here, untar detected the breakout. + // Let's move on verifying that indeed there was no breakout. + fmt.Printf("breakoutError: %v\n", err) + } + + // Check victim folder + f, err := os.Open(victim) + if err != nil { + // codepath taken if victim folder was removed + return fmt.Errorf("archive breakout: error reading %q: %v", victim, err) + } + defer f.Close() + + // Check contents of victim folder + // + // We are only interested in getting 2 files from the victim folder, because if all is well + // we expect only one result, the `hello` file. If there is a second result, it cannot + // hold the same name `hello` and we assume that a new file got created in the victim folder. + // That is enough to detect an archive breakout. + names, err := f.Readdirnames(2) + if err != nil { + // codepath taken if victim is not a folder + return fmt.Errorf("archive breakout: error reading directory content of %q: %v", victim, err) + } + for _, name := range names { + if name != "hello" { + // codepath taken if new file was created in victim folder + return fmt.Errorf("archive breakout: new file %q", name) + } + } + + // Check victim/hello + f, err = os.Open(hello) + if err != nil { + // codepath taken if read permissions were removed + return fmt.Errorf("archive breakout: could not lstat %q: %v", hello, err) + } + defer f.Close() + b, err := ioutil.ReadAll(f) + if err != nil { + return err + } + fi, err := f.Stat() + if err != nil { + return err + } + if helloStat.IsDir() != fi.IsDir() || + // TODO: cannot check for fi.ModTime() change + helloStat.Mode() != fi.Mode() || + helloStat.Size() != fi.Size() || + !bytes.Equal(helloData, b) { + // codepath taken if hello has been modified + return fmt.Errorf("archive breakout: file %q has been modified. Contents: expected=%q, got=%q. FileInfo: expected=%#v, got=%#v", hello, helloData, b, helloStat, fi) + } + + // Check that nothing in dest/ has the same content as victim/hello. + // Since victim/hello was generated with time.Now(), it is safe to assume + // that any file whose content matches exactly victim/hello, managed somehow + // to access victim/hello. + return filepath.Walk(dest, func(path string, info os.FileInfo, err error) error { + if info.IsDir() { + if err != nil { + // skip directory if error + return filepath.SkipDir + } + // enter directory + return nil + } + if err != nil { + // skip file if error + return nil + } + b, err := ioutil.ReadFile(path) + if err != nil { + // Houston, we have a problem. Aborting (space)walk. + return err + } + if bytes.Equal(helloData, b) { + return fmt.Errorf("archive breakout: file %q has been accessed via %q", hello, path) + } + return nil + }) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/wrap.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/wrap.go new file mode 100644 index 0000000000000..dfb335c0b6c04 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/wrap.go @@ -0,0 +1,59 @@ +package archive + +import ( + "archive/tar" + "bytes" + "io/ioutil" +) + +// Generate generates a new archive from the content provided +// as input. +// +// `files` is a sequence of path/content pairs. A new file is +// added to the archive for each pair. +// If the last pair is incomplete, the file is created with an +// empty content. For example: +// +// Generate("foo.txt", "hello world", "emptyfile") +// +// The above call will return an archive with 2 files: +// * ./foo.txt with content "hello world" +// * ./empty with empty content +// +// FIXME: stream content instead of buffering +// FIXME: specify permissions and other archive metadata +func Generate(input ...string) (Archive, error) { + files := parseStringPairs(input...) + buf := new(bytes.Buffer) + tw := tar.NewWriter(buf) + for _, file := range files { + name, content := file[0], file[1] + hdr := &tar.Header{ + Name: name, + Size: int64(len(content)), + } + if err := tw.WriteHeader(hdr); err != nil { + return nil, err + } + if _, err := tw.Write([]byte(content)); err != nil { + return nil, err + } + } + if err := tw.Close(); err != nil { + return nil, err + } + return ioutil.NopCloser(buf), nil +} + +func parseStringPairs(input ...string) (output [][2]string) { + output = make([][2]string, 0, len(input)/2+1) + for i := 0; i < len(input); i += 2 { + var pair [2]string + pair[0] = input[i] + if i+1 < len(input) { + pair[1] = input[i+1] + } + output = append(output, pair) + } + return +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/wrap_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/wrap_test.go new file mode 100644 index 0000000000000..46ab36697a75b --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive/wrap_test.go @@ -0,0 +1,98 @@ +package archive + +import ( + "archive/tar" + "bytes" + "io" + "testing" +) + +func TestGenerateEmptyFile(t *testing.T) { + archive, err := Generate("emptyFile") + if err != nil { + t.Fatal(err) + } + if archive == nil { + t.Fatal("The generated archive should not be nil.") + } + + expectedFiles := [][]string{ + {"emptyFile", ""}, + } + + tr := tar.NewReader(archive) + actualFiles := make([][]string, 0, 10) + i := 0 + for { + hdr, err := tr.Next() + if err == io.EOF { + break + } + if err != nil { + t.Fatal(err) + } + buf := new(bytes.Buffer) + buf.ReadFrom(tr) + content := buf.String() + actualFiles = append(actualFiles, []string{hdr.Name, content}) + i++ + } + if len(actualFiles) != len(expectedFiles) { + t.Fatalf("Number of expected file %d, got %d.", len(expectedFiles), len(actualFiles)) + } + for i := 0; i < len(expectedFiles); i++ { + actual := actualFiles[i] + expected := expectedFiles[i] + if actual[0] != expected[0] { + t.Fatalf("Expected name '%s', Actual name '%s'", expected[0], actual[0]) + } + if actual[1] != expected[1] { + t.Fatalf("Expected content '%s', Actual content '%s'", expected[1], actual[1]) + } + } +} + +func TestGenerateWithContent(t *testing.T) { + archive, err := Generate("file", "content") + if err != nil { + t.Fatal(err) + } + if archive == nil { + t.Fatal("The generated archive should not be nil.") + } + + expectedFiles := [][]string{ + {"file", "content"}, + } + + tr := tar.NewReader(archive) + actualFiles := make([][]string, 0, 10) + i := 0 + for { + hdr, err := tr.Next() + if err == io.EOF { + break + } + if err != nil { + t.Fatal(err) + } + buf := new(bytes.Buffer) + buf.ReadFrom(tr) + content := buf.String() + actualFiles = append(actualFiles, []string{hdr.Name, content}) + i++ + } + if len(actualFiles) != len(expectedFiles) { + t.Fatalf("Number of expected file %d, got %d.", len(expectedFiles), len(actualFiles)) + } + for i := 0; i < len(expectedFiles); i++ { + actual := actualFiles[i] + expected := expectedFiles[i] + if actual[0] != expected[0] { + t.Fatalf("Expected name '%s', Actual name '%s'", expected[0], actual[0]) + } + if actual[1] != expected[1] { + t.Fatalf("Expected content '%s', Actual content '%s'", expected[1], actual[1]) + } + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils/fileutils.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils/fileutils.go new file mode 100644 index 0000000000000..1b8cadc63ff77 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils/fileutils.go @@ -0,0 +1,196 @@ +package fileutils + +import ( + "errors" + "fmt" + "io" + "io/ioutil" + "os" + "path/filepath" + "strings" + + "github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus" +) + +// exclusion return true if the specified pattern is an exclusion +func exclusion(pattern string) bool { + return pattern[0] == '!' +} + +// empty return true if the specified pattern is empty +func empty(pattern string) bool { + return pattern == "" +} + +// CleanPatterns takes a slice of patterns returns a new +// slice of patterns cleaned with filepath.Clean, stripped +// of any empty patterns and lets the caller know whether the +// slice contains any exception patterns (prefixed with !). +func CleanPatterns(patterns []string) ([]string, [][]string, bool, error) { + // Loop over exclusion patterns and: + // 1. Clean them up. + // 2. Indicate whether we are dealing with any exception rules. + // 3. Error if we see a single exclusion marker on it's own (!). + cleanedPatterns := []string{} + patternDirs := [][]string{} + exceptions := false + for _, pattern := range patterns { + // Eliminate leading and trailing whitespace. + pattern = strings.TrimSpace(pattern) + if empty(pattern) { + continue + } + if exclusion(pattern) { + if len(pattern) == 1 { + return nil, nil, false, errors.New("Illegal exclusion pattern: !") + } + exceptions = true + } + pattern = filepath.Clean(pattern) + cleanedPatterns = append(cleanedPatterns, pattern) + if exclusion(pattern) { + pattern = pattern[1:] + } + patternDirs = append(patternDirs, strings.Split(pattern, "/")) + } + + return cleanedPatterns, patternDirs, exceptions, nil +} + +// Matches returns true if file matches any of the patterns +// and isn't excluded by any of the subsequent patterns. +func Matches(file string, patterns []string) (bool, error) { + file = filepath.Clean(file) + + if file == "." { + // Don't let them exclude everything, kind of silly. + return false, nil + } + + patterns, patDirs, _, err := CleanPatterns(patterns) + if err != nil { + return false, err + } + + return OptimizedMatches(file, patterns, patDirs) +} + +// OptimizedMatches is basically the same as fileutils.Matches() but optimized for archive.go. +// It will assume that the inputs have been preprocessed and therefore the function +// doen't need to do as much error checking and clean-up. This was done to avoid +// repeating these steps on each file being checked during the archive process. +// The more generic fileutils.Matches() can't make these assumptions. +func OptimizedMatches(file string, patterns []string, patDirs [][]string) (bool, error) { + matched := false + parentPath := filepath.Dir(file) + parentPathDirs := strings.Split(parentPath, "/") + + for i, pattern := range patterns { + negative := false + + if exclusion(pattern) { + negative = true + pattern = pattern[1:] + } + + match, err := filepath.Match(pattern, file) + if err != nil { + return false, err + } + + if !match && parentPath != "." { + // Check to see if the pattern matches one of our parent dirs. + if len(patDirs[i]) <= len(parentPathDirs) { + match, _ = filepath.Match(strings.Join(patDirs[i], "/"), + strings.Join(parentPathDirs[:len(patDirs[i])], "/")) + } + } + + if match { + matched = !negative + } + } + + if matched { + logrus.Debugf("Skipping excluded path: %s", file) + } + + return matched, nil +} + +// CopyFile copies from src to dst until either EOF is reached +// on src or an error occurs. It verifies src exists and remove +// the dst if it exists. +func CopyFile(src, dst string) (int64, error) { + cleanSrc := filepath.Clean(src) + cleanDst := filepath.Clean(dst) + if cleanSrc == cleanDst { + return 0, nil + } + sf, err := os.Open(cleanSrc) + if err != nil { + return 0, err + } + defer sf.Close() + if err := os.Remove(cleanDst); err != nil && !os.IsNotExist(err) { + return 0, err + } + df, err := os.Create(cleanDst) + if err != nil { + return 0, err + } + defer df.Close() + return io.Copy(df, sf) +} + +// GetTotalUsedFds Returns the number of used File Descriptors by +// reading it via /proc filesystem. +func GetTotalUsedFds() int { + if fds, err := ioutil.ReadDir(fmt.Sprintf("/proc/%d/fd", os.Getpid())); err != nil { + logrus.Errorf("Error opening /proc/%d/fd: %s", os.Getpid(), err) + } else { + return len(fds) + } + return -1 +} + +// ReadSymlinkedDirectory returns the target directory of a symlink. +// The target of the symbolic link may not be a file. +func ReadSymlinkedDirectory(path string) (string, error) { + var realPath string + var err error + if realPath, err = filepath.Abs(path); err != nil { + return "", fmt.Errorf("unable to get absolute path for %s: %s", path, err) + } + if realPath, err = filepath.EvalSymlinks(realPath); err != nil { + return "", fmt.Errorf("failed to canonicalise path for %s: %s", path, err) + } + realPathInfo, err := os.Stat(realPath) + if err != nil { + return "", fmt.Errorf("failed to stat target '%s' of '%s': %s", realPath, path, err) + } + if !realPathInfo.Mode().IsDir() { + return "", fmt.Errorf("canonical path points to a file '%s'", realPath) + } + return realPath, nil +} + +// CreateIfNotExists creates a file or a directory only if it does not already exist. +func CreateIfNotExists(path string, isDir bool) error { + if _, err := os.Stat(path); err != nil { + if os.IsNotExist(err) { + if isDir { + return os.MkdirAll(path, 0755) + } + if err := os.MkdirAll(filepath.Dir(path), 0755); err != nil { + return err + } + f, err := os.OpenFile(path, os.O_CREATE, 0755) + if err != nil { + return err + } + f.Close() + } + } + return nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils/fileutils_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils/fileutils_test.go new file mode 100644 index 0000000000000..b544ffbf2ed85 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils/fileutils_test.go @@ -0,0 +1,402 @@ +package fileutils + +import ( + "io/ioutil" + "os" + "path" + "path/filepath" + "testing" +) + +// CopyFile with invalid src +func TestCopyFileWithInvalidSrc(t *testing.T) { + tempFolder, err := ioutil.TempDir("", "docker-fileutils-test") + defer os.RemoveAll(tempFolder) + if err != nil { + t.Fatal(err) + } + bytes, err := CopyFile("/invalid/file/path", path.Join(tempFolder, "dest")) + if err == nil { + t.Fatal("Should have fail to copy an invalid src file") + } + if bytes != 0 { + t.Fatal("Should have written 0 bytes") + } + +} + +// CopyFile with invalid dest +func TestCopyFileWithInvalidDest(t *testing.T) { + tempFolder, err := ioutil.TempDir("", "docker-fileutils-test") + defer os.RemoveAll(tempFolder) + if err != nil { + t.Fatal(err) + } + src := path.Join(tempFolder, "file") + err = ioutil.WriteFile(src, []byte("content"), 0740) + if err != nil { + t.Fatal(err) + } + bytes, err := CopyFile(src, path.Join(tempFolder, "/invalid/dest/path")) + if err == nil { + t.Fatal("Should have fail to copy an invalid src file") + } + if bytes != 0 { + t.Fatal("Should have written 0 bytes") + } + +} + +// CopyFile with same src and dest +func TestCopyFileWithSameSrcAndDest(t *testing.T) { + tempFolder, err := ioutil.TempDir("", "docker-fileutils-test") + defer os.RemoveAll(tempFolder) + if err != nil { + t.Fatal(err) + } + file := path.Join(tempFolder, "file") + err = ioutil.WriteFile(file, []byte("content"), 0740) + if err != nil { + t.Fatal(err) + } + bytes, err := CopyFile(file, file) + if err != nil { + t.Fatal(err) + } + if bytes != 0 { + t.Fatal("Should have written 0 bytes as it is the same file.") + } +} + +// CopyFile with same src and dest but path is different and not clean +func TestCopyFileWithSameSrcAndDestWithPathNameDifferent(t *testing.T) { + tempFolder, err := ioutil.TempDir("", "docker-fileutils-test") + defer os.RemoveAll(tempFolder) + if err != nil { + t.Fatal(err) + } + testFolder := path.Join(tempFolder, "test") + err = os.MkdirAll(testFolder, 0740) + if err != nil { + t.Fatal(err) + } + file := path.Join(testFolder, "file") + sameFile := testFolder + "/../test/file" + err = ioutil.WriteFile(file, []byte("content"), 0740) + if err != nil { + t.Fatal(err) + } + bytes, err := CopyFile(file, sameFile) + if err != nil { + t.Fatal(err) + } + if bytes != 0 { + t.Fatal("Should have written 0 bytes as it is the same file.") + } +} + +func TestCopyFile(t *testing.T) { + tempFolder, err := ioutil.TempDir("", "docker-fileutils-test") + defer os.RemoveAll(tempFolder) + if err != nil { + t.Fatal(err) + } + src := path.Join(tempFolder, "src") + dest := path.Join(tempFolder, "dest") + ioutil.WriteFile(src, []byte("content"), 0777) + ioutil.WriteFile(dest, []byte("destContent"), 0777) + bytes, err := CopyFile(src, dest) + if err != nil { + t.Fatal(err) + } + if bytes != 7 { + t.Fatalf("Should have written %d bytes but wrote %d", 7, bytes) + } + actual, err := ioutil.ReadFile(dest) + if err != nil { + t.Fatal(err) + } + if string(actual) != "content" { + t.Fatalf("Dest content was '%s', expected '%s'", string(actual), "content") + } +} + +// Reading a symlink to a directory must return the directory +func TestReadSymlinkedDirectoryExistingDirectory(t *testing.T) { + var err error + if err = os.Mkdir("/tmp/testReadSymlinkToExistingDirectory", 0777); err != nil { + t.Errorf("failed to create directory: %s", err) + } + + if err = os.Symlink("/tmp/testReadSymlinkToExistingDirectory", "/tmp/dirLinkTest"); err != nil { + t.Errorf("failed to create symlink: %s", err) + } + + var path string + if path, err = ReadSymlinkedDirectory("/tmp/dirLinkTest"); err != nil { + t.Fatalf("failed to read symlink to directory: %s", err) + } + + if path != "/tmp/testReadSymlinkToExistingDirectory" { + t.Fatalf("symlink returned unexpected directory: %s", path) + } + + if err = os.Remove("/tmp/testReadSymlinkToExistingDirectory"); err != nil { + t.Errorf("failed to remove temporary directory: %s", err) + } + + if err = os.Remove("/tmp/dirLinkTest"); err != nil { + t.Errorf("failed to remove symlink: %s", err) + } +} + +// Reading a non-existing symlink must fail +func TestReadSymlinkedDirectoryNonExistingSymlink(t *testing.T) { + var path string + var err error + if path, err = ReadSymlinkedDirectory("/tmp/test/foo/Non/ExistingPath"); err == nil { + t.Fatalf("error expected for non-existing symlink") + } + + if path != "" { + t.Fatalf("expected empty path, but '%s' was returned", path) + } +} + +// Reading a symlink to a file must fail +func TestReadSymlinkedDirectoryToFile(t *testing.T) { + var err error + var file *os.File + + if file, err = os.Create("/tmp/testReadSymlinkToFile"); err != nil { + t.Fatalf("failed to create file: %s", err) + } + + file.Close() + + if err = os.Symlink("/tmp/testReadSymlinkToFile", "/tmp/fileLinkTest"); err != nil { + t.Errorf("failed to create symlink: %s", err) + } + + var path string + if path, err = ReadSymlinkedDirectory("/tmp/fileLinkTest"); err == nil { + t.Fatalf("ReadSymlinkedDirectory on a symlink to a file should've failed") + } + + if path != "" { + t.Fatalf("path should've been empty: %s", path) + } + + if err = os.Remove("/tmp/testReadSymlinkToFile"); err != nil { + t.Errorf("failed to remove file: %s", err) + } + + if err = os.Remove("/tmp/fileLinkTest"); err != nil { + t.Errorf("failed to remove symlink: %s", err) + } +} + +func TestWildcardMatches(t *testing.T) { + match, _ := Matches("fileutils.go", []string{"*"}) + if match != true { + t.Errorf("failed to get a wildcard match, got %v", match) + } +} + +// A simple pattern match should return true. +func TestPatternMatches(t *testing.T) { + match, _ := Matches("fileutils.go", []string{"*.go"}) + if match != true { + t.Errorf("failed to get a match, got %v", match) + } +} + +// An exclusion followed by an inclusion should return true. +func TestExclusionPatternMatchesPatternBefore(t *testing.T) { + match, _ := Matches("fileutils.go", []string{"!fileutils.go", "*.go"}) + if match != true { + t.Errorf("failed to get true match on exclusion pattern, got %v", match) + } +} + +// A folder pattern followed by an exception should return false. +func TestPatternMatchesFolderExclusions(t *testing.T) { + match, _ := Matches("docs/README.md", []string{"docs", "!docs/README.md"}) + if match != false { + t.Errorf("failed to get a false match on exclusion pattern, got %v", match) + } +} + +// A folder pattern followed by an exception should return false. +func TestPatternMatchesFolderWithSlashExclusions(t *testing.T) { + match, _ := Matches("docs/README.md", []string{"docs/", "!docs/README.md"}) + if match != false { + t.Errorf("failed to get a false match on exclusion pattern, got %v", match) + } +} + +// A folder pattern followed by an exception should return false. +func TestPatternMatchesFolderWildcardExclusions(t *testing.T) { + match, _ := Matches("docs/README.md", []string{"docs/*", "!docs/README.md"}) + if match != false { + t.Errorf("failed to get a false match on exclusion pattern, got %v", match) + } +} + +// A pattern followed by an exclusion should return false. +func TestExclusionPatternMatchesPatternAfter(t *testing.T) { + match, _ := Matches("fileutils.go", []string{"*.go", "!fileutils.go"}) + if match != false { + t.Errorf("failed to get false match on exclusion pattern, got %v", match) + } +} + +// A filename evaluating to . should return false. +func TestExclusionPatternMatchesWholeDirectory(t *testing.T) { + match, _ := Matches(".", []string{"*.go"}) + if match != false { + t.Errorf("failed to get false match on ., got %v", match) + } +} + +// A single ! pattern should return an error. +func TestSingleExclamationError(t *testing.T) { + _, err := Matches("fileutils.go", []string{"!"}) + if err == nil { + t.Errorf("failed to get an error for a single exclamation point, got %v", err) + } +} + +// A string preceded with a ! should return true from Exclusion. +func TestExclusion(t *testing.T) { + exclusion := exclusion("!") + if !exclusion { + t.Errorf("failed to get true for a single !, got %v", exclusion) + } +} + +// Matches with no patterns +func TestMatchesWithNoPatterns(t *testing.T) { + matches, err := Matches("/any/path/there", []string{}) + if err != nil { + t.Fatal(err) + } + if matches { + t.Fatalf("Should not have match anything") + } +} + +// Matches with malformed patterns +func TestMatchesWithMalformedPatterns(t *testing.T) { + matches, err := Matches("/any/path/there", []string{"["}) + if err == nil { + t.Fatal("Should have failed because of a malformed syntax in the pattern") + } + if matches { + t.Fatalf("Should not have match anything") + } +} + +// An empty string should return true from Empty. +func TestEmpty(t *testing.T) { + empty := empty("") + if !empty { + t.Errorf("failed to get true for an empty string, got %v", empty) + } +} + +func TestCleanPatterns(t *testing.T) { + cleaned, _, _, _ := CleanPatterns([]string{"docs", "config"}) + if len(cleaned) != 2 { + t.Errorf("expected 2 element slice, got %v", len(cleaned)) + } +} + +func TestCleanPatternsStripEmptyPatterns(t *testing.T) { + cleaned, _, _, _ := CleanPatterns([]string{"docs", "config", ""}) + if len(cleaned) != 2 { + t.Errorf("expected 2 element slice, got %v", len(cleaned)) + } +} + +func TestCleanPatternsExceptionFlag(t *testing.T) { + _, _, exceptions, _ := CleanPatterns([]string{"docs", "!docs/README.md"}) + if !exceptions { + t.Errorf("expected exceptions to be true, got %v", exceptions) + } +} + +func TestCleanPatternsLeadingSpaceTrimmed(t *testing.T) { + _, _, exceptions, _ := CleanPatterns([]string{"docs", " !docs/README.md"}) + if !exceptions { + t.Errorf("expected exceptions to be true, got %v", exceptions) + } +} + +func TestCleanPatternsTrailingSpaceTrimmed(t *testing.T) { + _, _, exceptions, _ := CleanPatterns([]string{"docs", "!docs/README.md "}) + if !exceptions { + t.Errorf("expected exceptions to be true, got %v", exceptions) + } +} + +func TestCleanPatternsErrorSingleException(t *testing.T) { + _, _, _, err := CleanPatterns([]string{"!"}) + if err == nil { + t.Errorf("expected error on single exclamation point, got %v", err) + } +} + +func TestCleanPatternsFolderSplit(t *testing.T) { + _, dirs, _, _ := CleanPatterns([]string{"docs/config/CONFIG.md"}) + if dirs[0][0] != "docs" { + t.Errorf("expected first element in dirs slice to be docs, got %v", dirs[0][1]) + } + if dirs[0][1] != "config" { + t.Errorf("expected first element in dirs slice to be config, got %v", dirs[0][1]) + } +} + +func TestCreateIfNotExistsDir(t *testing.T) { + tempFolder, err := ioutil.TempDir("", "docker-fileutils-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(tempFolder) + + folderToCreate := filepath.Join(tempFolder, "tocreate") + + if err := CreateIfNotExists(folderToCreate, true); err != nil { + t.Fatal(err) + } + fileinfo, err := os.Stat(folderToCreate) + if err != nil { + t.Fatalf("Should have create a folder, got %v", err) + } + + if !fileinfo.IsDir() { + t.Fatalf("Should have been a dir, seems it's not") + } +} + +func TestCreateIfNotExistsFile(t *testing.T) { + tempFolder, err := ioutil.TempDir("", "docker-fileutils-test") + if err != nil { + t.Fatal(err) + } + defer os.RemoveAll(tempFolder) + + fileToCreate := filepath.Join(tempFolder, "file/to/create") + + if err := CreateIfNotExists(fileToCreate, false); err != nil { + t.Fatal(err) + } + fileinfo, err := os.Stat(fileToCreate) + if err != nil { + t.Fatalf("Should have create a file, got %v", err) + } + + if fileinfo.IsDir() { + t.Fatalf("Should have been a file, seems it's not") + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir/homedir.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir/homedir.go new file mode 100644 index 0000000000000..dcae17882450c --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir/homedir.go @@ -0,0 +1,39 @@ +package homedir + +import ( + "os" + "runtime" + + "github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user" +) + +// Key returns the env var name for the user's home dir based on +// the platform being run on +func Key() string { + if runtime.GOOS == "windows" { + return "USERPROFILE" + } + return "HOME" +} + +// Get returns the home directory of the current user with the help of +// environment variables depending on the target operating system. +// Returned path should be used with "path/filepath" to form new paths. +func Get() string { + home := os.Getenv(Key()) + if home == "" && runtime.GOOS != "windows" { + if u, err := user.CurrentUser(); err == nil { + return u.Home + } + } + return home +} + +// GetShortcutString returns the string that is shortcut to user's home directory +// in the native shell of the platform running on. +func GetShortcutString() string { + if runtime.GOOS == "windows" { + return "%USERPROFILE%" // be careful while using in format functions + } + return "~" +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir/homedir_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir/homedir_test.go new file mode 100644 index 0000000000000..7a95cb2bd7d28 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir/homedir_test.go @@ -0,0 +1,24 @@ +package homedir + +import ( + "path/filepath" + "testing" +) + +func TestGet(t *testing.T) { + home := Get() + if home == "" { + t.Fatal("returned home directory is empty") + } + + if !filepath.IsAbs(home) { + t.Fatalf("returned path is not absolute: %s", home) + } +} + +func TestGetShortcutString(t *testing.T) { + shortcut := GetShortcutString() + if shortcut == "" { + t.Fatal("returned shortcut string is empty") + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/fmt.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/fmt.go new file mode 100644 index 0000000000000..801132ff3d355 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/fmt.go @@ -0,0 +1,14 @@ +package ioutils + +import ( + "fmt" + "io" +) + +// FprintfIfNotEmpty prints the string value if it's not empty +func FprintfIfNotEmpty(w io.Writer, format, value string) (int, error) { + if value != "" { + return fmt.Fprintf(w, format, value) + } + return 0, nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/fmt_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/fmt_test.go new file mode 100644 index 0000000000000..8968863296dad --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/fmt_test.go @@ -0,0 +1,17 @@ +package ioutils + +import "testing" + +func TestFprintfIfNotEmpty(t *testing.T) { + wc := NewWriteCounter(&NopWriter{}) + n, _ := FprintfIfNotEmpty(wc, "foo%s", "") + + if wc.Count != 0 || n != 0 { + t.Errorf("Wrong count: %v vs. %v vs. 0", wc.Count, n) + } + + n, _ = FprintfIfNotEmpty(wc, "foo%s", "bar") + if wc.Count != 6 || n != 6 { + t.Errorf("Wrong count: %v vs. %v vs. 6", wc.Count, n) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/multireader.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/multireader.go new file mode 100644 index 0000000000000..f231aa9daf5c0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/multireader.go @@ -0,0 +1,226 @@ +package ioutils + +import ( + "bytes" + "fmt" + "io" + "os" +) + +type pos struct { + idx int + offset int64 +} + +type multiReadSeeker struct { + readers []io.ReadSeeker + pos *pos + posIdx map[io.ReadSeeker]int +} + +func (r *multiReadSeeker) Seek(offset int64, whence int) (int64, error) { + var tmpOffset int64 + switch whence { + case os.SEEK_SET: + for i, rdr := range r.readers { + // get size of the current reader + s, err := rdr.Seek(0, os.SEEK_END) + if err != nil { + return -1, err + } + + if offset > tmpOffset+s { + if i == len(r.readers)-1 { + rdrOffset := s + (offset - tmpOffset) + if _, err := rdr.Seek(rdrOffset, os.SEEK_SET); err != nil { + return -1, err + } + r.pos = &pos{i, rdrOffset} + return offset, nil + } + + tmpOffset += s + continue + } + + rdrOffset := offset - tmpOffset + idx := i + + rdr.Seek(rdrOffset, os.SEEK_SET) + // make sure all following readers are at 0 + for _, rdr := range r.readers[i+1:] { + rdr.Seek(0, os.SEEK_SET) + } + + if rdrOffset == s && i != len(r.readers)-1 { + idx += 1 + rdrOffset = 0 + } + r.pos = &pos{idx, rdrOffset} + return offset, nil + } + case os.SEEK_END: + for _, rdr := range r.readers { + s, err := rdr.Seek(0, os.SEEK_END) + if err != nil { + return -1, err + } + tmpOffset += s + } + r.Seek(tmpOffset+offset, os.SEEK_SET) + return tmpOffset + offset, nil + case os.SEEK_CUR: + if r.pos == nil { + return r.Seek(offset, os.SEEK_SET) + } + // Just return the current offset + if offset == 0 { + return r.getCurOffset() + } + + curOffset, err := r.getCurOffset() + if err != nil { + return -1, err + } + rdr, rdrOffset, err := r.getReaderForOffset(curOffset + offset) + if err != nil { + return -1, err + } + + r.pos = &pos{r.posIdx[rdr], rdrOffset} + return curOffset + offset, nil + default: + return -1, fmt.Errorf("Invalid whence: %d", whence) + } + + return -1, fmt.Errorf("Error seeking for whence: %d, offset: %d", whence, offset) +} + +func (r *multiReadSeeker) getReaderForOffset(offset int64) (io.ReadSeeker, int64, error) { + var rdr io.ReadSeeker + var rdrOffset int64 + + for i, rdr := range r.readers { + offsetTo, err := r.getOffsetToReader(rdr) + if err != nil { + return nil, -1, err + } + if offsetTo > offset { + rdr = r.readers[i-1] + rdrOffset = offsetTo - offset + break + } + + if rdr == r.readers[len(r.readers)-1] { + rdrOffset = offsetTo + offset + break + } + } + + return rdr, rdrOffset, nil +} + +func (r *multiReadSeeker) getCurOffset() (int64, error) { + var totalSize int64 + for _, rdr := range r.readers[:r.pos.idx+1] { + if r.posIdx[rdr] == r.pos.idx { + totalSize += r.pos.offset + break + } + + size, err := getReadSeekerSize(rdr) + if err != nil { + return -1, fmt.Errorf("error getting seeker size: %v", err) + } + totalSize += size + } + return totalSize, nil +} + +func (r *multiReadSeeker) getOffsetToReader(rdr io.ReadSeeker) (int64, error) { + var offset int64 + for _, r := range r.readers { + if r == rdr { + break + } + + size, err := getReadSeekerSize(rdr) + if err != nil { + return -1, err + } + offset += size + } + return offset, nil +} + +func (r *multiReadSeeker) Read(b []byte) (int, error) { + if r.pos == nil { + r.pos = &pos{0, 0} + } + + bCap := int64(cap(b)) + buf := bytes.NewBuffer(nil) + var rdr io.ReadSeeker + + for _, rdr = range r.readers[r.pos.idx:] { + readBytes, err := io.CopyN(buf, rdr, bCap) + if err != nil && err != io.EOF { + return -1, err + } + bCap -= readBytes + + if bCap == 0 { + break + } + } + + rdrPos, err := rdr.Seek(0, os.SEEK_CUR) + if err != nil { + return -1, err + } + r.pos = &pos{r.posIdx[rdr], rdrPos} + return buf.Read(b) +} + +func getReadSeekerSize(rdr io.ReadSeeker) (int64, error) { + // save the current position + pos, err := rdr.Seek(0, os.SEEK_CUR) + if err != nil { + return -1, err + } + + // get the size + size, err := rdr.Seek(0, os.SEEK_END) + if err != nil { + return -1, err + } + + // reset the position + if _, err := rdr.Seek(pos, os.SEEK_SET); err != nil { + return -1, err + } + return size, nil +} + +// MultiReadSeeker returns a ReadSeeker that's the logical concatenation of the provided +// input readseekers. After calling this method the initial position is set to the +// beginning of the first ReadSeeker. At the end of a ReadSeeker, Read always advances +// to the beginning of the next ReadSeeker and returns EOF at the end of the last ReadSeeker. +// Seek can be used over the sum of lengths of all readseekers. +// +// When a MultiReadSeeker is used, no Read and Seek operations should be made on +// its ReadSeeker components. Also, users should make no assumption on the state +// of individual readseekers while the MultiReadSeeker is used. +func MultiReadSeeker(readers ...io.ReadSeeker) io.ReadSeeker { + if len(readers) == 1 { + return readers[0] + } + idx := make(map[io.ReadSeeker]int) + for i, rdr := range readers { + idx[rdr] = i + } + return &multiReadSeeker{ + readers: readers, + posIdx: idx, + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/multireader_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/multireader_test.go new file mode 100644 index 0000000000000..de495b56da449 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/multireader_test.go @@ -0,0 +1,149 @@ +package ioutils + +import ( + "bytes" + "fmt" + "io" + "io/ioutil" + "os" + "strings" + "testing" +) + +func TestMultiReadSeekerReadAll(t *testing.T) { + str := "hello world" + s1 := strings.NewReader(str + " 1") + s2 := strings.NewReader(str + " 2") + s3 := strings.NewReader(str + " 3") + mr := MultiReadSeeker(s1, s2, s3) + + expectedSize := int64(s1.Len() + s2.Len() + s3.Len()) + + b, err := ioutil.ReadAll(mr) + if err != nil { + t.Fatal(err) + } + + expected := "hello world 1hello world 2hello world 3" + if string(b) != expected { + t.Fatalf("ReadAll failed, got: %q, expected %q", string(b), expected) + } + + size, err := mr.Seek(0, os.SEEK_END) + if err != nil { + t.Fatal(err) + } + if size != expectedSize { + t.Fatalf("reader size does not match, got %d, expected %d", size, expectedSize) + } + + // Reset the position and read again + pos, err := mr.Seek(0, os.SEEK_SET) + if err != nil { + t.Fatal(err) + } + if pos != 0 { + t.Fatalf("expected position to be set to 0, got %d", pos) + } + + b, err = ioutil.ReadAll(mr) + if err != nil { + t.Fatal(err) + } + + if string(b) != expected { + t.Fatalf("ReadAll failed, got: %q, expected %q", string(b), expected) + } +} + +func TestMultiReadSeekerReadEach(t *testing.T) { + str := "hello world" + s1 := strings.NewReader(str + " 1") + s2 := strings.NewReader(str + " 2") + s3 := strings.NewReader(str + " 3") + mr := MultiReadSeeker(s1, s2, s3) + + var totalBytes int64 + for i, s := range []*strings.Reader{s1, s2, s3} { + sLen := int64(s.Len()) + buf := make([]byte, s.Len()) + expected := []byte(fmt.Sprintf("%s %d", str, i+1)) + + if _, err := mr.Read(buf); err != nil && err != io.EOF { + t.Fatal(err) + } + + if !bytes.Equal(buf, expected) { + t.Fatalf("expected %q to be %q", string(buf), string(expected)) + } + + pos, err := mr.Seek(0, os.SEEK_CUR) + if err != nil { + t.Fatalf("iteration: %d, error: %v", i+1, err) + } + + // check that the total bytes read is the current position of the seeker + totalBytes += sLen + if pos != totalBytes { + t.Fatalf("expected current position to be: %d, got: %d, iteration: %d", totalBytes, pos, i+1) + } + + // This tests not only that SEEK_SET and SEEK_CUR give the same values, but that the next iteration is in the expected position as well + newPos, err := mr.Seek(pos, os.SEEK_SET) + if err != nil { + t.Fatal(err) + } + if newPos != pos { + t.Fatalf("expected to get same position when calling SEEK_SET with value from SEEK_CUR, cur: %d, set: %d", pos, newPos) + } + } +} + +func TestMultiReadSeekerReadSpanningChunks(t *testing.T) { + str := "hello world" + s1 := strings.NewReader(str + " 1") + s2 := strings.NewReader(str + " 2") + s3 := strings.NewReader(str + " 3") + mr := MultiReadSeeker(s1, s2, s3) + + buf := make([]byte, s1.Len()+3) + _, err := mr.Read(buf) + if err != nil { + t.Fatal(err) + } + + // expected is the contents of s1 + 3 bytes from s2, ie, the `hel` at the end of this string + expected := "hello world 1hel" + if string(buf) != expected { + t.Fatalf("expected %s to be %s", string(buf), expected) + } +} + +func TestMultiReadSeekerNegativeSeek(t *testing.T) { + str := "hello world" + s1 := strings.NewReader(str + " 1") + s2 := strings.NewReader(str + " 2") + s3 := strings.NewReader(str + " 3") + mr := MultiReadSeeker(s1, s2, s3) + + s1Len := s1.Len() + s2Len := s2.Len() + s3Len := s3.Len() + + s, err := mr.Seek(int64(-1*s3.Len()), os.SEEK_END) + if err != nil { + t.Fatal(err) + } + if s != int64(s1Len+s2Len) { + t.Fatalf("expected %d to be %d", s, s1.Len()+s2.Len()) + } + + buf := make([]byte, s3Len) + if _, err := mr.Read(buf); err != nil && err != io.EOF { + t.Fatal(err) + } + expected := fmt.Sprintf("%s %d", str, 3) + if string(buf) != fmt.Sprintf("%s %d", str, 3) { + t.Fatalf("expected %q to be %q", string(buf), expected) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/readers.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/readers.go new file mode 100644 index 0000000000000..ff09baad178f1 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/readers.go @@ -0,0 +1,254 @@ +package ioutils + +import ( + "bytes" + "crypto/rand" + "crypto/sha256" + "encoding/hex" + "io" + "math/big" + "sync" + "time" +) + +type readCloserWrapper struct { + io.Reader + closer func() error +} + +func (r *readCloserWrapper) Close() error { + return r.closer() +} + +func NewReadCloserWrapper(r io.Reader, closer func() error) io.ReadCloser { + return &readCloserWrapper{ + Reader: r, + closer: closer, + } +} + +type readerErrWrapper struct { + reader io.Reader + closer func() +} + +func (r *readerErrWrapper) Read(p []byte) (int, error) { + n, err := r.reader.Read(p) + if err != nil { + r.closer() + } + return n, err +} + +func NewReaderErrWrapper(r io.Reader, closer func()) io.Reader { + return &readerErrWrapper{ + reader: r, + closer: closer, + } +} + +// bufReader allows the underlying reader to continue to produce +// output by pre-emptively reading from the wrapped reader. +// This is achieved by buffering this data in bufReader's +// expanding buffer. +type bufReader struct { + sync.Mutex + buf *bytes.Buffer + reader io.Reader + err error + wait sync.Cond + drainBuf []byte + reuseBuf []byte + maxReuse int64 + resetTimeout time.Duration + bufLenResetThreshold int64 + maxReadDataReset int64 +} + +func NewBufReader(r io.Reader) *bufReader { + var timeout int + if randVal, err := rand.Int(rand.Reader, big.NewInt(120)); err == nil { + timeout = int(randVal.Int64()) + 180 + } else { + timeout = 300 + } + reader := &bufReader{ + buf: &bytes.Buffer{}, + drainBuf: make([]byte, 1024), + reuseBuf: make([]byte, 4096), + maxReuse: 1000, + resetTimeout: time.Second * time.Duration(timeout), + bufLenResetThreshold: 100 * 1024, + maxReadDataReset: 10 * 1024 * 1024, + reader: r, + } + reader.wait.L = &reader.Mutex + go reader.drain() + return reader +} + +func NewBufReaderWithDrainbufAndBuffer(r io.Reader, drainBuffer []byte, buffer *bytes.Buffer) *bufReader { + reader := &bufReader{ + buf: buffer, + drainBuf: drainBuffer, + reader: r, + } + reader.wait.L = &reader.Mutex + go reader.drain() + return reader +} + +func (r *bufReader) drain() { + var ( + duration time.Duration + lastReset time.Time + now time.Time + reset bool + bufLen int64 + dataSinceReset int64 + maxBufLen int64 + reuseBufLen int64 + reuseCount int64 + ) + reuseBufLen = int64(len(r.reuseBuf)) + lastReset = time.Now() + for { + n, err := r.reader.Read(r.drainBuf) + dataSinceReset += int64(n) + r.Lock() + bufLen = int64(r.buf.Len()) + if bufLen > maxBufLen { + maxBufLen = bufLen + } + + // Avoid unbounded growth of the buffer over time. + // This has been discovered to be the only non-intrusive + // solution to the unbounded growth of the buffer. + // Alternative solutions such as compression, multiple + // buffers, channels and other similar pieces of code + // were reducing throughput, overall Docker performance + // or simply crashed Docker. + // This solution releases the buffer when specific + // conditions are met to avoid the continuous resizing + // of the buffer for long lived containers. + // + // Move data to the front of the buffer if it's + // smaller than what reuseBuf can store + if bufLen > 0 && reuseBufLen >= bufLen { + n, _ := r.buf.Read(r.reuseBuf) + r.buf.Write(r.reuseBuf[0:n]) + // Take action if the buffer has been reused too many + // times and if there's data in the buffer. + // The timeout is also used as means to avoid doing + // these operations more often or less often than + // required. + // The various conditions try to detect heavy activity + // in the buffer which might be indicators of heavy + // growth of the buffer. + } else if reuseCount >= r.maxReuse && bufLen > 0 { + now = time.Now() + duration = now.Sub(lastReset) + timeoutReached := duration >= r.resetTimeout + + // The timeout has been reached and the + // buffered data couldn't be moved to the front + // of the buffer, so the buffer gets reset. + if timeoutReached && bufLen > reuseBufLen { + reset = true + } + // The amount of buffered data is too high now, + // reset the buffer. + if timeoutReached && maxBufLen >= r.bufLenResetThreshold { + reset = true + } + // Reset the buffer if a certain amount of + // data has gone through the buffer since the + // last reset. + if timeoutReached && dataSinceReset >= r.maxReadDataReset { + reset = true + } + // The buffered data is moved to a fresh buffer, + // swap the old buffer with the new one and + // reset all counters. + if reset { + newbuf := &bytes.Buffer{} + newbuf.ReadFrom(r.buf) + r.buf = newbuf + lastReset = now + reset = false + dataSinceReset = 0 + maxBufLen = 0 + reuseCount = 0 + } + } + if err != nil { + r.err = err + } else { + r.buf.Write(r.drainBuf[0:n]) + } + reuseCount++ + r.wait.Signal() + r.Unlock() + callSchedulerIfNecessary() + if err != nil { + break + } + } +} + +func (r *bufReader) Read(p []byte) (n int, err error) { + r.Lock() + defer r.Unlock() + for { + n, err = r.buf.Read(p) + if n > 0 { + return n, err + } + if r.err != nil { + return 0, r.err + } + r.wait.Wait() + } +} + +func (r *bufReader) Close() error { + closer, ok := r.reader.(io.ReadCloser) + if !ok { + return nil + } + return closer.Close() +} + +func HashData(src io.Reader) (string, error) { + h := sha256.New() + if _, err := io.Copy(h, src); err != nil { + return "", err + } + return "sha256:" + hex.EncodeToString(h.Sum(nil)), nil +} + +type OnEOFReader struct { + Rc io.ReadCloser + Fn func() +} + +func (r *OnEOFReader) Read(p []byte) (n int, err error) { + n, err = r.Rc.Read(p) + if err == io.EOF { + r.runFunc() + } + return +} + +func (r *OnEOFReader) Close() error { + err := r.Rc.Close() + r.runFunc() + return err +} + +func (r *OnEOFReader) runFunc() { + if fn := r.Fn; fn != nil { + fn() + r.Fn = nil + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/readers_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/readers_test.go new file mode 100644 index 0000000000000..0a39b6ec6a199 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/readers_test.go @@ -0,0 +1,216 @@ +package ioutils + +import ( + "bytes" + "fmt" + "io" + "io/ioutil" + "strings" + "testing" +) + +// Implement io.Reader +type errorReader struct{} + +func (r *errorReader) Read(p []byte) (int, error) { + return 0, fmt.Errorf("Error reader always fail.") +} + +func TestReadCloserWrapperClose(t *testing.T) { + reader := strings.NewReader("A string reader") + wrapper := NewReadCloserWrapper(reader, func() error { + return fmt.Errorf("This will be called when closing") + }) + err := wrapper.Close() + if err == nil || !strings.Contains(err.Error(), "This will be called when closing") { + t.Fatalf("readCloserWrapper should have call the anonymous func and thus, fail.") + } +} + +func TestReaderErrWrapperReadOnError(t *testing.T) { + called := false + reader := &errorReader{} + wrapper := NewReaderErrWrapper(reader, func() { + called = true + }) + _, err := wrapper.Read([]byte{}) + if err == nil || !strings.Contains(err.Error(), "Error reader always fail.") { + t.Fatalf("readErrWrapper should returned an error") + } + if !called { + t.Fatalf("readErrWrapper should have call the anonymous function on failure") + } +} + +func TestReaderErrWrapperRead(t *testing.T) { + reader := strings.NewReader("a string reader.") + wrapper := NewReaderErrWrapper(reader, func() { + t.Fatalf("readErrWrapper should not have called the anonymous function") + }) + // Read 20 byte (should be ok with the string above) + num, err := wrapper.Read(make([]byte, 20)) + if err != nil { + t.Fatal(err) + } + if num != 16 { + t.Fatalf("readerErrWrapper should have read 16 byte, but read %d", num) + } +} + +func TestNewBufReaderWithDrainbufAndBuffer(t *testing.T) { + reader, writer := io.Pipe() + + drainBuffer := make([]byte, 1024) + buffer := bytes.Buffer{} + bufreader := NewBufReaderWithDrainbufAndBuffer(reader, drainBuffer, &buffer) + + // Write everything down to a Pipe + // Usually, a pipe should block but because of the buffered reader, + // the writes will go through + done := make(chan bool) + go func() { + writer.Write([]byte("hello world")) + writer.Close() + done <- true + }() + + // Drain the reader *after* everything has been written, just to verify + // it is indeed buffering + <-done + + output, err := ioutil.ReadAll(bufreader) + if err != nil { + t.Fatal(err) + } + if !bytes.Equal(output, []byte("hello world")) { + t.Error(string(output)) + } +} + +func TestBufReader(t *testing.T) { + reader, writer := io.Pipe() + bufreader := NewBufReader(reader) + + // Write everything down to a Pipe + // Usually, a pipe should block but because of the buffered reader, + // the writes will go through + done := make(chan bool) + go func() { + writer.Write([]byte("hello world")) + writer.Close() + done <- true + }() + + // Drain the reader *after* everything has been written, just to verify + // it is indeed buffering + <-done + output, err := ioutil.ReadAll(bufreader) + if err != nil { + t.Fatal(err) + } + if !bytes.Equal(output, []byte("hello world")) { + t.Error(string(output)) + } +} + +func TestBufReaderCloseWithNonReaderCloser(t *testing.T) { + reader := strings.NewReader("buffer") + bufreader := NewBufReader(reader) + + if err := bufreader.Close(); err != nil { + t.Fatal(err) + } + +} + +// implements io.ReadCloser +type simpleReaderCloser struct{} + +func (r *simpleReaderCloser) Read(p []byte) (n int, err error) { + return 0, nil +} + +func (r *simpleReaderCloser) Close() error { + return nil +} + +func TestBufReaderCloseWithReaderCloser(t *testing.T) { + reader := &simpleReaderCloser{} + bufreader := NewBufReader(reader) + + err := bufreader.Close() + if err != nil { + t.Fatal(err) + } + +} + +func TestHashData(t *testing.T) { + reader := strings.NewReader("hash-me") + actual, err := HashData(reader) + if err != nil { + t.Fatal(err) + } + expected := "sha256:4d11186aed035cc624d553e10db358492c84a7cd6b9670d92123c144930450aa" + if actual != expected { + t.Fatalf("Expecting %s, got %s", expected, actual) + } +} + +type repeatedReader struct { + readCount int + maxReads int + data []byte +} + +func newRepeatedReader(max int, data []byte) *repeatedReader { + return &repeatedReader{0, max, data} +} + +func (r *repeatedReader) Read(p []byte) (int, error) { + if r.readCount >= r.maxReads { + return 0, io.EOF + } + r.readCount++ + n := copy(p, r.data) + return n, nil +} + +func testWithData(data []byte, reads int) { + reader := newRepeatedReader(reads, data) + bufReader := NewBufReader(reader) + io.Copy(ioutil.Discard, bufReader) +} + +func Benchmark1M10BytesReads(b *testing.B) { + reads := 1000000 + readSize := int64(10) + data := make([]byte, readSize) + b.SetBytes(readSize * int64(reads)) + b.ResetTimer() + for i := 0; i < b.N; i++ { + testWithData(data, reads) + } +} + +func Benchmark1M1024BytesReads(b *testing.B) { + reads := 1000000 + readSize := int64(1024) + data := make([]byte, readSize) + b.SetBytes(readSize * int64(reads)) + b.ResetTimer() + for i := 0; i < b.N; i++ { + testWithData(data, reads) + } +} + +func Benchmark10k32KBytesReads(b *testing.B) { + reads := 10000 + readSize := int64(32 * 1024) + data := make([]byte, readSize) + b.SetBytes(readSize * int64(reads)) + b.ResetTimer() + for i := 0; i < b.N; i++ { + testWithData(data, reads) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/scheduler.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/scheduler.go new file mode 100644 index 0000000000000..3c88f29e3553b --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/scheduler.go @@ -0,0 +1,6 @@ +// +build !gccgo + +package ioutils + +func callSchedulerIfNecessary() { +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/scheduler_gccgo.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/scheduler_gccgo.go new file mode 100644 index 0000000000000..c11d02b9476b1 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/scheduler_gccgo.go @@ -0,0 +1,13 @@ +// +build gccgo + +package ioutils + +import ( + "runtime" +) + +func callSchedulerIfNecessary() { + //allow or force Go scheduler to switch context, without explicitly + //forcing this will make it hang when using gccgo implementation + runtime.Gosched() +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/writeflusher.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/writeflusher.go new file mode 100644 index 0000000000000..25095474df363 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/writeflusher.go @@ -0,0 +1,47 @@ +package ioutils + +import ( + "io" + "net/http" + "sync" +) + +type WriteFlusher struct { + sync.Mutex + w io.Writer + flusher http.Flusher + flushed bool +} + +func (wf *WriteFlusher) Write(b []byte) (n int, err error) { + wf.Lock() + defer wf.Unlock() + n, err = wf.w.Write(b) + wf.flushed = true + wf.flusher.Flush() + return n, err +} + +// Flush the stream immediately. +func (wf *WriteFlusher) Flush() { + wf.Lock() + defer wf.Unlock() + wf.flushed = true + wf.flusher.Flush() +} + +func (wf *WriteFlusher) Flushed() bool { + wf.Lock() + defer wf.Unlock() + return wf.flushed +} + +func NewWriteFlusher(w io.Writer) *WriteFlusher { + var flusher http.Flusher + if f, ok := w.(http.Flusher); ok { + flusher = f + } else { + flusher = &NopFlusher{} + } + return &WriteFlusher{w: w, flusher: flusher} +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/writers.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/writers.go new file mode 100644 index 0000000000000..43fdc44ea9686 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/writers.go @@ -0,0 +1,60 @@ +package ioutils + +import "io" + +type NopWriter struct{} + +func (*NopWriter) Write(buf []byte) (int, error) { + return len(buf), nil +} + +type nopWriteCloser struct { + io.Writer +} + +func (w *nopWriteCloser) Close() error { return nil } + +func NopWriteCloser(w io.Writer) io.WriteCloser { + return &nopWriteCloser{w} +} + +type NopFlusher struct{} + +func (f *NopFlusher) Flush() {} + +type writeCloserWrapper struct { + io.Writer + closer func() error +} + +func (r *writeCloserWrapper) Close() error { + return r.closer() +} + +func NewWriteCloserWrapper(r io.Writer, closer func() error) io.WriteCloser { + return &writeCloserWrapper{ + Writer: r, + closer: closer, + } +} + +// Wrap a concrete io.Writer and hold a count of the number +// of bytes written to the writer during a "session". +// This can be convenient when write return is masked +// (e.g., json.Encoder.Encode()) +type WriteCounter struct { + Count int64 + Writer io.Writer +} + +func NewWriteCounter(w io.Writer) *WriteCounter { + return &WriteCounter{ + Writer: w, + } +} + +func (wc *WriteCounter) Write(p []byte) (count int, err error) { + count, err = wc.Writer.Write(p) + wc.Count += int64(count) + return +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/writers_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/writers_test.go new file mode 100644 index 0000000000000..564b1cd4f5f79 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils/writers_test.go @@ -0,0 +1,65 @@ +package ioutils + +import ( + "bytes" + "strings" + "testing" +) + +func TestWriteCloserWrapperClose(t *testing.T) { + called := false + writer := bytes.NewBuffer([]byte{}) + wrapper := NewWriteCloserWrapper(writer, func() error { + called = true + return nil + }) + if err := wrapper.Close(); err != nil { + t.Fatal(err) + } + if !called { + t.Fatalf("writeCloserWrapper should have call the anonymous function.") + } +} + +func TestNopWriteCloser(t *testing.T) { + writer := bytes.NewBuffer([]byte{}) + wrapper := NopWriteCloser(writer) + if err := wrapper.Close(); err != nil { + t.Fatal("NopWriteCloser always return nil on Close.") + } + +} + +func TestNopWriter(t *testing.T) { + nw := &NopWriter{} + l, err := nw.Write([]byte{'c'}) + if err != nil { + t.Fatal(err) + } + if l != 1 { + t.Fatalf("Expected 1 got %d", l) + } +} + +func TestWriteCounter(t *testing.T) { + dummy1 := "This is a dummy string." + dummy2 := "This is another dummy string." + totalLength := int64(len(dummy1) + len(dummy2)) + + reader1 := strings.NewReader(dummy1) + reader2 := strings.NewReader(dummy2) + + var buffer bytes.Buffer + wc := NewWriteCounter(&buffer) + + reader1.WriteTo(wc) + reader2.WriteTo(wc) + + if wc.Count != totalLength { + t.Errorf("Wrong count: %d vs. %d", wc.Count, totalLength) + } + + if buffer.String() != dummy1+dummy2 { + t.Error("Wrong message written") + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/mflag/LICENSE b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/mflag/LICENSE new file mode 100644 index 0000000000000..ac74d8f0496ca --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/mflag/LICENSE @@ -0,0 +1,27 @@ +Copyright (c) 2014-2015 The Docker & Go Authors. All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are +met: + + * Redistributions of source code must retain the above copyright +notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above +copyright notice, this list of conditions and the following disclaimer +in the documentation and/or other materials provided with the +distribution. + * Neither the name of Google Inc. nor the names of its +contributors may be used to endorse or promote products derived from +this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/mflag/README.md b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/mflag/README.md new file mode 100644 index 0000000000000..da00efa336e9b --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/mflag/README.md @@ -0,0 +1,40 @@ +Package mflag (aka multiple-flag) implements command-line flag parsing. +It's an **hacky** fork of the [official golang package](http://golang.org/pkg/flag/) + +It adds: + +* both short and long flag version +`./example -s red` `./example --string blue` + +* multiple names for the same option +``` +$>./example -h +Usage of example: + -s, --string="": a simple string +``` + +___ +It is very flexible on purpose, so you can do things like: +``` +$>./example -h +Usage of example: + -s, -string, --string="": a simple string +``` + +Or: +``` +$>./example -h +Usage of example: + -oldflag, --newflag="": a simple string +``` + +You can also hide some flags from the usage, so if we want only `--newflag`: +``` +$>./example -h +Usage of example: + --newflag="": a simple string +$>./example -oldflag str +str +``` + +See [example.go](example/example.go) for more details. diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/mflag/flag.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/mflag/flag.go new file mode 100644 index 0000000000000..ebfa350104275 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/mflag/flag.go @@ -0,0 +1,1201 @@ +// Copyright 2014-2015 The Docker & Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// Package mflag implements command-line flag parsing. +// +// Usage: +// +// Define flags using flag.String(), Bool(), Int(), etc. +// +// This declares an integer flag, -f or --flagname, stored in the pointer ip, with type *int. +// import "flag /github.com/docker/docker/pkg/mflag" +// var ip = flag.Int([]string{"f", "-flagname"}, 1234, "help message for flagname") +// If you like, you can bind the flag to a variable using the Var() functions. +// var flagvar int +// func init() { +// // -flaghidden will work, but will be hidden from the usage +// flag.IntVar(&flagvar, []string{"f", "#flaghidden", "-flagname"}, 1234, "help message for flagname") +// } +// Or you can create custom flags that satisfy the Value interface (with +// pointer receivers) and couple them to flag parsing by +// flag.Var(&flagVal, []string{"name"}, "help message for flagname") +// For such flags, the default value is just the initial value of the variable. +// +// You can also add "deprecated" flags, they are still usable, but are not shown +// in the usage and will display a warning when you try to use them. `#` before +// an option means this option is deprecated, if there is an following option +// without `#` ahead, then that's the replacement, if not, it will just be removed: +// var ip = flag.Int([]string{"#f", "#flagname", "-flagname"}, 1234, "help message for flagname") +// this will display: `Warning: '-f' is deprecated, it will be replaced by '--flagname' soon. See usage.` or +// this will display: `Warning: '-flagname' is deprecated, it will be replaced by '--flagname' soon. See usage.` +// var ip = flag.Int([]string{"f", "#flagname"}, 1234, "help message for flagname") +// will display: `Warning: '-flagname' is deprecated, it will be removed soon. See usage.` +// so you can only use `-f`. +// +// You can also group one letter flags, bif you declare +// var v = flag.Bool([]string{"v", "-verbose"}, false, "help message for verbose") +// var s = flag.Bool([]string{"s", "-slow"}, false, "help message for slow") +// you will be able to use the -vs or -sv +// +// After all flags are defined, call +// flag.Parse() +// to parse the command line into the defined flags. +// +// Flags may then be used directly. If you're using the flags themselves, +// they are all pointers; if you bind to variables, they're values. +// fmt.Println("ip has value ", *ip) +// fmt.Println("flagvar has value ", flagvar) +// +// After parsing, the arguments after the flag are available as the +// slice flag.Args() or individually as flag.Arg(i). +// The arguments are indexed from 0 through flag.NArg()-1. +// +// Command line flag syntax: +// -flag +// -flag=x +// -flag="x" +// -flag='x' +// -flag x // non-boolean flags only +// One or two minus signs may be used; they are equivalent. +// The last form is not permitted for boolean flags because the +// meaning of the command +// cmd -x * +// will change if there is a file called 0, false, etc. You must +// use the -flag=false form to turn off a boolean flag. +// +// Flag parsing stops just before the first non-flag argument +// ("-" is a non-flag argument) or after the terminator "--". +// +// Integer flags accept 1234, 0664, 0x1234 and may be negative. +// Boolean flags may be 1, 0, t, f, true, false, TRUE, FALSE, True, False. +// Duration flags accept any input valid for time.ParseDuration. +// +// The default set of command-line flags is controlled by +// top-level functions. The FlagSet type allows one to define +// independent sets of flags, such as to implement subcommands +// in a command-line interface. The methods of FlagSet are +// analogous to the top-level functions for the command-line +// flag set. + +package mflag + +import ( + "errors" + "fmt" + "io" + "os" + "runtime" + "sort" + "strconv" + "strings" + "text/tabwriter" + "time" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/homedir" +) + +// ErrHelp is the error returned if the flag -help is invoked but no such flag is defined. +var ErrHelp = errors.New("flag: help requested") + +// ErrRetry is the error returned if you need to try letter by letter +var ErrRetry = errors.New("flag: retry") + +// -- bool Value +type boolValue bool + +func newBoolValue(val bool, p *bool) *boolValue { + *p = val + return (*boolValue)(p) +} + +func (b *boolValue) Set(s string) error { + v, err := strconv.ParseBool(s) + *b = boolValue(v) + return err +} + +func (b *boolValue) Get() interface{} { return bool(*b) } + +func (b *boolValue) String() string { return fmt.Sprintf("%v", *b) } + +func (b *boolValue) IsBoolFlag() bool { return true } + +// optional interface to indicate boolean flags that can be +// supplied without "=value" text +type boolFlag interface { + Value + IsBoolFlag() bool +} + +// -- int Value +type intValue int + +func newIntValue(val int, p *int) *intValue { + *p = val + return (*intValue)(p) +} + +func (i *intValue) Set(s string) error { + v, err := strconv.ParseInt(s, 0, 64) + *i = intValue(v) + return err +} + +func (i *intValue) Get() interface{} { return int(*i) } + +func (i *intValue) String() string { return fmt.Sprintf("%v", *i) } + +// -- int64 Value +type int64Value int64 + +func newInt64Value(val int64, p *int64) *int64Value { + *p = val + return (*int64Value)(p) +} + +func (i *int64Value) Set(s string) error { + v, err := strconv.ParseInt(s, 0, 64) + *i = int64Value(v) + return err +} + +func (i *int64Value) Get() interface{} { return int64(*i) } + +func (i *int64Value) String() string { return fmt.Sprintf("%v", *i) } + +// -- uint Value +type uintValue uint + +func newUintValue(val uint, p *uint) *uintValue { + *p = val + return (*uintValue)(p) +} + +func (i *uintValue) Set(s string) error { + v, err := strconv.ParseUint(s, 0, 64) + *i = uintValue(v) + return err +} + +func (i *uintValue) Get() interface{} { return uint(*i) } + +func (i *uintValue) String() string { return fmt.Sprintf("%v", *i) } + +// -- uint64 Value +type uint64Value uint64 + +func newUint64Value(val uint64, p *uint64) *uint64Value { + *p = val + return (*uint64Value)(p) +} + +func (i *uint64Value) Set(s string) error { + v, err := strconv.ParseUint(s, 0, 64) + *i = uint64Value(v) + return err +} + +func (i *uint64Value) Get() interface{} { return uint64(*i) } + +func (i *uint64Value) String() string { return fmt.Sprintf("%v", *i) } + +// -- string Value +type stringValue string + +func newStringValue(val string, p *string) *stringValue { + *p = val + return (*stringValue)(p) +} + +func (s *stringValue) Set(val string) error { + *s = stringValue(val) + return nil +} + +func (s *stringValue) Get() interface{} { return string(*s) } + +func (s *stringValue) String() string { return fmt.Sprintf("%s", *s) } + +// -- float64 Value +type float64Value float64 + +func newFloat64Value(val float64, p *float64) *float64Value { + *p = val + return (*float64Value)(p) +} + +func (f *float64Value) Set(s string) error { + v, err := strconv.ParseFloat(s, 64) + *f = float64Value(v) + return err +} + +func (f *float64Value) Get() interface{} { return float64(*f) } + +func (f *float64Value) String() string { return fmt.Sprintf("%v", *f) } + +// -- time.Duration Value +type durationValue time.Duration + +func newDurationValue(val time.Duration, p *time.Duration) *durationValue { + *p = val + return (*durationValue)(p) +} + +func (d *durationValue) Set(s string) error { + v, err := time.ParseDuration(s) + *d = durationValue(v) + return err +} + +func (d *durationValue) Get() interface{} { return time.Duration(*d) } + +func (d *durationValue) String() string { return (*time.Duration)(d).String() } + +// Value is the interface to the dynamic value stored in a flag. +// (The default value is represented as a string.) +// +// If a Value has an IsBoolFlag() bool method returning true, +// the command-line parser makes -name equivalent to -name=true +// rather than using the next command-line argument. +type Value interface { + String() string + Set(string) error +} + +// Getter is an interface that allows the contents of a Value to be retrieved. +// It wraps the Value interface, rather than being part of it, because it +// appeared after Go 1 and its compatibility rules. All Value types provided +// by this package satisfy the Getter interface. +type Getter interface { + Value + Get() interface{} +} + +// ErrorHandling defines how to handle flag parsing errors. +type ErrorHandling int + +// ErrorHandling strategies available when a flag parsing error occurs +const ( + ContinueOnError ErrorHandling = iota + ExitOnError + PanicOnError +) + +// A FlagSet represents a set of defined flags. The zero value of a FlagSet +// has no name and has ContinueOnError error handling. +type FlagSet struct { + // Usage is the function called when an error occurs while parsing flags. + // The field is a function (not a method) that may be changed to point to + // a custom error handler. + Usage func() + ShortUsage func() + + name string + parsed bool + actual map[string]*Flag + formal map[string]*Flag + args []string // arguments after flags + errorHandling ErrorHandling + output io.Writer // nil means stderr; use Out() accessor + nArgRequirements []nArgRequirement +} + +// A Flag represents the state of a flag. +type Flag struct { + Names []string // name as it appears on command line + Usage string // help message + Value Value // value as set + DefValue string // default value (as text); for usage message +} + +type flagSlice []string + +func (p flagSlice) Len() int { return len(p) } +func (p flagSlice) Less(i, j int) bool { + pi, pj := strings.TrimPrefix(p[i], "-"), strings.TrimPrefix(p[j], "-") + lpi, lpj := strings.ToLower(pi), strings.ToLower(pj) + if lpi != lpj { + return lpi < lpj + } + return pi < pj +} +func (p flagSlice) Swap(i, j int) { p[i], p[j] = p[j], p[i] } + +// sortFlags returns the flags as a slice in lexicographical sorted order. +func sortFlags(flags map[string]*Flag) []*Flag { + var list flagSlice + + // The sorted list is based on the first name, when flag map might use the other names. + nameMap := make(map[string]string) + + for n, f := range flags { + fName := strings.TrimPrefix(f.Names[0], "#") + nameMap[fName] = n + if len(f.Names) == 1 { + list = append(list, fName) + continue + } + + found := false + for _, name := range list { + if name == fName { + found = true + break + } + } + if !found { + list = append(list, fName) + } + } + sort.Sort(list) + result := make([]*Flag, len(list)) + for i, name := range list { + result[i] = flags[nameMap[name]] + } + return result +} + +// Name returns the name of the FlagSet. +func (fs *FlagSet) Name() string { + return fs.name +} + +// Out returns the destination for usage and error messages. +func (fs *FlagSet) Out() io.Writer { + if fs.output == nil { + return os.Stderr + } + return fs.output +} + +// SetOutput sets the destination for usage and error messages. +// If output is nil, os.Stderr is used. +func (fs *FlagSet) SetOutput(output io.Writer) { + fs.output = output +} + +// VisitAll visits the flags in lexicographical order, calling fn for each. +// It visits all flags, even those not set. +func (fs *FlagSet) VisitAll(fn func(*Flag)) { + for _, flag := range sortFlags(fs.formal) { + fn(flag) + } +} + +// VisitAll visits the command-line flags in lexicographical order, calling +// fn for each. It visits all flags, even those not set. +func VisitAll(fn func(*Flag)) { + CommandLine.VisitAll(fn) +} + +// Visit visits the flags in lexicographical order, calling fn for each. +// It visits only those flags that have been set. +func (fs *FlagSet) Visit(fn func(*Flag)) { + for _, flag := range sortFlags(fs.actual) { + fn(flag) + } +} + +// Visit visits the command-line flags in lexicographical order, calling fn +// for each. It visits only those flags that have been set. +func Visit(fn func(*Flag)) { + CommandLine.Visit(fn) +} + +// Lookup returns the Flag structure of the named flag, returning nil if none exists. +func (fs *FlagSet) Lookup(name string) *Flag { + return fs.formal[name] +} + +// IsSet indicates whether the specified flag is set in the given FlagSet +func (fs *FlagSet) IsSet(name string) bool { + return fs.actual[name] != nil +} + +// Lookup returns the Flag structure of the named command-line flag, +// returning nil if none exists. +func Lookup(name string) *Flag { + return CommandLine.formal[name] +} + +// IsSet indicates whether the specified flag was specified at all on the cmd line. +func IsSet(name string) bool { + return CommandLine.IsSet(name) +} + +type nArgRequirementType int + +// Indicator used to pass to BadArgs function +const ( + Exact nArgRequirementType = iota + Max + Min +) + +type nArgRequirement struct { + Type nArgRequirementType + N int +} + +// Require adds a requirement about the number of arguments for the FlagSet. +// The first parameter can be Exact, Max, or Min to respectively specify the exact, +// the maximum, or the minimal number of arguments required. +// The actual check is done in FlagSet.CheckArgs(). +func (fs *FlagSet) Require(nArgRequirementType nArgRequirementType, nArg int) { + fs.nArgRequirements = append(fs.nArgRequirements, nArgRequirement{nArgRequirementType, nArg}) +} + +// CheckArgs uses the requirements set by FlagSet.Require() to validate +// the number of arguments. If the requirements are not met, +// an error message string is returned. +func (fs *FlagSet) CheckArgs() (message string) { + for _, req := range fs.nArgRequirements { + var arguments string + if req.N == 1 { + arguments = "1 argument" + } else { + arguments = fmt.Sprintf("%d arguments", req.N) + } + + str := func(kind string) string { + return fmt.Sprintf("%q requires %s%s", fs.name, kind, arguments) + } + + switch req.Type { + case Exact: + if fs.NArg() != req.N { + return str("") + } + case Max: + if fs.NArg() > req.N { + return str("a maximum of ") + } + case Min: + if fs.NArg() < req.N { + return str("a minimum of ") + } + } + } + return "" +} + +// Set sets the value of the named flag. +func (fs *FlagSet) Set(name, value string) error { + flag, ok := fs.formal[name] + if !ok { + return fmt.Errorf("no such flag -%v", name) + } + if err := flag.Value.Set(value); err != nil { + return err + } + if fs.actual == nil { + fs.actual = make(map[string]*Flag) + } + fs.actual[name] = flag + return nil +} + +// Set sets the value of the named command-line flag. +func Set(name, value string) error { + return CommandLine.Set(name, value) +} + +// PrintDefaults prints, to standard error unless configured +// otherwise, the default values of all defined flags in the set. +func (fs *FlagSet) PrintDefaults() { + writer := tabwriter.NewWriter(fs.Out(), 20, 1, 3, ' ', 0) + home := homedir.Get() + + // Don't substitute when HOME is / + if runtime.GOOS != "windows" && home == "/" { + home = "" + } + + // Add a blank line between cmd description and list of options + if fs.FlagCount() > 0 { + fmt.Fprintln(writer, "") + } + + fs.VisitAll(func(flag *Flag) { + format := " -%s=%s" + names := []string{} + for _, name := range flag.Names { + if name[0] != '#' { + names = append(names, name) + } + } + if len(names) > 0 && len(flag.Usage) > 0 { + val := flag.DefValue + + if home != "" && strings.HasPrefix(val, home) { + val = homedir.GetShortcutString() + val[len(home):] + } + + fmt.Fprintf(writer, format, strings.Join(names, ", -"), val) + for i, line := range strings.Split(flag.Usage, "\n") { + if i != 0 { + line = " " + line + } + fmt.Fprintln(writer, "\t", line) + } + } + }) + writer.Flush() +} + +// PrintDefaults prints to standard error the default values of all defined command-line flags. +func PrintDefaults() { + CommandLine.PrintDefaults() +} + +// defaultUsage is the default function to print a usage message. +func defaultUsage(fs *FlagSet) { + if fs.name == "" { + fmt.Fprintf(fs.Out(), "Usage:\n") + } else { + fmt.Fprintf(fs.Out(), "Usage of %s:\n", fs.name) + } + fs.PrintDefaults() +} + +// NOTE: Usage is not just defaultUsage(CommandLine) +// because it serves (via godoc flag Usage) as the example +// for how to write your own usage function. + +// Usage prints to standard error a usage message documenting all defined command-line flags. +// The function is a variable that may be changed to point to a custom function. +var Usage = func() { + fmt.Fprintf(CommandLine.Out(), "Usage of %s:\n", os.Args[0]) + PrintDefaults() +} + +// Usage prints to standard error a usage message documenting the standard command layout +// The function is a variable that may be changed to point to a custom function. +var ShortUsage = func() { + fmt.Fprintf(CommandLine.output, "Usage of %s:\n", os.Args[0]) +} + +// FlagCount returns the number of flags that have been defined. +func (fs *FlagSet) FlagCount() int { return len(sortFlags(fs.formal)) } + +// FlagCountUndeprecated returns the number of undeprecated flags that have been defined. +func (fs *FlagSet) FlagCountUndeprecated() int { + count := 0 + for _, flag := range sortFlags(fs.formal) { + for _, name := range flag.Names { + if name[0] != '#' { + count++ + break + } + } + } + return count +} + +// NFlag returns the number of flags that have been set. +func (fs *FlagSet) NFlag() int { return len(fs.actual) } + +// NFlag returns the number of command-line flags that have been set. +func NFlag() int { return len(CommandLine.actual) } + +// Arg returns the i'th argument. Arg(0) is the first remaining argument +// after flags have been processed. +func (fs *FlagSet) Arg(i int) string { + if i < 0 || i >= len(fs.args) { + return "" + } + return fs.args[i] +} + +// Arg returns the i'th command-line argument. Arg(0) is the first remaining argument +// after flags have been processed. +func Arg(i int) string { + return CommandLine.Arg(i) +} + +// NArg is the number of arguments remaining after flags have been processed. +func (fs *FlagSet) NArg() int { return len(fs.args) } + +// NArg is the number of arguments remaining after flags have been processed. +func NArg() int { return len(CommandLine.args) } + +// Args returns the non-flag arguments. +func (fs *FlagSet) Args() []string { return fs.args } + +// Args returns the non-flag command-line arguments. +func Args() []string { return CommandLine.args } + +// BoolVar defines a bool flag with specified name, default value, and usage string. +// The argument p points to a bool variable in which to store the value of the flag. +func (fs *FlagSet) BoolVar(p *bool, names []string, value bool, usage string) { + fs.Var(newBoolValue(value, p), names, usage) +} + +// BoolVar defines a bool flag with specified name, default value, and usage string. +// The argument p points to a bool variable in which to store the value of the flag. +func BoolVar(p *bool, names []string, value bool, usage string) { + CommandLine.Var(newBoolValue(value, p), names, usage) +} + +// Bool defines a bool flag with specified name, default value, and usage string. +// The return value is the address of a bool variable that stores the value of the flag. +func (fs *FlagSet) Bool(names []string, value bool, usage string) *bool { + p := new(bool) + fs.BoolVar(p, names, value, usage) + return p +} + +// Bool defines a bool flag with specified name, default value, and usage string. +// The return value is the address of a bool variable that stores the value of the flag. +func Bool(names []string, value bool, usage string) *bool { + return CommandLine.Bool(names, value, usage) +} + +// IntVar defines an int flag with specified name, default value, and usage string. +// The argument p points to an int variable in which to store the value of the flag. +func (fs *FlagSet) IntVar(p *int, names []string, value int, usage string) { + fs.Var(newIntValue(value, p), names, usage) +} + +// IntVar defines an int flag with specified name, default value, and usage string. +// The argument p points to an int variable in which to store the value of the flag. +func IntVar(p *int, names []string, value int, usage string) { + CommandLine.Var(newIntValue(value, p), names, usage) +} + +// Int defines an int flag with specified name, default value, and usage string. +// The return value is the address of an int variable that stores the value of the flag. +func (fs *FlagSet) Int(names []string, value int, usage string) *int { + p := new(int) + fs.IntVar(p, names, value, usage) + return p +} + +// Int defines an int flag with specified name, default value, and usage string. +// The return value is the address of an int variable that stores the value of the flag. +func Int(names []string, value int, usage string) *int { + return CommandLine.Int(names, value, usage) +} + +// Int64Var defines an int64 flag with specified name, default value, and usage string. +// The argument p points to an int64 variable in which to store the value of the flag. +func (fs *FlagSet) Int64Var(p *int64, names []string, value int64, usage string) { + fs.Var(newInt64Value(value, p), names, usage) +} + +// Int64Var defines an int64 flag with specified name, default value, and usage string. +// The argument p points to an int64 variable in which to store the value of the flag. +func Int64Var(p *int64, names []string, value int64, usage string) { + CommandLine.Var(newInt64Value(value, p), names, usage) +} + +// Int64 defines an int64 flag with specified name, default value, and usage string. +// The return value is the address of an int64 variable that stores the value of the flag. +func (fs *FlagSet) Int64(names []string, value int64, usage string) *int64 { + p := new(int64) + fs.Int64Var(p, names, value, usage) + return p +} + +// Int64 defines an int64 flag with specified name, default value, and usage string. +// The return value is the address of an int64 variable that stores the value of the flag. +func Int64(names []string, value int64, usage string) *int64 { + return CommandLine.Int64(names, value, usage) +} + +// UintVar defines a uint flag with specified name, default value, and usage string. +// The argument p points to a uint variable in which to store the value of the flag. +func (fs *FlagSet) UintVar(p *uint, names []string, value uint, usage string) { + fs.Var(newUintValue(value, p), names, usage) +} + +// UintVar defines a uint flag with specified name, default value, and usage string. +// The argument p points to a uint variable in which to store the value of the flag. +func UintVar(p *uint, names []string, value uint, usage string) { + CommandLine.Var(newUintValue(value, p), names, usage) +} + +// Uint defines a uint flag with specified name, default value, and usage string. +// The return value is the address of a uint variable that stores the value of the flag. +func (fs *FlagSet) Uint(names []string, value uint, usage string) *uint { + p := new(uint) + fs.UintVar(p, names, value, usage) + return p +} + +// Uint defines a uint flag with specified name, default value, and usage string. +// The return value is the address of a uint variable that stores the value of the flag. +func Uint(names []string, value uint, usage string) *uint { + return CommandLine.Uint(names, value, usage) +} + +// Uint64Var defines a uint64 flag with specified name, default value, and usage string. +// The argument p points to a uint64 variable in which to store the value of the flag. +func (fs *FlagSet) Uint64Var(p *uint64, names []string, value uint64, usage string) { + fs.Var(newUint64Value(value, p), names, usage) +} + +// Uint64Var defines a uint64 flag with specified name, default value, and usage string. +// The argument p points to a uint64 variable in which to store the value of the flag. +func Uint64Var(p *uint64, names []string, value uint64, usage string) { + CommandLine.Var(newUint64Value(value, p), names, usage) +} + +// Uint64 defines a uint64 flag with specified name, default value, and usage string. +// The return value is the address of a uint64 variable that stores the value of the flag. +func (fs *FlagSet) Uint64(names []string, value uint64, usage string) *uint64 { + p := new(uint64) + fs.Uint64Var(p, names, value, usage) + return p +} + +// Uint64 defines a uint64 flag with specified name, default value, and usage string. +// The return value is the address of a uint64 variable that stores the value of the flag. +func Uint64(names []string, value uint64, usage string) *uint64 { + return CommandLine.Uint64(names, value, usage) +} + +// StringVar defines a string flag with specified name, default value, and usage string. +// The argument p points to a string variable in which to store the value of the flag. +func (fs *FlagSet) StringVar(p *string, names []string, value string, usage string) { + fs.Var(newStringValue(value, p), names, usage) +} + +// StringVar defines a string flag with specified name, default value, and usage string. +// The argument p points to a string variable in which to store the value of the flag. +func StringVar(p *string, names []string, value string, usage string) { + CommandLine.Var(newStringValue(value, p), names, usage) +} + +// String defines a string flag with specified name, default value, and usage string. +// The return value is the address of a string variable that stores the value of the flag. +func (fs *FlagSet) String(names []string, value string, usage string) *string { + p := new(string) + fs.StringVar(p, names, value, usage) + return p +} + +// String defines a string flag with specified name, default value, and usage string. +// The return value is the address of a string variable that stores the value of the flag. +func String(names []string, value string, usage string) *string { + return CommandLine.String(names, value, usage) +} + +// Float64Var defines a float64 flag with specified name, default value, and usage string. +// The argument p points to a float64 variable in which to store the value of the flag. +func (fs *FlagSet) Float64Var(p *float64, names []string, value float64, usage string) { + fs.Var(newFloat64Value(value, p), names, usage) +} + +// Float64Var defines a float64 flag with specified name, default value, and usage string. +// The argument p points to a float64 variable in which to store the value of the flag. +func Float64Var(p *float64, names []string, value float64, usage string) { + CommandLine.Var(newFloat64Value(value, p), names, usage) +} + +// Float64 defines a float64 flag with specified name, default value, and usage string. +// The return value is the address of a float64 variable that stores the value of the flag. +func (fs *FlagSet) Float64(names []string, value float64, usage string) *float64 { + p := new(float64) + fs.Float64Var(p, names, value, usage) + return p +} + +// Float64 defines a float64 flag with specified name, default value, and usage string. +// The return value is the address of a float64 variable that stores the value of the flag. +func Float64(names []string, value float64, usage string) *float64 { + return CommandLine.Float64(names, value, usage) +} + +// DurationVar defines a time.Duration flag with specified name, default value, and usage string. +// The argument p points to a time.Duration variable in which to store the value of the flag. +func (fs *FlagSet) DurationVar(p *time.Duration, names []string, value time.Duration, usage string) { + fs.Var(newDurationValue(value, p), names, usage) +} + +// DurationVar defines a time.Duration flag with specified name, default value, and usage string. +// The argument p points to a time.Duration variable in which to store the value of the flag. +func DurationVar(p *time.Duration, names []string, value time.Duration, usage string) { + CommandLine.Var(newDurationValue(value, p), names, usage) +} + +// Duration defines a time.Duration flag with specified name, default value, and usage string. +// The return value is the address of a time.Duration variable that stores the value of the flag. +func (fs *FlagSet) Duration(names []string, value time.Duration, usage string) *time.Duration { + p := new(time.Duration) + fs.DurationVar(p, names, value, usage) + return p +} + +// Duration defines a time.Duration flag with specified name, default value, and usage string. +// The return value is the address of a time.Duration variable that stores the value of the flag. +func Duration(names []string, value time.Duration, usage string) *time.Duration { + return CommandLine.Duration(names, value, usage) +} + +// Var defines a flag with the specified name and usage string. The type and +// value of the flag are represented by the first argument, of type Value, which +// typically holds a user-defined implementation of Value. For instance, the +// caller could create a flag that turns a comma-separated string into a slice +// of strings by giving the slice the methods of Value; in particular, Set would +// decompose the comma-separated string into the slice. +func (fs *FlagSet) Var(value Value, names []string, usage string) { + // Remember the default value as a string; it won't change. + flag := &Flag{names, usage, value, value.String()} + for _, name := range names { + name = strings.TrimPrefix(name, "#") + _, alreadythere := fs.formal[name] + if alreadythere { + var msg string + if fs.name == "" { + msg = fmt.Sprintf("flag redefined: %s", name) + } else { + msg = fmt.Sprintf("%s flag redefined: %s", fs.name, name) + } + fmt.Fprintln(fs.Out(), msg) + panic(msg) // Happens only if flags are declared with identical names + } + if fs.formal == nil { + fs.formal = make(map[string]*Flag) + } + fs.formal[name] = flag + } +} + +// Var defines a flag with the specified name and usage string. The type and +// value of the flag are represented by the first argument, of type Value, which +// typically holds a user-defined implementation of Value. For instance, the +// caller could create a flag that turns a comma-separated string into a slice +// of strings by giving the slice the methods of Value; in particular, Set would +// decompose the comma-separated string into the slice. +func Var(value Value, names []string, usage string) { + CommandLine.Var(value, names, usage) +} + +// failf prints to standard error a formatted error and usage message and +// returns the error. +func (fs *FlagSet) failf(format string, a ...interface{}) error { + err := fmt.Errorf(format, a...) + fmt.Fprintln(fs.Out(), err) + if os.Args[0] == fs.name { + fmt.Fprintf(fs.Out(), "See '%s --help'.\n", os.Args[0]) + } else { + fmt.Fprintf(fs.Out(), "See '%s %s --help'.\n", os.Args[0], fs.name) + } + return err +} + +// usage calls the Usage method for the flag set, or the usage function if +// the flag set is CommandLine. +func (fs *FlagSet) usage() { + if fs == CommandLine { + Usage() + } else if fs.Usage == nil { + defaultUsage(fs) + } else { + fs.Usage() + } +} + +func trimQuotes(str string) string { + if len(str) == 0 { + return str + } + type quote struct { + start, end byte + } + + // All valid quote types. + quotes := []quote{ + // Double quotes + { + start: '"', + end: '"', + }, + + // Single quotes + { + start: '\'', + end: '\'', + }, + } + + for _, quote := range quotes { + // Only strip if outermost match. + if str[0] == quote.start && str[len(str)-1] == quote.end { + str = str[1 : len(str)-1] + break + } + } + + return str +} + +// parseOne parses one flag. It reports whether a flag was seen. +func (fs *FlagSet) parseOne() (bool, string, error) { + if len(fs.args) == 0 { + return false, "", nil + } + s := fs.args[0] + if len(s) == 0 || s[0] != '-' || len(s) == 1 { + return false, "", nil + } + if s[1] == '-' && len(s) == 2 { // "--" terminates the flags + fs.args = fs.args[1:] + return false, "", nil + } + name := s[1:] + if len(name) == 0 || name[0] == '=' { + return false, "", fs.failf("bad flag syntax: %s", s) + } + + // it's a flag. does it have an argument? + fs.args = fs.args[1:] + hasValue := false + value := "" + if i := strings.Index(name, "="); i != -1 { + value = trimQuotes(name[i+1:]) + hasValue = true + name = name[:i] + } + + m := fs.formal + flag, alreadythere := m[name] // BUG + if !alreadythere { + if name == "-help" || name == "help" || name == "h" { // special case for nice help message. + fs.usage() + return false, "", ErrHelp + } + if len(name) > 0 && name[0] == '-' { + return false, "", fs.failf("flag provided but not defined: -%s", name) + } + return false, name, ErrRetry + } + if fv, ok := flag.Value.(boolFlag); ok && fv.IsBoolFlag() { // special case: doesn't need an arg + if hasValue { + if err := fv.Set(value); err != nil { + return false, "", fs.failf("invalid boolean value %q for -%s: %v", value, name, err) + } + } else { + fv.Set("true") + } + } else { + // It must have a value, which might be the next argument. + if !hasValue && len(fs.args) > 0 { + // value is the next arg + hasValue = true + value, fs.args = fs.args[0], fs.args[1:] + } + if !hasValue { + return false, "", fs.failf("flag needs an argument: -%s", name) + } + if err := flag.Value.Set(value); err != nil { + return false, "", fs.failf("invalid value %q for flag -%s: %v", value, name, err) + } + } + if fs.actual == nil { + fs.actual = make(map[string]*Flag) + } + fs.actual[name] = flag + for i, n := range flag.Names { + if n == fmt.Sprintf("#%s", name) { + replacement := "" + for j := i; j < len(flag.Names); j++ { + if flag.Names[j][0] != '#' { + replacement = flag.Names[j] + break + } + } + if replacement != "" { + fmt.Fprintf(fs.Out(), "Warning: '-%s' is deprecated, it will be replaced by '-%s' soon. See usage.\n", name, replacement) + } else { + fmt.Fprintf(fs.Out(), "Warning: '-%s' is deprecated, it will be removed soon. See usage.\n", name) + } + } + } + return true, "", nil +} + +// Parse parses flag definitions from the argument list, which should not +// include the command name. Must be called after all flags in the FlagSet +// are defined and before flags are accessed by the program. +// The return value will be ErrHelp if -help was set but not defined. +func (fs *FlagSet) Parse(arguments []string) error { + fs.parsed = true + fs.args = arguments + for { + seen, name, err := fs.parseOne() + if seen { + continue + } + if err == nil { + break + } + if err == ErrRetry { + if len(name) > 1 { + err = nil + for _, letter := range strings.Split(name, "") { + fs.args = append([]string{"-" + letter}, fs.args...) + seen2, _, err2 := fs.parseOne() + if seen2 { + continue + } + if err2 != nil { + err = fs.failf("flag provided but not defined: -%s", name) + break + } + } + if err == nil { + continue + } + } else { + err = fs.failf("flag provided but not defined: -%s", name) + } + } + switch fs.errorHandling { + case ContinueOnError: + return err + case ExitOnError: + os.Exit(2) + case PanicOnError: + panic(err) + } + } + return nil +} + +// ParseFlags is a utility function that adds a help flag if withHelp is true, +// calls fs.Parse(args) and prints a relevant error message if there are +// incorrect number of arguments. It returns error only if error handling is +// set to ContinueOnError and parsing fails. If error handling is set to +// ExitOnError, it's safe to ignore the return value. +func (fs *FlagSet) ParseFlags(args []string, withHelp bool) error { + var help *bool + if withHelp { + help = fs.Bool([]string{"#help", "-help"}, false, "Print usage") + } + if err := fs.Parse(args); err != nil { + return err + } + if help != nil && *help { + fs.SetOutput(os.Stdout) + fs.Usage() + os.Exit(0) + } + if str := fs.CheckArgs(); str != "" { + fs.SetOutput(os.Stderr) + fs.ReportError(str, withHelp) + fs.ShortUsage() + os.Exit(1) + } + return nil +} + +// ReportError is a utility method that prints a user-friendly message +// containing the error that occured during parsing and a suggestion to get help +func (fs *FlagSet) ReportError(str string, withHelp bool) { + if withHelp { + if os.Args[0] == fs.Name() { + str += ".\nSee '" + os.Args[0] + " --help'" + } else { + str += ".\nSee '" + os.Args[0] + " " + fs.Name() + " --help'" + } + } + fmt.Fprintf(fs.Out(), "docker: %s.\n", str) +} + +// Parsed reports whether fs.Parse has been called. +func (fs *FlagSet) Parsed() bool { + return fs.parsed +} + +// Parse parses the command-line flags from os.Args[1:]. Must be called +// after all flags are defined and before flags are accessed by the program. +func Parse() { + // Ignore errors; CommandLine is set for ExitOnError. + CommandLine.Parse(os.Args[1:]) +} + +// Parsed returns true if the command-line flags have been parsed. +func Parsed() bool { + return CommandLine.Parsed() +} + +// CommandLine is the default set of command-line flags, parsed from os.Args. +// The top-level functions such as BoolVar, Arg, and on are wrappers for the +// methods of CommandLine. +var CommandLine = NewFlagSet(os.Args[0], ExitOnError) + +// NewFlagSet returns a new, empty flag set with the specified name and +// error handling property. +func NewFlagSet(name string, errorHandling ErrorHandling) *FlagSet { + f := &FlagSet{ + name: name, + errorHandling: errorHandling, + } + return f +} + +// Init sets the name and error handling property for a flag set. +// By default, the zero FlagSet uses an empty name and the +// ContinueOnError error handling policy. +func (fs *FlagSet) Init(name string, errorHandling ErrorHandling) { + fs.name = name + fs.errorHandling = errorHandling +} + +type mergeVal struct { + Value + key string + fset *FlagSet +} + +func (v mergeVal) Set(s string) error { + return v.fset.Set(v.key, s) +} + +func (v mergeVal) IsBoolFlag() bool { + if b, ok := v.Value.(boolFlag); ok { + return b.IsBoolFlag() + } + return false +} + +// Merge is an helper function that merges n FlagSets into a single dest FlagSet +// In case of name collision between the flagsets it will apply +// the destination FlagSet's errorHandling behaviour. +func Merge(dest *FlagSet, flagsets ...*FlagSet) error { + for _, fset := range flagsets { + for k, f := range fset.formal { + if _, ok := dest.formal[k]; ok { + var err error + if fset.name == "" { + err = fmt.Errorf("flag redefined: %s", k) + } else { + err = fmt.Errorf("%s flag redefined: %s", fset.name, k) + } + fmt.Fprintln(fset.Out(), err.Error()) + // Happens only if flags are declared with identical names + switch dest.errorHandling { + case ContinueOnError: + return err + case ExitOnError: + os.Exit(2) + case PanicOnError: + panic(err) + } + } + newF := *f + newF.Value = mergeVal{f.Value, k, fset} + dest.formal[k] = &newF + } + } + return nil +} + +// IsEmpty reports if the FlagSet is actually empty. +func (fs *FlagSet) IsEmpty() bool { + return len(fs.actual) == 0 +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/mflag/flag_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/mflag/flag_test.go new file mode 100644 index 0000000000000..85f32c8aa4620 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/mflag/flag_test.go @@ -0,0 +1,516 @@ +// Copyright 2014-2015 The Docker & Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package mflag + +import ( + "bytes" + "fmt" + "os" + "sort" + "strings" + "testing" + "time" +) + +// ResetForTesting clears all flag state and sets the usage function as directed. +// After calling ResetForTesting, parse errors in flag handling will not +// exit the program. +func ResetForTesting(usage func()) { + CommandLine = NewFlagSet(os.Args[0], ContinueOnError) + Usage = usage +} +func boolString(s string) string { + if s == "0" { + return "false" + } + return "true" +} + +func TestEverything(t *testing.T) { + ResetForTesting(nil) + Bool([]string{"test_bool"}, false, "bool value") + Int([]string{"test_int"}, 0, "int value") + Int64([]string{"test_int64"}, 0, "int64 value") + Uint([]string{"test_uint"}, 0, "uint value") + Uint64([]string{"test_uint64"}, 0, "uint64 value") + String([]string{"test_string"}, "0", "string value") + Float64([]string{"test_float64"}, 0, "float64 value") + Duration([]string{"test_duration"}, 0, "time.Duration value") + + m := make(map[string]*Flag) + desired := "0" + visitor := func(f *Flag) { + for _, name := range f.Names { + if len(name) > 5 && name[0:5] == "test_" { + m[name] = f + ok := false + switch { + case f.Value.String() == desired: + ok = true + case name == "test_bool" && f.Value.String() == boolString(desired): + ok = true + case name == "test_duration" && f.Value.String() == desired+"s": + ok = true + } + if !ok { + t.Error("Visit: bad value", f.Value.String(), "for", name) + } + } + } + } + VisitAll(visitor) + if len(m) != 8 { + t.Error("VisitAll misses some flags") + for k, v := range m { + t.Log(k, *v) + } + } + m = make(map[string]*Flag) + Visit(visitor) + if len(m) != 0 { + t.Errorf("Visit sees unset flags") + for k, v := range m { + t.Log(k, *v) + } + } + // Now set all flags + Set("test_bool", "true") + Set("test_int", "1") + Set("test_int64", "1") + Set("test_uint", "1") + Set("test_uint64", "1") + Set("test_string", "1") + Set("test_float64", "1") + Set("test_duration", "1s") + desired = "1" + Visit(visitor) + if len(m) != 8 { + t.Error("Visit fails after set") + for k, v := range m { + t.Log(k, *v) + } + } + // Now test they're visited in sort order. + var flagNames []string + Visit(func(f *Flag) { + for _, name := range f.Names { + flagNames = append(flagNames, name) + } + }) + if !sort.StringsAreSorted(flagNames) { + t.Errorf("flag names not sorted: %v", flagNames) + } +} + +func TestGet(t *testing.T) { + ResetForTesting(nil) + Bool([]string{"test_bool"}, true, "bool value") + Int([]string{"test_int"}, 1, "int value") + Int64([]string{"test_int64"}, 2, "int64 value") + Uint([]string{"test_uint"}, 3, "uint value") + Uint64([]string{"test_uint64"}, 4, "uint64 value") + String([]string{"test_string"}, "5", "string value") + Float64([]string{"test_float64"}, 6, "float64 value") + Duration([]string{"test_duration"}, 7, "time.Duration value") + + visitor := func(f *Flag) { + for _, name := range f.Names { + if len(name) > 5 && name[0:5] == "test_" { + g, ok := f.Value.(Getter) + if !ok { + t.Errorf("Visit: value does not satisfy Getter: %T", f.Value) + return + } + switch name { + case "test_bool": + ok = g.Get() == true + case "test_int": + ok = g.Get() == int(1) + case "test_int64": + ok = g.Get() == int64(2) + case "test_uint": + ok = g.Get() == uint(3) + case "test_uint64": + ok = g.Get() == uint64(4) + case "test_string": + ok = g.Get() == "5" + case "test_float64": + ok = g.Get() == float64(6) + case "test_duration": + ok = g.Get() == time.Duration(7) + } + if !ok { + t.Errorf("Visit: bad value %T(%v) for %s", g.Get(), g.Get(), name) + } + } + } + } + VisitAll(visitor) +} + +func testParse(f *FlagSet, t *testing.T) { + if f.Parsed() { + t.Error("f.Parse() = true before Parse") + } + boolFlag := f.Bool([]string{"bool"}, false, "bool value") + bool2Flag := f.Bool([]string{"bool2"}, false, "bool2 value") + f.Bool([]string{"bool3"}, false, "bool3 value") + bool4Flag := f.Bool([]string{"bool4"}, false, "bool4 value") + intFlag := f.Int([]string{"-int"}, 0, "int value") + int64Flag := f.Int64([]string{"-int64"}, 0, "int64 value") + uintFlag := f.Uint([]string{"uint"}, 0, "uint value") + uint64Flag := f.Uint64([]string{"-uint64"}, 0, "uint64 value") + stringFlag := f.String([]string{"string"}, "0", "string value") + f.String([]string{"string2"}, "0", "string2 value") + singleQuoteFlag := f.String([]string{"squote"}, "", "single quoted value") + doubleQuoteFlag := f.String([]string{"dquote"}, "", "double quoted value") + mixedQuoteFlag := f.String([]string{"mquote"}, "", "mixed quoted value") + mixed2QuoteFlag := f.String([]string{"mquote2"}, "", "mixed2 quoted value") + nestedQuoteFlag := f.String([]string{"nquote"}, "", "nested quoted value") + nested2QuoteFlag := f.String([]string{"nquote2"}, "", "nested2 quoted value") + float64Flag := f.Float64([]string{"float64"}, 0, "float64 value") + durationFlag := f.Duration([]string{"duration"}, 5*time.Second, "time.Duration value") + extra := "one-extra-argument" + args := []string{ + "-bool", + "-bool2=true", + "-bool4=false", + "--int", "22", + "--int64", "0x23", + "-uint", "24", + "--uint64", "25", + "-string", "hello", + "-squote='single'", + `-dquote="double"`, + `-mquote='mixed"`, + `-mquote2="mixed2'`, + `-nquote="'single nested'"`, + `-nquote2='"double nested"'`, + "-float64", "2718e28", + "-duration", "2m", + extra, + } + if err := f.Parse(args); err != nil { + t.Fatal(err) + } + if !f.Parsed() { + t.Error("f.Parse() = false after Parse") + } + if *boolFlag != true { + t.Error("bool flag should be true, is ", *boolFlag) + } + if *bool2Flag != true { + t.Error("bool2 flag should be true, is ", *bool2Flag) + } + if !f.IsSet("bool2") { + t.Error("bool2 should be marked as set") + } + if f.IsSet("bool3") { + t.Error("bool3 should not be marked as set") + } + if !f.IsSet("bool4") { + t.Error("bool4 should be marked as set") + } + if *bool4Flag != false { + t.Error("bool4 flag should be false, is ", *bool4Flag) + } + if *intFlag != 22 { + t.Error("int flag should be 22, is ", *intFlag) + } + if *int64Flag != 0x23 { + t.Error("int64 flag should be 0x23, is ", *int64Flag) + } + if *uintFlag != 24 { + t.Error("uint flag should be 24, is ", *uintFlag) + } + if *uint64Flag != 25 { + t.Error("uint64 flag should be 25, is ", *uint64Flag) + } + if *stringFlag != "hello" { + t.Error("string flag should be `hello`, is ", *stringFlag) + } + if !f.IsSet("string") { + t.Error("string flag should be marked as set") + } + if f.IsSet("string2") { + t.Error("string2 flag should not be marked as set") + } + if *singleQuoteFlag != "single" { + t.Error("single quote string flag should be `single`, is ", *singleQuoteFlag) + } + if *doubleQuoteFlag != "double" { + t.Error("double quote string flag should be `double`, is ", *doubleQuoteFlag) + } + if *mixedQuoteFlag != `'mixed"` { + t.Error("mixed quote string flag should be `'mixed\"`, is ", *mixedQuoteFlag) + } + if *mixed2QuoteFlag != `"mixed2'` { + t.Error("mixed2 quote string flag should be `\"mixed2'`, is ", *mixed2QuoteFlag) + } + if *nestedQuoteFlag != "'single nested'" { + t.Error("nested quote string flag should be `'single nested'`, is ", *nestedQuoteFlag) + } + if *nested2QuoteFlag != `"double nested"` { + t.Error("double quote string flag should be `\"double nested\"`, is ", *nested2QuoteFlag) + } + if *float64Flag != 2718e28 { + t.Error("float64 flag should be 2718e28, is ", *float64Flag) + } + if *durationFlag != 2*time.Minute { + t.Error("duration flag should be 2m, is ", *durationFlag) + } + if len(f.Args()) != 1 { + t.Error("expected one argument, got", len(f.Args())) + } else if f.Args()[0] != extra { + t.Errorf("expected argument %q got %q", extra, f.Args()[0]) + } +} + +func testPanic(f *FlagSet, t *testing.T) { + f.Int([]string{"-int"}, 0, "int value") + if f.Parsed() { + t.Error("f.Parse() = true before Parse") + } + args := []string{ + "-int", "21", + } + f.Parse(args) +} + +func TestParsePanic(t *testing.T) { + ResetForTesting(func() {}) + testPanic(CommandLine, t) +} + +func TestParse(t *testing.T) { + ResetForTesting(func() { t.Error("bad parse") }) + testParse(CommandLine, t) +} + +func TestFlagSetParse(t *testing.T) { + testParse(NewFlagSet("test", ContinueOnError), t) +} + +// Declare a user-defined flag type. +type flagVar []string + +func (f *flagVar) String() string { + return fmt.Sprint([]string(*f)) +} + +func (f *flagVar) Set(value string) error { + *f = append(*f, value) + return nil +} + +func TestUserDefined(t *testing.T) { + var flags FlagSet + flags.Init("test", ContinueOnError) + var v flagVar + flags.Var(&v, []string{"v"}, "usage") + if err := flags.Parse([]string{"-v", "1", "-v", "2", "-v=3"}); err != nil { + t.Error(err) + } + if len(v) != 3 { + t.Fatal("expected 3 args; got ", len(v)) + } + expect := "[1 2 3]" + if v.String() != expect { + t.Errorf("expected value %q got %q", expect, v.String()) + } +} + +// Declare a user-defined boolean flag type. +type boolFlagVar struct { + count int +} + +func (b *boolFlagVar) String() string { + return fmt.Sprintf("%d", b.count) +} + +func (b *boolFlagVar) Set(value string) error { + if value == "true" { + b.count++ + } + return nil +} + +func (b *boolFlagVar) IsBoolFlag() bool { + return b.count < 4 +} + +func TestUserDefinedBool(t *testing.T) { + var flags FlagSet + flags.Init("test", ContinueOnError) + var b boolFlagVar + var err error + flags.Var(&b, []string{"b"}, "usage") + if err = flags.Parse([]string{"-b", "-b", "-b", "-b=true", "-b=false", "-b", "barg", "-b"}); err != nil { + if b.count < 4 { + t.Error(err) + } + } + + if b.count != 4 { + t.Errorf("want: %d; got: %d", 4, b.count) + } + + if err == nil { + t.Error("expected error; got none") + } +} + +func TestSetOutput(t *testing.T) { + var flags FlagSet + var buf bytes.Buffer + flags.SetOutput(&buf) + flags.Init("test", ContinueOnError) + flags.Parse([]string{"-unknown"}) + if out := buf.String(); !strings.Contains(out, "-unknown") { + t.Logf("expected output mentioning unknown; got %q", out) + } +} + +// This tests that one can reset the flags. This still works but not well, and is +// superseded by FlagSet. +func TestChangingArgs(t *testing.T) { + ResetForTesting(func() { t.Fatal("bad parse") }) + oldArgs := os.Args + defer func() { os.Args = oldArgs }() + os.Args = []string{"cmd", "-before", "subcmd", "-after", "args"} + before := Bool([]string{"before"}, false, "") + if err := CommandLine.Parse(os.Args[1:]); err != nil { + t.Fatal(err) + } + cmd := Arg(0) + os.Args = Args() + after := Bool([]string{"after"}, false, "") + Parse() + args := Args() + + if !*before || cmd != "subcmd" || !*after || len(args) != 1 || args[0] != "args" { + t.Fatalf("expected true subcmd true [args] got %v %v %v %v", *before, cmd, *after, args) + } +} + +// Test that -help invokes the usage message and returns ErrHelp. +func TestHelp(t *testing.T) { + var helpCalled = false + fs := NewFlagSet("help test", ContinueOnError) + fs.Usage = func() { helpCalled = true } + var flag bool + fs.BoolVar(&flag, []string{"flag"}, false, "regular flag") + // Regular flag invocation should work + err := fs.Parse([]string{"-flag=true"}) + if err != nil { + t.Fatal("expected no error; got ", err) + } + if !flag { + t.Error("flag was not set by -flag") + } + if helpCalled { + t.Error("help called for regular flag") + helpCalled = false // reset for next test + } + // Help flag should work as expected. + err = fs.Parse([]string{"-help"}) + if err == nil { + t.Fatal("error expected") + } + if err != ErrHelp { + t.Fatal("expected ErrHelp; got ", err) + } + if !helpCalled { + t.Fatal("help was not called") + } + // If we define a help flag, that should override. + var help bool + fs.BoolVar(&help, []string{"help"}, false, "help flag") + helpCalled = false + err = fs.Parse([]string{"-help"}) + if err != nil { + t.Fatal("expected no error for defined -help; got ", err) + } + if helpCalled { + t.Fatal("help was called; should not have been for defined help flag") + } +} + +// Test the flag count functions. +func TestFlagCounts(t *testing.T) { + fs := NewFlagSet("help test", ContinueOnError) + var flag bool + fs.BoolVar(&flag, []string{"flag1"}, false, "regular flag") + fs.BoolVar(&flag, []string{"#deprecated1"}, false, "regular flag") + fs.BoolVar(&flag, []string{"f", "flag2"}, false, "regular flag") + fs.BoolVar(&flag, []string{"#d", "#deprecated2"}, false, "regular flag") + fs.BoolVar(&flag, []string{"flag3"}, false, "regular flag") + fs.BoolVar(&flag, []string{"g", "#flag4", "-flag4"}, false, "regular flag") + + if fs.FlagCount() != 6 { + t.Fatal("FlagCount wrong. ", fs.FlagCount()) + } + if fs.FlagCountUndeprecated() != 4 { + t.Fatal("FlagCountUndeprecated wrong. ", fs.FlagCountUndeprecated()) + } + if fs.NFlag() != 0 { + t.Fatal("NFlag wrong. ", fs.NFlag()) + } + err := fs.Parse([]string{"-fd", "-g", "-flag4"}) + if err != nil { + t.Fatal("expected no error for defined -help; got ", err) + } + if fs.NFlag() != 4 { + t.Fatal("NFlag wrong. ", fs.NFlag()) + } +} + +// Show up bug in sortFlags +func TestSortFlags(t *testing.T) { + fs := NewFlagSet("help TestSortFlags", ContinueOnError) + + var err error + + var b bool + fs.BoolVar(&b, []string{"b", "-banana"}, false, "usage") + + err = fs.Parse([]string{"--banana=true"}) + if err != nil { + t.Fatal("expected no error; got ", err) + } + + count := 0 + + fs.VisitAll(func(flag *Flag) { + count++ + if flag == nil { + t.Fatal("VisitAll should not return a nil flag") + } + }) + flagcount := fs.FlagCount() + if flagcount != count { + t.Fatalf("FlagCount (%d) != number (%d) of elements visited", flagcount, count) + } + // Make sure its idempotent + if flagcount != fs.FlagCount() { + t.Fatalf("FlagCount (%d) != fs.FlagCount() (%d) of elements visited", flagcount, fs.FlagCount()) + } + + count = 0 + fs.Visit(func(flag *Flag) { + count++ + if flag == nil { + t.Fatal("Visit should not return a nil flag") + } + }) + nflag := fs.NFlag() + if nflag != count { + t.Fatalf("NFlag (%d) != number (%d) of elements visited", nflag, count) + } + if nflag != fs.NFlag() { + t.Fatalf("NFlag (%d) != fs.NFlag() (%d) of elements visited", nflag, fs.NFlag()) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers/parsers.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers/parsers.go new file mode 100644 index 0000000000000..e326a11911818 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers/parsers.go @@ -0,0 +1,187 @@ +// Package parsers provides helper functions to parse and validate different type +// of string. It can be hosts, unix addresses, tcp addresses, filters, kernel +// operating system versions. +package parsers + +import ( + "fmt" + "net/url" + "path" + "runtime" + "strconv" + "strings" +) + +// ParseHost parses the specified address and returns an address that will be used as the host. +// Depending of the address specified, will use the defaultTCPAddr or defaultUnixAddr +// FIXME: Change this not to receive default value as parameter +func ParseHost(defaultTCPAddr, defaultUnixAddr, addr string) (string, error) { + addr = strings.TrimSpace(addr) + if addr == "" { + if runtime.GOOS != "windows" { + addr = fmt.Sprintf("unix://%s", defaultUnixAddr) + } else { + // Note - defaultTCPAddr already includes tcp:// prefix + addr = defaultTCPAddr + } + } + addrParts := strings.Split(addr, "://") + if len(addrParts) == 1 { + addrParts = []string{"tcp", addrParts[0]} + } + + switch addrParts[0] { + case "tcp": + return ParseTCPAddr(addrParts[1], defaultTCPAddr) + case "unix": + return ParseUnixAddr(addrParts[1], defaultUnixAddr) + case "fd": + return addr, nil + default: + return "", fmt.Errorf("Invalid bind address format: %s", addr) + } +} + +// ParseUnixAddr parses and validates that the specified address is a valid UNIX +// socket address. It returns a formatted UNIX socket address, either using the +// address parsed from addr, or the contents of defaultAddr if addr is a blank +// string. +func ParseUnixAddr(addr string, defaultAddr string) (string, error) { + addr = strings.TrimPrefix(addr, "unix://") + if strings.Contains(addr, "://") { + return "", fmt.Errorf("Invalid proto, expected unix: %s", addr) + } + if addr == "" { + addr = defaultAddr + } + return fmt.Sprintf("unix://%s", addr), nil +} + +// ParseTCPAddr parses and validates that the specified address is a valid TCP +// address. It returns a formatted TCP address, either using the address parsed +// from addr, or the contents of defaultAddr if addr is a blank string. +func ParseTCPAddr(addr string, defaultAddr string) (string, error) { + addr = strings.TrimPrefix(addr, "tcp://") + if strings.Contains(addr, "://") || addr == "" { + return "", fmt.Errorf("Invalid proto, expected tcp: %s", addr) + } + + u, err := url.Parse("tcp://" + addr) + if err != nil { + return "", err + } + hostParts := strings.Split(u.Host, ":") + if len(hostParts) != 2 { + return "", fmt.Errorf("Invalid bind address format: %s", addr) + } + host := hostParts[0] + if host == "" { + host = defaultAddr + } + + p, err := strconv.Atoi(hostParts[1]) + if err != nil && p == 0 { + return "", fmt.Errorf("Invalid bind address format: %s", addr) + } + return fmt.Sprintf("tcp://%s:%d%s", host, p, u.Path), nil +} + +// ParseRepositoryTag gets a repos name and returns the right reposName + tag|digest +// The tag can be confusing because of a port in a repository name. +// Ex: localhost.localdomain:5000/samalba/hipache:latest +// Digest ex: localhost:5000/foo/bar@sha256:bc8813ea7b3603864987522f02a76101c17ad122e1c46d790efc0fca78ca7bfb +func ParseRepositoryTag(repos string) (string, string) { + n := strings.Index(repos, "@") + if n >= 0 { + parts := strings.Split(repos, "@") + return parts[0], parts[1] + } + n = strings.LastIndex(repos, ":") + if n < 0 { + return repos, "" + } + if tag := repos[n+1:]; !strings.Contains(tag, "/") { + return repos[:n], tag + } + return repos, "" +} + +// PartParser parses and validates the specified string (data) using the specified template +// e.g. ip:public:private -> 192.168.0.1:80:8000 +func PartParser(template, data string) (map[string]string, error) { + // ip:public:private + var ( + templateParts = strings.Split(template, ":") + parts = strings.Split(data, ":") + out = make(map[string]string, len(templateParts)) + ) + if len(parts) != len(templateParts) { + return nil, fmt.Errorf("Invalid format to parse. %s should match template %s", data, template) + } + + for i, t := range templateParts { + value := "" + if len(parts) > i { + value = parts[i] + } + out[t] = value + } + return out, nil +} + +// ParseKeyValueOpt parses and validates the specified string as a key/value pair (key=value) +func ParseKeyValueOpt(opt string) (string, string, error) { + parts := strings.SplitN(opt, "=", 2) + if len(parts) != 2 { + return "", "", fmt.Errorf("Unable to parse key/value option: %s", opt) + } + return strings.TrimSpace(parts[0]), strings.TrimSpace(parts[1]), nil +} + +// ParsePortRange parses and validates the specified string as a port-range (8000-9000) +func ParsePortRange(ports string) (uint64, uint64, error) { + if ports == "" { + return 0, 0, fmt.Errorf("Empty string specified for ports.") + } + if !strings.Contains(ports, "-") { + start, err := strconv.ParseUint(ports, 10, 16) + end := start + return start, end, err + } + + parts := strings.Split(ports, "-") + start, err := strconv.ParseUint(parts[0], 10, 16) + if err != nil { + return 0, 0, err + } + end, err := strconv.ParseUint(parts[1], 10, 16) + if err != nil { + return 0, 0, err + } + if end < start { + return 0, 0, fmt.Errorf("Invalid range specified for the Port: %s", ports) + } + return start, end, nil +} + +// ParseLink parses and validates the specified string as a link format (name:alias) +func ParseLink(val string) (string, string, error) { + if val == "" { + return "", "", fmt.Errorf("empty string specified for links") + } + arr := strings.Split(val, ":") + if len(arr) > 2 { + return "", "", fmt.Errorf("bad format for links: %s", val) + } + if len(arr) == 1 { + return val, val, nil + } + // This is kept because we can actually get an HostConfig with links + // from an already created container and the format is not `foo:bar` + // but `/foo:/c1/bar` + if strings.HasPrefix(arr[0], "/") { + _, alias := path.Split(arr[1]) + return arr[0][1:], alias, nil + } + return arr[0], arr[1], nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers/parsers_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers/parsers_test.go new file mode 100644 index 0000000000000..903c66afb49d9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/parsers/parsers_test.go @@ -0,0 +1,210 @@ +package parsers + +import ( + "strings" + "testing" +) + +func TestParseHost(t *testing.T) { + var ( + defaultHTTPHost = "127.0.0.1" + defaultUnix = "/var/run/docker.sock" + ) + invalids := map[string]string{ + "0.0.0.0": "Invalid bind address format: 0.0.0.0", + "tcp://": "Invalid proto, expected tcp: ", + "tcp:a.b.c.d": "Invalid bind address format: tcp:a.b.c.d", + "tcp:a.b.c.d/path": "Invalid bind address format: tcp:a.b.c.d/path", + "udp://127.0.0.1": "Invalid bind address format: udp://127.0.0.1", + "udp://127.0.0.1:2375": "Invalid bind address format: udp://127.0.0.1:2375", + } + valids := map[string]string{ + "0.0.0.1:5555": "tcp://0.0.0.1:5555", + "0.0.0.1:5555/path": "tcp://0.0.0.1:5555/path", + ":6666": "tcp://127.0.0.1:6666", + ":6666/path": "tcp://127.0.0.1:6666/path", + "tcp://:7777": "tcp://127.0.0.1:7777", + "tcp://:7777/path": "tcp://127.0.0.1:7777/path", + "": "unix:///var/run/docker.sock", + "unix:///run/docker.sock": "unix:///run/docker.sock", + "unix://": "unix:///var/run/docker.sock", + "fd://": "fd://", + "fd://something": "fd://something", + } + for invalidAddr, expectedError := range invalids { + if addr, err := ParseHost(defaultHTTPHost, defaultUnix, invalidAddr); err == nil || err.Error() != expectedError { + t.Errorf("tcp %v address expected error %v return, got %s and addr %v", invalidAddr, expectedError, err, addr) + } + } + for validAddr, expectedAddr := range valids { + if addr, err := ParseHost(defaultHTTPHost, defaultUnix, validAddr); err != nil || addr != expectedAddr { + t.Errorf("%v -> expected %v, got %v", validAddr, expectedAddr, addr) + } + } +} + +func TestParseInvalidUnixAddrInvalid(t *testing.T) { + if _, err := ParseUnixAddr("unix://tcp://127.0.0.1", "unix:///var/run/docker.sock"); err == nil || err.Error() != "Invalid proto, expected unix: tcp://127.0.0.1" { + t.Fatalf("Expected an error, got %v", err) + } +} + +func TestParseRepositoryTag(t *testing.T) { + if repo, tag := ParseRepositoryTag("root"); repo != "root" || tag != "" { + t.Errorf("Expected repo: '%s' and tag: '%s', got '%s' and '%s'", "root", "", repo, tag) + } + if repo, tag := ParseRepositoryTag("root:tag"); repo != "root" || tag != "tag" { + t.Errorf("Expected repo: '%s' and tag: '%s', got '%s' and '%s'", "root", "tag", repo, tag) + } + if repo, digest := ParseRepositoryTag("root@sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855"); repo != "root" || digest != "sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855" { + t.Errorf("Expected repo: '%s' and digest: '%s', got '%s' and '%s'", "root", "sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", repo, digest) + } + if repo, tag := ParseRepositoryTag("user/repo"); repo != "user/repo" || tag != "" { + t.Errorf("Expected repo: '%s' and tag: '%s', got '%s' and '%s'", "user/repo", "", repo, tag) + } + if repo, tag := ParseRepositoryTag("user/repo:tag"); repo != "user/repo" || tag != "tag" { + t.Errorf("Expected repo: '%s' and tag: '%s', got '%s' and '%s'", "user/repo", "tag", repo, tag) + } + if repo, digest := ParseRepositoryTag("user/repo@sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855"); repo != "user/repo" || digest != "sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855" { + t.Errorf("Expected repo: '%s' and digest: '%s', got '%s' and '%s'", "user/repo", "sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", repo, digest) + } + if repo, tag := ParseRepositoryTag("url:5000/repo"); repo != "url:5000/repo" || tag != "" { + t.Errorf("Expected repo: '%s' and tag: '%s', got '%s' and '%s'", "url:5000/repo", "", repo, tag) + } + if repo, tag := ParseRepositoryTag("url:5000/repo:tag"); repo != "url:5000/repo" || tag != "tag" { + t.Errorf("Expected repo: '%s' and tag: '%s', got '%s' and '%s'", "url:5000/repo", "tag", repo, tag) + } + if repo, digest := ParseRepositoryTag("url:5000/repo@sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855"); repo != "url:5000/repo" || digest != "sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855" { + t.Errorf("Expected repo: '%s' and digest: '%s', got '%s' and '%s'", "url:5000/repo", "sha256:e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", repo, digest) + } +} + +func TestParsePortMapping(t *testing.T) { + if _, err := PartParser("ip:public:private", "192.168.1.1:80"); err == nil { + t.Fatalf("Expected an error, got %v", err) + } + data, err := PartParser("ip:public:private", "192.168.1.1:80:8080") + if err != nil { + t.Fatal(err) + } + + if len(data) != 3 { + t.FailNow() + } + if data["ip"] != "192.168.1.1" { + t.Fail() + } + if data["public"] != "80" { + t.Fail() + } + if data["private"] != "8080" { + t.Fail() + } +} + +func TestParseKeyValueOpt(t *testing.T) { + invalids := map[string]string{ + "": "Unable to parse key/value option: ", + "key": "Unable to parse key/value option: key", + } + for invalid, expectedError := range invalids { + if _, _, err := ParseKeyValueOpt(invalid); err == nil || err.Error() != expectedError { + t.Fatalf("Expected error %v for %v, got %v", expectedError, invalid, err) + } + } + valids := map[string][]string{ + "key=value": {"key", "value"}, + " key = value ": {"key", "value"}, + "key=value1=value2": {"key", "value1=value2"}, + " key = value1 = value2 ": {"key", "value1 = value2"}, + } + for valid, expectedKeyValue := range valids { + key, value, err := ParseKeyValueOpt(valid) + if err != nil { + t.Fatal(err) + } + if key != expectedKeyValue[0] || value != expectedKeyValue[1] { + t.Fatalf("Expected {%v: %v} got {%v: %v}", expectedKeyValue[0], expectedKeyValue[1], key, value) + } + } +} + +func TestParsePortRange(t *testing.T) { + if start, end, err := ParsePortRange("8000-8080"); err != nil || start != 8000 || end != 8080 { + t.Fatalf("Error: %s or Expecting {start,end} values {8000,8080} but found {%d,%d}.", err, start, end) + } +} + +func TestParsePortRangeEmpty(t *testing.T) { + if _, _, err := ParsePortRange(""); err == nil || err.Error() != "Empty string specified for ports." { + t.Fatalf("Expected error 'Empty string specified for ports.', got %v", err) + } +} + +func TestParsePortRangeWithNoRange(t *testing.T) { + start, end, err := ParsePortRange("8080") + if err != nil { + t.Fatal(err) + } + if start != 8080 || end != 8080 { + t.Fatalf("Expected start and end to be the same and equal to 8080, but were %v and %v", start, end) + } +} + +func TestParsePortRangeIncorrectRange(t *testing.T) { + if _, _, err := ParsePortRange("9000-8080"); err == nil || !strings.Contains(err.Error(), "Invalid range specified for the Port") { + t.Fatalf("Expecting error 'Invalid range specified for the Port' but received %s.", err) + } +} + +func TestParsePortRangeIncorrectEndRange(t *testing.T) { + if _, _, err := ParsePortRange("8000-a"); err == nil || !strings.Contains(err.Error(), "invalid syntax") { + t.Fatalf("Expecting error 'Invalid range specified for the Port' but received %s.", err) + } + + if _, _, err := ParsePortRange("8000-30a"); err == nil || !strings.Contains(err.Error(), "invalid syntax") { + t.Fatalf("Expecting error 'Invalid range specified for the Port' but received %s.", err) + } +} + +func TestParsePortRangeIncorrectStartRange(t *testing.T) { + if _, _, err := ParsePortRange("a-8000"); err == nil || !strings.Contains(err.Error(), "invalid syntax") { + t.Fatalf("Expecting error 'Invalid range specified for the Port' but received %s.", err) + } + + if _, _, err := ParsePortRange("30a-8000"); err == nil || !strings.Contains(err.Error(), "invalid syntax") { + t.Fatalf("Expecting error 'Invalid range specified for the Port' but received %s.", err) + } +} + +func TestParseLink(t *testing.T) { + name, alias, err := ParseLink("name:alias") + if err != nil { + t.Fatalf("Expected not to error out on a valid name:alias format but got: %v", err) + } + if name != "name" { + t.Fatalf("Link name should have been name, got %s instead", name) + } + if alias != "alias" { + t.Fatalf("Link alias should have been alias, got %s instead", alias) + } + // short format definition + name, alias, err = ParseLink("name") + if err != nil { + t.Fatalf("Expected not to error out on a valid name only format but got: %v", err) + } + if name != "name" { + t.Fatalf("Link name should have been name, got %s instead", name) + } + if alias != "name" { + t.Fatalf("Link alias should have been name, got %s instead", alias) + } + // empty string link definition is not allowed + if _, _, err := ParseLink(""); err == nil || !strings.Contains(err.Error(), "empty string specified for links") { + t.Fatalf("Expected error 'empty string specified for links' but got: %v", err) + } + // more than two colons are not allowed + if _, _, err := ParseLink("link:alias:wrong"); err == nil || !strings.Contains(err.Error(), "bad format for links: link:alias:wrong") { + t.Fatalf("Expected error 'bad format for links: link:alias:wrong' but got: %v", err) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/pools/pools.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/pools/pools.go new file mode 100644 index 0000000000000..515fb4d0508ad --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/pools/pools.go @@ -0,0 +1,119 @@ +// Package pools provides a collection of pools which provide various +// data types with buffers. These can be used to lower the number of +// memory allocations and reuse buffers. +// +// New pools should be added to this package to allow them to be +// shared across packages. +// +// Utility functions which operate on pools should be added to this +// package to allow them to be reused. +package pools + +import ( + "bufio" + "io" + "sync" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ioutils" +) + +var ( + // BufioReader32KPool is a pool which returns bufio.Reader with a 32K buffer. + BufioReader32KPool *BufioReaderPool + // BufioWriter32KPool is a pool which returns bufio.Writer with a 32K buffer. + BufioWriter32KPool *BufioWriterPool +) + +const buffer32K = 32 * 1024 + +// BufioReaderPool is a bufio reader that uses sync.Pool. +type BufioReaderPool struct { + pool sync.Pool +} + +func init() { + BufioReader32KPool = newBufioReaderPoolWithSize(buffer32K) + BufioWriter32KPool = newBufioWriterPoolWithSize(buffer32K) +} + +// newBufioReaderPoolWithSize is unexported because new pools should be +// added here to be shared where required. +func newBufioReaderPoolWithSize(size int) *BufioReaderPool { + pool := sync.Pool{ + New: func() interface{} { return bufio.NewReaderSize(nil, size) }, + } + return &BufioReaderPool{pool: pool} +} + +// Get returns a bufio.Reader which reads from r. The buffer size is that of the pool. +func (bufPool *BufioReaderPool) Get(r io.Reader) *bufio.Reader { + buf := bufPool.pool.Get().(*bufio.Reader) + buf.Reset(r) + return buf +} + +// Put puts the bufio.Reader back into the pool. +func (bufPool *BufioReaderPool) Put(b *bufio.Reader) { + b.Reset(nil) + bufPool.pool.Put(b) +} + +// Copy is a convenience wrapper which uses a buffer to avoid allocation in io.Copy. +func Copy(dst io.Writer, src io.Reader) (written int64, err error) { + buf := BufioReader32KPool.Get(src) + written, err = io.Copy(dst, buf) + BufioReader32KPool.Put(buf) + return +} + +// NewReadCloserWrapper returns a wrapper which puts the bufio.Reader back +// into the pool and closes the reader if it's an io.ReadCloser. +func (bufPool *BufioReaderPool) NewReadCloserWrapper(buf *bufio.Reader, r io.Reader) io.ReadCloser { + return ioutils.NewReadCloserWrapper(r, func() error { + if readCloser, ok := r.(io.ReadCloser); ok { + readCloser.Close() + } + bufPool.Put(buf) + return nil + }) +} + +// BufioWriterPool is a bufio writer that uses sync.Pool. +type BufioWriterPool struct { + pool sync.Pool +} + +// newBufioWriterPoolWithSize is unexported because new pools should be +// added here to be shared where required. +func newBufioWriterPoolWithSize(size int) *BufioWriterPool { + pool := sync.Pool{ + New: func() interface{} { return bufio.NewWriterSize(nil, size) }, + } + return &BufioWriterPool{pool: pool} +} + +// Get returns a bufio.Writer which writes to w. The buffer size is that of the pool. +func (bufPool *BufioWriterPool) Get(w io.Writer) *bufio.Writer { + buf := bufPool.pool.Get().(*bufio.Writer) + buf.Reset(w) + return buf +} + +// Put puts the bufio.Writer back into the pool. +func (bufPool *BufioWriterPool) Put(b *bufio.Writer) { + b.Reset(nil) + bufPool.pool.Put(b) +} + +// NewWriteCloserWrapper returns a wrapper which puts the bufio.Writer back +// into the pool and closes the writer if it's an io.Writecloser. +func (bufPool *BufioWriterPool) NewWriteCloserWrapper(buf *bufio.Writer, w io.Writer) io.WriteCloser { + return ioutils.NewWriteCloserWrapper(w, func() error { + buf.Flush() + if writeCloser, ok := w.(io.WriteCloser); ok { + writeCloser.Close() + } + bufPool.Put(buf) + return nil + }) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/pools/pools_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/pools/pools_test.go new file mode 100644 index 0000000000000..78689800b4076 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/pools/pools_test.go @@ -0,0 +1,162 @@ +package pools + +import ( + "bufio" + "bytes" + "io" + "strings" + "testing" +) + +func TestBufioReaderPoolGetWithNoReaderShouldCreateOne(t *testing.T) { + reader := BufioReader32KPool.Get(nil) + if reader == nil { + t.Fatalf("BufioReaderPool should have create a bufio.Reader but did not.") + } +} + +func TestBufioReaderPoolPutAndGet(t *testing.T) { + sr := bufio.NewReader(strings.NewReader("foobar")) + reader := BufioReader32KPool.Get(sr) + if reader == nil { + t.Fatalf("BufioReaderPool should not return a nil reader.") + } + // verify the first 3 byte + buf1 := make([]byte, 3) + _, err := reader.Read(buf1) + if err != nil { + t.Fatal(err) + } + if actual := string(buf1); actual != "foo" { + t.Fatalf("The first letter should have been 'foo' but was %v", actual) + } + BufioReader32KPool.Put(reader) + // Try to read the next 3 bytes + _, err = sr.Read(make([]byte, 3)) + if err == nil || err != io.EOF { + t.Fatalf("The buffer should have been empty, issue an EOF error.") + } +} + +type simpleReaderCloser struct { + io.Reader + closed bool +} + +func (r *simpleReaderCloser) Close() error { + r.closed = true + return nil +} + +func TestNewReadCloserWrapperWithAReadCloser(t *testing.T) { + br := bufio.NewReader(strings.NewReader("")) + sr := &simpleReaderCloser{ + Reader: strings.NewReader("foobar"), + closed: false, + } + reader := BufioReader32KPool.NewReadCloserWrapper(br, sr) + if reader == nil { + t.Fatalf("NewReadCloserWrapper should not return a nil reader.") + } + // Verify the content of reader + buf := make([]byte, 3) + _, err := reader.Read(buf) + if err != nil { + t.Fatal(err) + } + if actual := string(buf); actual != "foo" { + t.Fatalf("The first 3 letter should have been 'foo' but were %v", actual) + } + reader.Close() + // Read 3 more bytes "bar" + _, err = reader.Read(buf) + if err != nil { + t.Fatal(err) + } + if actual := string(buf); actual != "bar" { + t.Fatalf("The first 3 letter should have been 'bar' but were %v", actual) + } + if !sr.closed { + t.Fatalf("The ReaderCloser should have been closed, it is not.") + } +} + +func TestBufioWriterPoolGetWithNoReaderShouldCreateOne(t *testing.T) { + writer := BufioWriter32KPool.Get(nil) + if writer == nil { + t.Fatalf("BufioWriterPool should have create a bufio.Writer but did not.") + } +} + +func TestBufioWriterPoolPutAndGet(t *testing.T) { + buf := new(bytes.Buffer) + bw := bufio.NewWriter(buf) + writer := BufioWriter32KPool.Get(bw) + if writer == nil { + t.Fatalf("BufioReaderPool should not return a nil writer.") + } + written, err := writer.Write([]byte("foobar")) + if err != nil { + t.Fatal(err) + } + if written != 6 { + t.Fatalf("Should have written 6 bytes, but wrote %v bytes", written) + } + // Make sure we Flush all the way ? + writer.Flush() + bw.Flush() + if len(buf.Bytes()) != 6 { + t.Fatalf("The buffer should contain 6 bytes ('foobar') but contains %v ('%v')", buf.Bytes(), string(buf.Bytes())) + } + // Reset the buffer + buf.Reset() + BufioWriter32KPool.Put(writer) + // Try to write something + written, err = writer.Write([]byte("barfoo")) + if err != nil { + t.Fatal(err) + } + // If we now try to flush it, it should panic (the writer is nil) + // recover it + defer func() { + if r := recover(); r == nil { + t.Fatal("Trying to flush the writter should have 'paniced', did not.") + } + }() + writer.Flush() +} + +type simpleWriterCloser struct { + io.Writer + closed bool +} + +func (r *simpleWriterCloser) Close() error { + r.closed = true + return nil +} + +func TestNewWriteCloserWrapperWithAWriteCloser(t *testing.T) { + buf := new(bytes.Buffer) + bw := bufio.NewWriter(buf) + sw := &simpleWriterCloser{ + Writer: new(bytes.Buffer), + closed: false, + } + bw.Flush() + writer := BufioWriter32KPool.NewWriteCloserWrapper(bw, sw) + if writer == nil { + t.Fatalf("BufioReaderPool should not return a nil writer.") + } + written, err := writer.Write([]byte("foobar")) + if err != nil { + t.Fatal(err) + } + if written != 6 { + t.Fatalf("Should have written 6 bytes, but wrote %v bytes", written) + } + writer.Close() + if !sw.closed { + t.Fatalf("The ReaderCloser should have been closed, it is not.") + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/promise/promise.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/promise/promise.go new file mode 100644 index 0000000000000..dd52b9082f744 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/promise/promise.go @@ -0,0 +1,11 @@ +package promise + +// Go is a basic promise implementation: it wraps calls a function in a goroutine, +// and returns a channel which will later return the function's return value. +func Go(f func() error) chan error { + ch := make(chan error, 1) + go func() { + ch <- f() + }() + return ch +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy/stdcopy.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy/stdcopy.go new file mode 100644 index 0000000000000..63b3df79fce5f --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy/stdcopy.go @@ -0,0 +1,168 @@ +package stdcopy + +import ( + "encoding/binary" + "errors" + "io" + + "github.com/fsouza/go-dockerclient/external/github.com/Sirupsen/logrus" +) + +const ( + StdWriterPrefixLen = 8 + StdWriterFdIndex = 0 + StdWriterSizeIndex = 4 +) + +type StdType [StdWriterPrefixLen]byte + +var ( + Stdin StdType = StdType{0: 0} + Stdout StdType = StdType{0: 1} + Stderr StdType = StdType{0: 2} +) + +type StdWriter struct { + io.Writer + prefix StdType + sizeBuf []byte +} + +func (w *StdWriter) Write(buf []byte) (n int, err error) { + var n1, n2 int + if w == nil || w.Writer == nil { + return 0, errors.New("Writer not instantiated") + } + binary.BigEndian.PutUint32(w.prefix[4:], uint32(len(buf))) + n1, err = w.Writer.Write(w.prefix[:]) + if err != nil { + n = n1 - StdWriterPrefixLen + } else { + n2, err = w.Writer.Write(buf) + n = n1 + n2 - StdWriterPrefixLen + } + if n < 0 { + n = 0 + } + return +} + +// NewStdWriter instantiates a new Writer. +// Everything written to it will be encapsulated using a custom format, +// and written to the underlying `w` stream. +// This allows multiple write streams (e.g. stdout and stderr) to be muxed into a single connection. +// `t` indicates the id of the stream to encapsulate. +// It can be stdcopy.Stdin, stdcopy.Stdout, stdcopy.Stderr. +func NewStdWriter(w io.Writer, t StdType) *StdWriter { + return &StdWriter{ + Writer: w, + prefix: t, + sizeBuf: make([]byte, 4), + } +} + +var ErrInvalidStdHeader = errors.New("Unrecognized input header") + +// StdCopy is a modified version of io.Copy. +// +// StdCopy will demultiplex `src`, assuming that it contains two streams, +// previously multiplexed together using a StdWriter instance. +// As it reads from `src`, StdCopy will write to `dstout` and `dsterr`. +// +// StdCopy will read until it hits EOF on `src`. It will then return a nil error. +// In other words: if `err` is non nil, it indicates a real underlying error. +// +// `written` will hold the total number of bytes written to `dstout` and `dsterr`. +func StdCopy(dstout, dsterr io.Writer, src io.Reader) (written int64, err error) { + var ( + buf = make([]byte, 32*1024+StdWriterPrefixLen+1) + bufLen = len(buf) + nr, nw int + er, ew error + out io.Writer + frameSize int + ) + + for { + // Make sure we have at least a full header + for nr < StdWriterPrefixLen { + var nr2 int + nr2, er = src.Read(buf[nr:]) + nr += nr2 + if er == io.EOF { + if nr < StdWriterPrefixLen { + logrus.Debugf("Corrupted prefix: %v", buf[:nr]) + return written, nil + } + break + } + if er != nil { + logrus.Debugf("Error reading header: %s", er) + return 0, er + } + } + + // Check the first byte to know where to write + switch buf[StdWriterFdIndex] { + case 0: + fallthrough + case 1: + // Write on stdout + out = dstout + case 2: + // Write on stderr + out = dsterr + default: + logrus.Debugf("Error selecting output fd: (%d)", buf[StdWriterFdIndex]) + return 0, ErrInvalidStdHeader + } + + // Retrieve the size of the frame + frameSize = int(binary.BigEndian.Uint32(buf[StdWriterSizeIndex : StdWriterSizeIndex+4])) + logrus.Debugf("framesize: %d", frameSize) + + // Check if the buffer is big enough to read the frame. + // Extend it if necessary. + if frameSize+StdWriterPrefixLen > bufLen { + logrus.Debugf("Extending buffer cap by %d (was %d)", frameSize+StdWriterPrefixLen-bufLen+1, len(buf)) + buf = append(buf, make([]byte, frameSize+StdWriterPrefixLen-bufLen+1)...) + bufLen = len(buf) + } + + // While the amount of bytes read is less than the size of the frame + header, we keep reading + for nr < frameSize+StdWriterPrefixLen { + var nr2 int + nr2, er = src.Read(buf[nr:]) + nr += nr2 + if er == io.EOF { + if nr < frameSize+StdWriterPrefixLen { + logrus.Debugf("Corrupted frame: %v", buf[StdWriterPrefixLen:nr]) + return written, nil + } + break + } + if er != nil { + logrus.Debugf("Error reading frame: %s", er) + return 0, er + } + } + + // Write the retrieved frame (without header) + nw, ew = out.Write(buf[StdWriterPrefixLen : frameSize+StdWriterPrefixLen]) + if ew != nil { + logrus.Debugf("Error writing frame: %s", ew) + return 0, ew + } + // If the frame has not been fully written: error + if nw != frameSize { + logrus.Debugf("Error Short Write: (%d on %d)", nw, frameSize) + return 0, io.ErrShortWrite + } + written += int64(nw) + + // Move the rest of the buffer to the beginning + copy(buf, buf[frameSize+StdWriterPrefixLen:]) + // Move the index + nr -= frameSize + StdWriterPrefixLen + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy/stdcopy_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy/stdcopy_test.go new file mode 100644 index 0000000000000..a9fd73a49eddc --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy/stdcopy_test.go @@ -0,0 +1,85 @@ +package stdcopy + +import ( + "bytes" + "io/ioutil" + "strings" + "testing" +) + +func TestNewStdWriter(t *testing.T) { + writer := NewStdWriter(ioutil.Discard, Stdout) + if writer == nil { + t.Fatalf("NewStdWriter with an invalid StdType should not return nil.") + } +} + +func TestWriteWithUnitializedStdWriter(t *testing.T) { + writer := StdWriter{ + Writer: nil, + prefix: Stdout, + sizeBuf: make([]byte, 4), + } + n, err := writer.Write([]byte("Something here")) + if n != 0 || err == nil { + t.Fatalf("Should fail when given an uncomplete or uninitialized StdWriter") + } +} + +func TestWriteWithNilBytes(t *testing.T) { + writer := NewStdWriter(ioutil.Discard, Stdout) + n, err := writer.Write(nil) + if err != nil { + t.Fatalf("Shouldn't have fail when given no data") + } + if n > 0 { + t.Fatalf("Write should have written 0 byte, but has written %d", n) + } +} + +func TestWrite(t *testing.T) { + writer := NewStdWriter(ioutil.Discard, Stdout) + data := []byte("Test StdWrite.Write") + n, err := writer.Write(data) + if err != nil { + t.Fatalf("Error while writing with StdWrite") + } + if n != len(data) { + t.Fatalf("Write should have writen %d byte but wrote %d.", len(data), n) + } +} + +func TestStdCopyWithInvalidInputHeader(t *testing.T) { + dstOut := NewStdWriter(ioutil.Discard, Stdout) + dstErr := NewStdWriter(ioutil.Discard, Stderr) + src := strings.NewReader("Invalid input") + _, err := StdCopy(dstOut, dstErr, src) + if err == nil { + t.Fatal("StdCopy with invalid input header should fail.") + } +} + +func TestStdCopyWithCorruptedPrefix(t *testing.T) { + data := []byte{0x01, 0x02, 0x03} + src := bytes.NewReader(data) + written, err := StdCopy(nil, nil, src) + if err != nil { + t.Fatalf("StdCopy should not return an error with corrupted prefix.") + } + if written != 0 { + t.Fatalf("StdCopy should have written 0, but has written %d", written) + } +} + +func BenchmarkWrite(b *testing.B) { + w := NewStdWriter(ioutil.Discard, Stdout) + data := []byte("Test line for testing stdwriter performance\n") + data = bytes.Repeat(data, 100) + b.SetBytes(int64(len(data))) + b.ResetTimer() + for i := 0; i < b.N; i++ { + if _, err := w.Write(data); err != nil { + b.Fatal(err) + } + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/errors.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/errors.go new file mode 100644 index 0000000000000..63045186fe8be --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/errors.go @@ -0,0 +1,9 @@ +package system + +import ( + "errors" +) + +var ( + ErrNotSupportedPlatform = errors.New("platform and architecture is not supported") +) diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/events_windows.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/events_windows.go new file mode 100644 index 0000000000000..23f7c618bc6c6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/events_windows.go @@ -0,0 +1,83 @@ +package system + +// This file implements syscalls for Win32 events which are not implemented +// in golang. + +import ( + "syscall" + "unsafe" +) + +const ( + EVENT_ALL_ACCESS = 0x1F0003 + EVENT_MODIFY_STATUS = 0x0002 +) + +var ( + procCreateEvent = modkernel32.NewProc("CreateEventW") + procOpenEvent = modkernel32.NewProc("OpenEventW") + procSetEvent = modkernel32.NewProc("SetEvent") + procResetEvent = modkernel32.NewProc("ResetEvent") + procPulseEvent = modkernel32.NewProc("PulseEvent") +) + +func CreateEvent(eventAttributes *syscall.SecurityAttributes, manualReset bool, initialState bool, name string) (handle syscall.Handle, err error) { + namep, _ := syscall.UTF16PtrFromString(name) + var _p1 uint32 = 0 + if manualReset { + _p1 = 1 + } + var _p2 uint32 = 0 + if initialState { + _p2 = 1 + } + r0, _, e1 := procCreateEvent.Call(uintptr(unsafe.Pointer(eventAttributes)), uintptr(_p1), uintptr(_p2), uintptr(unsafe.Pointer(namep))) + use(unsafe.Pointer(namep)) + handle = syscall.Handle(r0) + if handle == syscall.InvalidHandle { + err = e1 + } + return +} + +func OpenEvent(desiredAccess uint32, inheritHandle bool, name string) (handle syscall.Handle, err error) { + namep, _ := syscall.UTF16PtrFromString(name) + var _p1 uint32 = 0 + if inheritHandle { + _p1 = 1 + } + r0, _, e1 := procOpenEvent.Call(uintptr(desiredAccess), uintptr(_p1), uintptr(unsafe.Pointer(namep))) + use(unsafe.Pointer(namep)) + handle = syscall.Handle(r0) + if handle == syscall.InvalidHandle { + err = e1 + } + return +} + +func SetEvent(handle syscall.Handle) (err error) { + return setResetPulse(handle, procSetEvent) +} + +func ResetEvent(handle syscall.Handle) (err error) { + return setResetPulse(handle, procResetEvent) +} + +func PulseEvent(handle syscall.Handle) (err error) { + return setResetPulse(handle, procPulseEvent) +} + +func setResetPulse(handle syscall.Handle, proc *syscall.LazyProc) (err error) { + r0, _, _ := proc.Call(uintptr(handle)) + if r0 != 0 { + err = syscall.Errno(r0) + } + return +} + +var temp unsafe.Pointer + +// use ensures a variable is kept alive without the GC freeing while still needed +func use(p unsafe.Pointer) { + temp = p +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/filesys.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/filesys.go new file mode 100644 index 0000000000000..e1f70e8dac1ad --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/filesys.go @@ -0,0 +1,11 @@ +// +build !windows + +package system + +import ( + "os" +) + +func MkdirAll(path string, perm os.FileMode) error { + return os.MkdirAll(path, perm) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/filesys_windows.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/filesys_windows.go new file mode 100644 index 0000000000000..90b500608efaa --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/filesys_windows.go @@ -0,0 +1,64 @@ +// +build windows + +package system + +import ( + "os" + "regexp" + "syscall" +) + +// MkdirAll implementation that is volume path aware for Windows. +func MkdirAll(path string, perm os.FileMode) error { + if re := regexp.MustCompile(`^\\\\\?\\Volume{[a-z0-9-]+}$`); re.MatchString(path) { + return nil + } + + // The rest of this method is copied from os.MkdirAll and should be kept + // as-is to ensure compatibility. + + // Fast path: if we can tell whether path is a directory or file, stop with success or error. + dir, err := os.Stat(path) + if err == nil { + if dir.IsDir() { + return nil + } + return &os.PathError{ + Op: "mkdir", + Path: path, + Err: syscall.ENOTDIR, + } + } + + // Slow path: make sure parent exists and then call Mkdir for path. + i := len(path) + for i > 0 && os.IsPathSeparator(path[i-1]) { // Skip trailing path separator. + i-- + } + + j := i + for j > 0 && !os.IsPathSeparator(path[j-1]) { // Scan backward over element. + j-- + } + + if j > 1 { + // Create parent + err = MkdirAll(path[0:j-1], perm) + if err != nil { + return err + } + } + + // Parent now exists; invoke Mkdir and use its result. + err = os.Mkdir(path, perm) + if err != nil { + // Handle arguments like "foo/." by + // double-checking that directory doesn't exist. + dir, err1 := os.Lstat(path) + if err1 == nil && dir.IsDir() { + return nil + } + return err + } + return nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/lstat.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/lstat.go new file mode 100644 index 0000000000000..d0e43b3709784 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/lstat.go @@ -0,0 +1,19 @@ +// +build !windows + +package system + +import ( + "syscall" +) + +// Lstat takes a path to a file and returns +// a system.Stat_t type pertaining to that file. +// +// Throws an error if the file does not exist +func Lstat(path string) (*Stat_t, error) { + s := &syscall.Stat_t{} + if err := syscall.Lstat(path, s); err != nil { + return nil, err + } + return fromStatT(s) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/lstat_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/lstat_test.go new file mode 100644 index 0000000000000..6bac492eb1a76 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/lstat_test.go @@ -0,0 +1,28 @@ +package system + +import ( + "os" + "testing" +) + +// TestLstat tests Lstat for existing and non existing files +func TestLstat(t *testing.T) { + file, invalid, _, dir := prepareFiles(t) + defer os.RemoveAll(dir) + + statFile, err := Lstat(file) + if err != nil { + t.Fatal(err) + } + if statFile == nil { + t.Fatal("returned empty stat for existing file") + } + + statInvalid, err := Lstat(invalid) + if err == nil { + t.Fatal("did not return error for non-existing file") + } + if statInvalid != nil { + t.Fatal("returned non-nil stat for non-existing file") + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/lstat_windows.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/lstat_windows.go new file mode 100644 index 0000000000000..eee1be26eb5ac --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/lstat_windows.go @@ -0,0 +1,29 @@ +// +build windows + +package system + +import ( + "os" +) + +// Some explanation for my own sanity, and hopefully maintainers in the +// future. +// +// Lstat calls os.Lstat to get a fileinfo interface back. +// This is then copied into our own locally defined structure. +// Note the Linux version uses fromStatT to do the copy back, +// but that not strictly necessary when already in an OS specific module. + +func Lstat(path string) (*Stat_t, error) { + fi, err := os.Lstat(path) + if err != nil { + return nil, err + } + + return &Stat_t{ + name: fi.Name(), + size: fi.Size(), + mode: fi.Mode(), + modTime: fi.ModTime(), + isDir: fi.IsDir()}, nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo.go new file mode 100644 index 0000000000000..3b6e947e6753d --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo.go @@ -0,0 +1,17 @@ +package system + +// MemInfo contains memory statistics of the host system. +type MemInfo struct { + // Total usable RAM (i.e. physical RAM minus a few reserved bits and the + // kernel binary code). + MemTotal int64 + + // Amount of free memory. + MemFree int64 + + // Total amount of swap space available. + SwapTotal int64 + + // Amount of swap space that is currently unused. + SwapFree int64 +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo_linux.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo_linux.go new file mode 100644 index 0000000000000..41f2bab60370d --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo_linux.go @@ -0,0 +1,71 @@ +package system + +import ( + "bufio" + "errors" + "io" + "os" + "strconv" + "strings" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/units" +) + +var ( + ErrMalformed = errors.New("malformed file") +) + +// ReadMemInfo retrieves memory statistics of the host system and returns a +// MemInfo type. +func ReadMemInfo() (*MemInfo, error) { + file, err := os.Open("/proc/meminfo") + if err != nil { + return nil, err + } + defer file.Close() + return parseMemInfo(file) +} + +// parseMemInfo parses the /proc/meminfo file into +// a MemInfo object given a io.Reader to the file. +// +// Throws error if there are problems reading from the file +func parseMemInfo(reader io.Reader) (*MemInfo, error) { + meminfo := &MemInfo{} + scanner := bufio.NewScanner(reader) + for scanner.Scan() { + // Expected format: ["MemTotal:", "1234", "kB"] + parts := strings.Fields(scanner.Text()) + + // Sanity checks: Skip malformed entries. + if len(parts) < 3 || parts[2] != "kB" { + continue + } + + // Convert to bytes. + size, err := strconv.Atoi(parts[1]) + if err != nil { + continue + } + bytes := int64(size) * units.KiB + + switch parts[0] { + case "MemTotal:": + meminfo.MemTotal = bytes + case "MemFree:": + meminfo.MemFree = bytes + case "SwapTotal:": + meminfo.SwapTotal = bytes + case "SwapFree:": + meminfo.SwapFree = bytes + } + + } + + // Handle errors that may have occurred during the reading of the file. + if err := scanner.Err(); err != nil { + return nil, err + } + + return meminfo, nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo_linux_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo_linux_test.go new file mode 100644 index 0000000000000..87830ccb2e236 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo_linux_test.go @@ -0,0 +1,38 @@ +package system + +import ( + "strings" + "testing" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/units" +) + +// TestMemInfo tests parseMemInfo with a static meminfo string +func TestMemInfo(t *testing.T) { + const input = ` + MemTotal: 1 kB + MemFree: 2 kB + SwapTotal: 3 kB + SwapFree: 4 kB + Malformed1: + Malformed2: 1 + Malformed3: 2 MB + Malformed4: X kB + ` + meminfo, err := parseMemInfo(strings.NewReader(input)) + if err != nil { + t.Fatal(err) + } + if meminfo.MemTotal != 1*units.KiB { + t.Fatalf("Unexpected MemTotal: %d", meminfo.MemTotal) + } + if meminfo.MemFree != 2*units.KiB { + t.Fatalf("Unexpected MemFree: %d", meminfo.MemFree) + } + if meminfo.SwapTotal != 3*units.KiB { + t.Fatalf("Unexpected SwapTotal: %d", meminfo.SwapTotal) + } + if meminfo.SwapFree != 4*units.KiB { + t.Fatalf("Unexpected SwapFree: %d", meminfo.SwapFree) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo_unsupported.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo_unsupported.go new file mode 100644 index 0000000000000..604d338754c5f --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo_unsupported.go @@ -0,0 +1,7 @@ +// +build !linux,!windows + +package system + +func ReadMemInfo() (*MemInfo, error) { + return nil, ErrNotSupportedPlatform +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo_windows.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo_windows.go new file mode 100644 index 0000000000000..d46642598cf6e --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/meminfo_windows.go @@ -0,0 +1,44 @@ +package system + +import ( + "syscall" + "unsafe" +) + +var ( + modkernel32 = syscall.NewLazyDLL("kernel32.dll") + + procGlobalMemoryStatusEx = modkernel32.NewProc("GlobalMemoryStatusEx") +) + +// https://msdn.microsoft.com/en-us/library/windows/desktop/aa366589(v=vs.85).aspx +// https://msdn.microsoft.com/en-us/library/windows/desktop/aa366770(v=vs.85).aspx +type memorystatusex struct { + dwLength uint32 + dwMemoryLoad uint32 + ullTotalPhys uint64 + ullAvailPhys uint64 + ullTotalPageFile uint64 + ullAvailPageFile uint64 + ullTotalVirtual uint64 + ullAvailVirtual uint64 + ullAvailExtendedVirtual uint64 +} + +// ReadMemInfo retrieves memory statistics of the host system and returns a +// MemInfo type. +func ReadMemInfo() (*MemInfo, error) { + msi := &memorystatusex{ + dwLength: 64, + } + r1, _, _ := procGlobalMemoryStatusEx.Call(uintptr(unsafe.Pointer(msi))) + if r1 == 0 { + return &MemInfo{}, nil + } + return &MemInfo{ + MemTotal: int64(msi.ullTotalPhys), + MemFree: int64(msi.ullAvailPhys), + SwapTotal: int64(msi.ullTotalPageFile), + SwapFree: int64(msi.ullAvailPageFile), + }, nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/mknod.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/mknod.go new file mode 100644 index 0000000000000..26617eb08f406 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/mknod.go @@ -0,0 +1,20 @@ +// +build !windows + +package system + +import ( + "syscall" +) + +// Mknod creates a filesystem node (file, device special file or named pipe) named path +// with attributes specified by mode and dev +func Mknod(path string, mode uint32, dev int) error { + return syscall.Mknod(path, mode, dev) +} + +// Linux device nodes are a bit weird due to backwards compat with 16 bit device nodes. +// They are, from low to high: the lower 8 bits of the minor, then 12 bits of the major, +// then the top 12 bits of the minor +func Mkdev(major int64, minor int64) uint32 { + return uint32(((minor & 0xfff00) << 12) | ((major & 0xfff) << 8) | (minor & 0xff)) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/mknod_windows.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/mknod_windows.go new file mode 100644 index 0000000000000..1811542ab3f3e --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/mknod_windows.go @@ -0,0 +1,11 @@ +// +build windows + +package system + +func Mknod(path string, mode uint32, dev int) error { + return ErrNotSupportedPlatform +} + +func Mkdev(major int64, minor int64) uint32 { + panic("Mkdev not implemented on Windows.") +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat.go new file mode 100644 index 0000000000000..e2ecfe52febe8 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat.go @@ -0,0 +1,46 @@ +// +build !windows + +package system + +import ( + "syscall" +) + +// Stat_t type contains status of a file. It contains metadata +// like permission, owner, group, size, etc about a file +type Stat_t struct { + mode uint32 + uid uint32 + gid uint32 + rdev uint64 + size int64 + mtim syscall.Timespec +} + +func (s Stat_t) Mode() uint32 { + return s.mode +} + +func (s Stat_t) Uid() uint32 { + return s.uid +} + +func (s Stat_t) Gid() uint32 { + return s.gid +} + +func (s Stat_t) Rdev() uint64 { + return s.rdev +} + +func (s Stat_t) Size() int64 { + return s.size +} + +func (s Stat_t) Mtim() syscall.Timespec { + return s.mtim +} + +func (s Stat_t) GetLastModification() syscall.Timespec { + return s.Mtim() +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_freebsd.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_freebsd.go new file mode 100644 index 0000000000000..4b2198b3aab42 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_freebsd.go @@ -0,0 +1,27 @@ +package system + +import ( + "syscall" +) + +// fromStatT converts a syscall.Stat_t type to a system.Stat_t type +func fromStatT(s *syscall.Stat_t) (*Stat_t, error) { + return &Stat_t{size: s.Size, + mode: uint32(s.Mode), + uid: s.Uid, + gid: s.Gid, + rdev: uint64(s.Rdev), + mtim: s.Mtimespec}, nil +} + +// Stat takes a path to a file and returns +// a system.Stat_t type pertaining to that file. +// +// Throws an error if the file does not exist +func Stat(path string) (*Stat_t, error) { + s := &syscall.Stat_t{} + if err := syscall.Stat(path, s); err != nil { + return nil, err + } + return fromStatT(s) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_linux.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_linux.go new file mode 100644 index 0000000000000..80262d9519294 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_linux.go @@ -0,0 +1,33 @@ +package system + +import ( + "syscall" +) + +// fromStatT converts a syscall.Stat_t type to a system.Stat_t type +func fromStatT(s *syscall.Stat_t) (*Stat_t, error) { + return &Stat_t{size: s.Size, + mode: s.Mode, + uid: s.Uid, + gid: s.Gid, + rdev: s.Rdev, + mtim: s.Mtim}, nil +} + +// FromStatT exists only on linux, and loads a system.Stat_t from a +// syscal.Stat_t. +func FromStatT(s *syscall.Stat_t) (*Stat_t, error) { + return fromStatT(s) +} + +// Stat takes a path to a file and returns +// a system.Stat_t type pertaining to that file. +// +// Throws an error if the file does not exist +func Stat(path string) (*Stat_t, error) { + s := &syscall.Stat_t{} + if err := syscall.Stat(path, s); err != nil { + return nil, err + } + return fromStatT(s) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_test.go new file mode 100644 index 0000000000000..4534129200ef7 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_test.go @@ -0,0 +1,37 @@ +package system + +import ( + "os" + "syscall" + "testing" +) + +// TestFromStatT tests fromStatT for a tempfile +func TestFromStatT(t *testing.T) { + file, _, _, dir := prepareFiles(t) + defer os.RemoveAll(dir) + + stat := &syscall.Stat_t{} + err := syscall.Lstat(file, stat) + + s, err := fromStatT(stat) + if err != nil { + t.Fatal(err) + } + + if stat.Mode != s.Mode() { + t.Fatal("got invalid mode") + } + if stat.Uid != s.Uid() { + t.Fatal("got invalid uid") + } + if stat.Gid != s.Gid() { + t.Fatal("got invalid gid") + } + if stat.Rdev != s.Rdev() { + t.Fatal("got invalid rdev") + } + if stat.Mtim != s.Mtim() { + t.Fatal("got invalid mtim") + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_unsupported.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_unsupported.go new file mode 100644 index 0000000000000..5251ae2129fc5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_unsupported.go @@ -0,0 +1,17 @@ +// +build !linux,!windows,!freebsd + +package system + +import ( + "syscall" +) + +// fromStatT creates a system.Stat_t type from a syscall.Stat_t type +func fromStatT(s *syscall.Stat_t) (*Stat_t, error) { + return &Stat_t{size: s.Size, + mode: uint32(s.Mode), + uid: s.Uid, + gid: s.Gid, + rdev: uint64(s.Rdev), + mtim: s.Mtimespec}, nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_windows.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_windows.go new file mode 100644 index 0000000000000..b1fd39e83f4e0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/stat_windows.go @@ -0,0 +1,36 @@ +// +build windows + +package system + +import ( + "os" + "time" +) + +type Stat_t struct { + name string + size int64 + mode os.FileMode + modTime time.Time + isDir bool +} + +func (s Stat_t) Name() string { + return s.name +} + +func (s Stat_t) Size() int64 { + return s.size +} + +func (s Stat_t) Mode() os.FileMode { + return s.mode +} + +func (s Stat_t) ModTime() time.Time { + return s.modTime +} + +func (s Stat_t) IsDir() bool { + return s.isDir +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/umask.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/umask.go new file mode 100644 index 0000000000000..fddbecd390335 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/umask.go @@ -0,0 +1,11 @@ +// +build !windows + +package system + +import ( + "syscall" +) + +func Umask(newmask int) (oldmask int, err error) { + return syscall.Umask(newmask), nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/umask_windows.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/umask_windows.go new file mode 100644 index 0000000000000..3be563f89e663 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/umask_windows.go @@ -0,0 +1,8 @@ +// +build windows + +package system + +func Umask(newmask int) (oldmask int, err error) { + // should not be called on cli code path + return 0, ErrNotSupportedPlatform +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_darwin.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_darwin.go new file mode 100644 index 0000000000000..4c6002fe8e26a --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_darwin.go @@ -0,0 +1,11 @@ +package system + +import "syscall" + +func LUtimesNano(path string, ts []syscall.Timespec) error { + return ErrNotSupportedPlatform +} + +func UtimesNano(path string, ts []syscall.Timespec) error { + return syscall.UtimesNano(path, ts) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_freebsd.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_freebsd.go new file mode 100644 index 0000000000000..ceaa044c1c56b --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_freebsd.go @@ -0,0 +1,24 @@ +package system + +import ( + "syscall" + "unsafe" +) + +func LUtimesNano(path string, ts []syscall.Timespec) error { + var _path *byte + _path, err := syscall.BytePtrFromString(path) + if err != nil { + return err + } + + if _, _, err := syscall.Syscall(syscall.SYS_LUTIMES, uintptr(unsafe.Pointer(_path)), uintptr(unsafe.Pointer(&ts[0])), 0); err != 0 && err != syscall.ENOSYS { + return err + } + + return nil +} + +func UtimesNano(path string, ts []syscall.Timespec) error { + return syscall.UtimesNano(path, ts) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_linux.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_linux.go new file mode 100644 index 0000000000000..8f90298271e9d --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_linux.go @@ -0,0 +1,28 @@ +package system + +import ( + "syscall" + "unsafe" +) + +func LUtimesNano(path string, ts []syscall.Timespec) error { + // These are not currently available in syscall + AT_FDCWD := -100 + AT_SYMLINK_NOFOLLOW := 0x100 + + var _path *byte + _path, err := syscall.BytePtrFromString(path) + if err != nil { + return err + } + + if _, _, err := syscall.Syscall6(syscall.SYS_UTIMENSAT, uintptr(AT_FDCWD), uintptr(unsafe.Pointer(_path)), uintptr(unsafe.Pointer(&ts[0])), uintptr(AT_SYMLINK_NOFOLLOW), 0, 0); err != 0 && err != syscall.ENOSYS { + return err + } + + return nil +} + +func UtimesNano(path string, ts []syscall.Timespec) error { + return syscall.UtimesNano(path, ts) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_test.go new file mode 100644 index 0000000000000..350cce1ead47e --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_test.go @@ -0,0 +1,66 @@ +package system + +import ( + "io/ioutil" + "os" + "path/filepath" + "syscall" + "testing" +) + +// prepareFiles creates files for testing in the temp directory +func prepareFiles(t *testing.T) (string, string, string, string) { + dir, err := ioutil.TempDir("", "docker-system-test") + if err != nil { + t.Fatal(err) + } + + file := filepath.Join(dir, "exist") + if err := ioutil.WriteFile(file, []byte("hello"), 0644); err != nil { + t.Fatal(err) + } + + invalid := filepath.Join(dir, "doesnt-exist") + + symlink := filepath.Join(dir, "symlink") + if err := os.Symlink(file, symlink); err != nil { + t.Fatal(err) + } + + return file, invalid, symlink, dir +} + +func TestLUtimesNano(t *testing.T) { + file, invalid, symlink, dir := prepareFiles(t) + defer os.RemoveAll(dir) + + before, err := os.Stat(file) + if err != nil { + t.Fatal(err) + } + + ts := []syscall.Timespec{{0, 0}, {0, 0}} + if err := LUtimesNano(symlink, ts); err != nil { + t.Fatal(err) + } + + symlinkInfo, err := os.Lstat(symlink) + if err != nil { + t.Fatal(err) + } + if before.ModTime().Unix() == symlinkInfo.ModTime().Unix() { + t.Fatal("The modification time of the symlink should be different") + } + + fileInfo, err := os.Stat(file) + if err != nil { + t.Fatal(err) + } + if before.ModTime().Unix() != fileInfo.ModTime().Unix() { + t.Fatal("The modification time of the file should be same") + } + + if err := LUtimesNano(invalid, ts); err == nil { + t.Fatal("Doesn't return an error on a non-existing file") + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_unsupported.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_unsupported.go new file mode 100644 index 0000000000000..adf2734f2770c --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/utimes_unsupported.go @@ -0,0 +1,13 @@ +// +build !linux,!freebsd,!darwin + +package system + +import "syscall" + +func LUtimesNano(path string, ts []syscall.Timespec) error { + return ErrNotSupportedPlatform +} + +func UtimesNano(path string, ts []syscall.Timespec) error { + return ErrNotSupportedPlatform +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/xattrs_linux.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/xattrs_linux.go new file mode 100644 index 0000000000000..00edb201b58ff --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/xattrs_linux.go @@ -0,0 +1,59 @@ +package system + +import ( + "syscall" + "unsafe" +) + +// Returns a nil slice and nil error if the xattr is not set +func Lgetxattr(path string, attr string) ([]byte, error) { + pathBytes, err := syscall.BytePtrFromString(path) + if err != nil { + return nil, err + } + attrBytes, err := syscall.BytePtrFromString(attr) + if err != nil { + return nil, err + } + + dest := make([]byte, 128) + destBytes := unsafe.Pointer(&dest[0]) + sz, _, errno := syscall.Syscall6(syscall.SYS_LGETXATTR, uintptr(unsafe.Pointer(pathBytes)), uintptr(unsafe.Pointer(attrBytes)), uintptr(destBytes), uintptr(len(dest)), 0, 0) + if errno == syscall.ENODATA { + return nil, nil + } + if errno == syscall.ERANGE { + dest = make([]byte, sz) + destBytes := unsafe.Pointer(&dest[0]) + sz, _, errno = syscall.Syscall6(syscall.SYS_LGETXATTR, uintptr(unsafe.Pointer(pathBytes)), uintptr(unsafe.Pointer(attrBytes)), uintptr(destBytes), uintptr(len(dest)), 0, 0) + } + if errno != 0 { + return nil, errno + } + + return dest[:sz], nil +} + +var _zero uintptr + +func Lsetxattr(path string, attr string, data []byte, flags int) error { + pathBytes, err := syscall.BytePtrFromString(path) + if err != nil { + return err + } + attrBytes, err := syscall.BytePtrFromString(attr) + if err != nil { + return err + } + var dataBytes unsafe.Pointer + if len(data) > 0 { + dataBytes = unsafe.Pointer(&data[0]) + } else { + dataBytes = unsafe.Pointer(&_zero) + } + _, _, errno := syscall.Syscall6(syscall.SYS_LSETXATTR, uintptr(unsafe.Pointer(pathBytes)), uintptr(unsafe.Pointer(attrBytes)), uintptr(dataBytes), uintptr(len(data)), uintptr(flags), 0) + if errno != 0 { + return errno + } + return nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/xattrs_unsupported.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/xattrs_unsupported.go new file mode 100644 index 0000000000000..0060c167dc257 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/system/xattrs_unsupported.go @@ -0,0 +1,11 @@ +// +build !linux + +package system + +func Lgetxattr(path string, attr string) ([]byte, error) { + return nil, ErrNotSupportedPlatform +} + +func Lsetxattr(path string, attr string, data []byte, flags int) error { + return ErrNotSupportedPlatform +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit/ulimit.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit/ulimit.go new file mode 100644 index 0000000000000..8fb0d804de138 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit/ulimit.go @@ -0,0 +1,111 @@ +// Package ulimit provides structure and helper function to parse and represent +// resource limits (Rlimit and Ulimit, its human friendly version). +package ulimit + +import ( + "fmt" + "strconv" + "strings" +) + +// Ulimit is a human friendly version of Rlimit. +type Ulimit struct { + Name string + Hard int64 + Soft int64 +} + +// Rlimit specifies the resource limits, such as max open files. +type Rlimit struct { + Type int `json:"type,omitempty"` + Hard uint64 `json:"hard,omitempty"` + Soft uint64 `json:"soft,omitempty"` +} + +const ( + // magic numbers for making the syscall + // some of these are defined in the syscall package, but not all. + // Also since Windows client doesn't get access to the syscall package, need to + // define these here + rlimitAs = 9 + rlimitCore = 4 + rlimitCPU = 0 + rlimitData = 2 + rlimitFsize = 1 + rlimitLocks = 10 + rlimitMemlock = 8 + rlimitMsgqueue = 12 + rlimitNice = 13 + rlimitNofile = 7 + rlimitNproc = 6 + rlimitRss = 5 + rlimitRtprio = 14 + rlimitRttime = 15 + rlimitSigpending = 11 + rlimitStack = 3 +) + +var ulimitNameMapping = map[string]int{ + //"as": rlimitAs, // Disabled since this doesn't seem usable with the way Docker inits a container. + "core": rlimitCore, + "cpu": rlimitCPU, + "data": rlimitData, + "fsize": rlimitFsize, + "locks": rlimitLocks, + "memlock": rlimitMemlock, + "msgqueue": rlimitMsgqueue, + "nice": rlimitNice, + "nofile": rlimitNofile, + "nproc": rlimitNproc, + "rss": rlimitRss, + "rtprio": rlimitRtprio, + "rttime": rlimitRttime, + "sigpending": rlimitSigpending, + "stack": rlimitStack, +} + +// Parse parses and returns a Ulimit from the specified string. +func Parse(val string) (*Ulimit, error) { + parts := strings.SplitN(val, "=", 2) + if len(parts) != 2 { + return nil, fmt.Errorf("invalid ulimit argument: %s", val) + } + + if _, exists := ulimitNameMapping[parts[0]]; !exists { + return nil, fmt.Errorf("invalid ulimit type: %s", parts[0]) + } + + limitVals := strings.SplitN(parts[1], ":", 2) + if len(limitVals) > 2 { + return nil, fmt.Errorf("too many limit value arguments - %s, can only have up to two, `soft[:hard]`", parts[1]) + } + + soft, err := strconv.ParseInt(limitVals[0], 10, 64) + if err != nil { + return nil, err + } + + hard := soft // in case no hard was set + if len(limitVals) == 2 { + hard, err = strconv.ParseInt(limitVals[1], 10, 64) + } + if soft > hard { + return nil, fmt.Errorf("ulimit soft limit must be less than or equal to hard limit: %d > %d", soft, hard) + } + + return &Ulimit{Name: parts[0], Soft: soft, Hard: hard}, nil +} + +// GetRlimit returns the RLimit corresponding to Ulimit. +func (u *Ulimit) GetRlimit() (*Rlimit, error) { + t, exists := ulimitNameMapping[u.Name] + if !exists { + return nil, fmt.Errorf("invalid ulimit name %s", u.Name) + } + + return &Rlimit{Type: t, Soft: uint64(u.Soft), Hard: uint64(u.Hard)}, nil +} + +func (u *Ulimit) String() string { + return fmt.Sprintf("%s=%d:%d", u.Name, u.Soft, u.Hard) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit/ulimit_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit/ulimit_test.go new file mode 100644 index 0000000000000..1e8c881f51d4b --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/ulimit/ulimit_test.go @@ -0,0 +1,55 @@ +package ulimit + +import "testing" + +func TestParseValid(t *testing.T) { + u1 := &Ulimit{"nofile", 1024, 512} + if u2, _ := Parse("nofile=512:1024"); *u1 != *u2 { + t.Fatalf("expected %q, but got %q", u1, u2) + } +} + +func TestParseInvalidLimitType(t *testing.T) { + if _, err := Parse("notarealtype=1024:1024"); err == nil { + t.Fatalf("expected error on invalid ulimit type") + } +} + +func TestParseBadFormat(t *testing.T) { + if _, err := Parse("nofile:1024:1024"); err == nil { + t.Fatal("expected error on bad syntax") + } + + if _, err := Parse("nofile"); err == nil { + t.Fatal("expected error on bad syntax") + } + + if _, err := Parse("nofile="); err == nil { + t.Fatal("expected error on bad syntax") + } + if _, err := Parse("nofile=:"); err == nil { + t.Fatal("expected error on bad syntax") + } + if _, err := Parse("nofile=:1024"); err == nil { + t.Fatal("expected error on bad syntax") + } +} + +func TestParseHardLessThanSoft(t *testing.T) { + if _, err := Parse("nofile:1024:1"); err == nil { + t.Fatal("expected error on hard limit less than soft limit") + } +} + +func TestParseInvalidValueType(t *testing.T) { + if _, err := Parse("nofile:asdf"); err == nil { + t.Fatal("expected error on bad value type") + } +} + +func TestStringOutput(t *testing.T) { + u := &Ulimit{"nofile", 1024, 512} + if s := u.String(); s != "nofile=512:1024" { + t.Fatal("expected String to return nofile=512:1024, but got", s) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/units/duration.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/units/duration.go new file mode 100644 index 0000000000000..c219a8a968c40 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/units/duration.go @@ -0,0 +1,33 @@ +// Package units provides helper function to parse and print size and time units +// in human-readable format. +package units + +import ( + "fmt" + "time" +) + +// HumanDuration returns a human-readable approximation of a duration +// (eg. "About a minute", "4 hours ago", etc.). +func HumanDuration(d time.Duration) string { + if seconds := int(d.Seconds()); seconds < 1 { + return "Less than a second" + } else if seconds < 60 { + return fmt.Sprintf("%d seconds", seconds) + } else if minutes := int(d.Minutes()); minutes == 1 { + return "About a minute" + } else if minutes < 60 { + return fmt.Sprintf("%d minutes", minutes) + } else if hours := int(d.Hours()); hours == 1 { + return "About an hour" + } else if hours < 48 { + return fmt.Sprintf("%d hours", hours) + } else if hours < 24*7*2 { + return fmt.Sprintf("%d days", hours/24) + } else if hours < 24*30*3 { + return fmt.Sprintf("%d weeks", hours/24/7) + } else if hours < 24*365*2 { + return fmt.Sprintf("%d months", hours/24/30) + } + return fmt.Sprintf("%d years", int(d.Hours())/24/365) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/units/duration_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/units/duration_test.go new file mode 100644 index 0000000000000..fcfb6b7bbd7a0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/units/duration_test.go @@ -0,0 +1,46 @@ +package units + +import ( + "testing" + "time" +) + +func TestHumanDuration(t *testing.T) { + // Useful duration abstractions + day := 24 * time.Hour + week := 7 * day + month := 30 * day + year := 365 * day + + assertEquals(t, "Less than a second", HumanDuration(450*time.Millisecond)) + assertEquals(t, "47 seconds", HumanDuration(47*time.Second)) + assertEquals(t, "About a minute", HumanDuration(1*time.Minute)) + assertEquals(t, "3 minutes", HumanDuration(3*time.Minute)) + assertEquals(t, "35 minutes", HumanDuration(35*time.Minute)) + assertEquals(t, "35 minutes", HumanDuration(35*time.Minute+40*time.Second)) + assertEquals(t, "About an hour", HumanDuration(1*time.Hour)) + assertEquals(t, "About an hour", HumanDuration(1*time.Hour+45*time.Minute)) + assertEquals(t, "3 hours", HumanDuration(3*time.Hour)) + assertEquals(t, "3 hours", HumanDuration(3*time.Hour+59*time.Minute)) + assertEquals(t, "4 hours", HumanDuration(3*time.Hour+60*time.Minute)) + assertEquals(t, "24 hours", HumanDuration(24*time.Hour)) + assertEquals(t, "36 hours", HumanDuration(1*day+12*time.Hour)) + assertEquals(t, "2 days", HumanDuration(2*day)) + assertEquals(t, "7 days", HumanDuration(7*day)) + assertEquals(t, "13 days", HumanDuration(13*day+5*time.Hour)) + assertEquals(t, "2 weeks", HumanDuration(2*week)) + assertEquals(t, "2 weeks", HumanDuration(2*week+4*day)) + assertEquals(t, "3 weeks", HumanDuration(3*week)) + assertEquals(t, "4 weeks", HumanDuration(4*week)) + assertEquals(t, "4 weeks", HumanDuration(4*week+3*day)) + assertEquals(t, "4 weeks", HumanDuration(1*month)) + assertEquals(t, "6 weeks", HumanDuration(1*month+2*week)) + assertEquals(t, "8 weeks", HumanDuration(2*month)) + assertEquals(t, "3 months", HumanDuration(3*month+1*week)) + assertEquals(t, "5 months", HumanDuration(5*month+2*week)) + assertEquals(t, "13 months", HumanDuration(13*month)) + assertEquals(t, "23 months", HumanDuration(23*month)) + assertEquals(t, "24 months", HumanDuration(24*month)) + assertEquals(t, "2 years", HumanDuration(24*month+2*week)) + assertEquals(t, "3 years", HumanDuration(3*year+2*month)) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/units/size.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/units/size.go new file mode 100644 index 0000000000000..2fde3b412a4bf --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/units/size.go @@ -0,0 +1,95 @@ +package units + +import ( + "fmt" + "regexp" + "strconv" + "strings" +) + +// See: http://en.wikipedia.org/wiki/Binary_prefix +const ( + // Decimal + + KB = 1000 + MB = 1000 * KB + GB = 1000 * MB + TB = 1000 * GB + PB = 1000 * TB + + // Binary + + KiB = 1024 + MiB = 1024 * KiB + GiB = 1024 * MiB + TiB = 1024 * GiB + PiB = 1024 * TiB +) + +type unitMap map[string]int64 + +var ( + decimalMap = unitMap{"k": KB, "m": MB, "g": GB, "t": TB, "p": PB} + binaryMap = unitMap{"k": KiB, "m": MiB, "g": GiB, "t": TiB, "p": PiB} + sizeRegex = regexp.MustCompile(`^(\d+)([kKmMgGtTpP])?[bB]?$`) +) + +var decimapAbbrs = []string{"B", "kB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB"} +var binaryAbbrs = []string{"B", "KiB", "MiB", "GiB", "TiB", "PiB", "EiB", "ZiB", "YiB"} + +// CustomSize returns a human-readable approximation of a size +// using custom format. +func CustomSize(format string, size float64, base float64, _map []string) string { + i := 0 + for size >= base { + size = size / base + i++ + } + return fmt.Sprintf(format, size, _map[i]) +} + +// HumanSize returns a human-readable approximation of a size +// using SI standard (eg. "44kB", "17MB"). +func HumanSize(size float64) string { + return CustomSize("%.4g %s", size, 1000.0, decimapAbbrs) +} + +// BytesSize returns a human-readable size in bytes, kibibytes, +// mebibytes, gibibytes, or tebibytes (eg. "44kiB", "17MiB"). +func BytesSize(size float64) string { + return CustomSize("%.4g %s", size, 1024.0, binaryAbbrs) +} + +// FromHumanSize returns an integer from a human-readable specification of a +// size using SI standard (eg. "44kB", "17MB"). +func FromHumanSize(size string) (int64, error) { + return parseSize(size, decimalMap) +} + +// RAMInBytes parses a human-readable string representing an amount of RAM +// in bytes, kibibytes, mebibytes, gibibytes, or tebibytes and +// returns the number of bytes, or -1 if the string is unparseable. +// Units are case-insensitive, and the 'b' suffix is optional. +func RAMInBytes(size string) (int64, error) { + return parseSize(size, binaryMap) +} + +// Parses the human-readable size string into the amount it represents. +func parseSize(sizeStr string, uMap unitMap) (int64, error) { + matches := sizeRegex.FindStringSubmatch(sizeStr) + if len(matches) != 3 { + return -1, fmt.Errorf("invalid size: '%s'", sizeStr) + } + + size, err := strconv.ParseInt(matches[1], 10, 0) + if err != nil { + return -1, err + } + + unitPrefix := strings.ToLower(matches[2]) + if mul, ok := uMap[unitPrefix]; ok { + size *= mul + } + + return size, nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/units/size_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/units/size_test.go new file mode 100644 index 0000000000000..67c3b81e6b4e9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/units/size_test.go @@ -0,0 +1,108 @@ +package units + +import ( + "reflect" + "runtime" + "strings" + "testing" +) + +func TestBytesSize(t *testing.T) { + assertEquals(t, "1 KiB", BytesSize(1024)) + assertEquals(t, "1 MiB", BytesSize(1024*1024)) + assertEquals(t, "1 MiB", BytesSize(1048576)) + assertEquals(t, "2 MiB", BytesSize(2*MiB)) + assertEquals(t, "3.42 GiB", BytesSize(3.42*GiB)) + assertEquals(t, "5.372 TiB", BytesSize(5.372*TiB)) + assertEquals(t, "2.22 PiB", BytesSize(2.22*PiB)) +} + +func TestHumanSize(t *testing.T) { + assertEquals(t, "1 kB", HumanSize(1000)) + assertEquals(t, "1.024 kB", HumanSize(1024)) + assertEquals(t, "1 MB", HumanSize(1000000)) + assertEquals(t, "1.049 MB", HumanSize(1048576)) + assertEquals(t, "2 MB", HumanSize(2*MB)) + assertEquals(t, "3.42 GB", HumanSize(float64(3.42*GB))) + assertEquals(t, "5.372 TB", HumanSize(float64(5.372*TB))) + assertEquals(t, "2.22 PB", HumanSize(float64(2.22*PB))) +} + +func TestFromHumanSize(t *testing.T) { + assertSuccessEquals(t, 32, FromHumanSize, "32") + assertSuccessEquals(t, 32, FromHumanSize, "32b") + assertSuccessEquals(t, 32, FromHumanSize, "32B") + assertSuccessEquals(t, 32*KB, FromHumanSize, "32k") + assertSuccessEquals(t, 32*KB, FromHumanSize, "32K") + assertSuccessEquals(t, 32*KB, FromHumanSize, "32kb") + assertSuccessEquals(t, 32*KB, FromHumanSize, "32Kb") + assertSuccessEquals(t, 32*MB, FromHumanSize, "32Mb") + assertSuccessEquals(t, 32*GB, FromHumanSize, "32Gb") + assertSuccessEquals(t, 32*TB, FromHumanSize, "32Tb") + assertSuccessEquals(t, 32*PB, FromHumanSize, "32Pb") + + assertError(t, FromHumanSize, "") + assertError(t, FromHumanSize, "hello") + assertError(t, FromHumanSize, "-32") + assertError(t, FromHumanSize, "32.3") + assertError(t, FromHumanSize, " 32 ") + assertError(t, FromHumanSize, "32.3Kb") + assertError(t, FromHumanSize, "32 mb") + assertError(t, FromHumanSize, "32m b") + assertError(t, FromHumanSize, "32bm") +} + +func TestRAMInBytes(t *testing.T) { + assertSuccessEquals(t, 32, RAMInBytes, "32") + assertSuccessEquals(t, 32, RAMInBytes, "32b") + assertSuccessEquals(t, 32, RAMInBytes, "32B") + assertSuccessEquals(t, 32*KiB, RAMInBytes, "32k") + assertSuccessEquals(t, 32*KiB, RAMInBytes, "32K") + assertSuccessEquals(t, 32*KiB, RAMInBytes, "32kb") + assertSuccessEquals(t, 32*KiB, RAMInBytes, "32Kb") + assertSuccessEquals(t, 32*MiB, RAMInBytes, "32Mb") + assertSuccessEquals(t, 32*GiB, RAMInBytes, "32Gb") + assertSuccessEquals(t, 32*TiB, RAMInBytes, "32Tb") + assertSuccessEquals(t, 32*PiB, RAMInBytes, "32Pb") + assertSuccessEquals(t, 32*PiB, RAMInBytes, "32PB") + assertSuccessEquals(t, 32*PiB, RAMInBytes, "32P") + + assertError(t, RAMInBytes, "") + assertError(t, RAMInBytes, "hello") + assertError(t, RAMInBytes, "-32") + assertError(t, RAMInBytes, "32.3") + assertError(t, RAMInBytes, " 32 ") + assertError(t, RAMInBytes, "32.3Kb") + assertError(t, RAMInBytes, "32 mb") + assertError(t, RAMInBytes, "32m b") + assertError(t, RAMInBytes, "32bm") +} + +func assertEquals(t *testing.T, expected, actual interface{}) { + if expected != actual { + t.Errorf("Expected '%v' but got '%v'", expected, actual) + } +} + +// func that maps to the parse function signatures as testing abstraction +type parseFn func(string) (int64, error) + +// Define 'String()' for pretty-print +func (fn parseFn) String() string { + fnName := runtime.FuncForPC(reflect.ValueOf(fn).Pointer()).Name() + return fnName[strings.LastIndex(fnName, ".")+1:] +} + +func assertSuccessEquals(t *testing.T, expected int64, fn parseFn, arg string) { + res, err := fn(arg) + if err != nil || res != expected { + t.Errorf("%s(\"%s\") -> expected '%d' but got '%d' with error '%v'", fn, arg, expected, res, err) + } +} + +func assertError(t *testing.T, fn parseFn, arg string) { + res, err := fn(arg) + if err == nil && res != -1 { + t.Errorf("%s(\"%s\") -> expected error but got '%d'", fn, arg, res) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume/volume.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume/volume.go new file mode 100644 index 0000000000000..19c9d77ad9a05 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/docker/volume/volume.go @@ -0,0 +1,61 @@ +package volume + +// DefaultDriverName is the driver name used for the driver +// implemented in the local package. +const DefaultDriverName string = "local" + +// Driver is for creating and removing volumes. +type Driver interface { + // Name returns the name of the volume driver. + Name() string + // Create makes a new volume with the given id. + Create(string) (Volume, error) + // Remove deletes the volume. + Remove(Volume) error +} + +// Volume is a place to store data. It is backed by a specific driver, and can be mounted. +type Volume interface { + // Name returns the name of the volume + Name() string + // DriverName returns the name of the driver which owns this volume. + DriverName() string + // Path returns the absolute path to the volume. + Path() string + // Mount mounts the volume and returns the absolute path to + // where it can be consumed. + Mount() (string, error) + // Unmount unmounts the volume when it is no longer in use. + Unmount() error +} + +// read-write modes +var rwModes = map[string]bool{ + "rw": true, + "rw,Z": true, + "rw,z": true, + "z,rw": true, + "Z,rw": true, + "Z": true, + "z": true, +} + +// read-only modes +var roModes = map[string]bool{ + "ro": true, + "ro,Z": true, + "ro,z": true, + "z,ro": true, + "Z,ro": true, +} + +// ValidateMountMode will make sure the mount mode is valid. +// returns if it's a valid mount mode and if it's read-write or not. +func ValidateMountMode(mode string) (bool, bool) { + return roModes[mode] || rwModes[mode], rwModes[mode] +} + +// ReadWrite tells you if a mode string is a valid read-only mode or not. +func ReadWrite(mode string) bool { + return rwModes[mode] +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/MAINTAINERS b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/MAINTAINERS new file mode 100644 index 0000000000000..edbe200669461 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/MAINTAINERS @@ -0,0 +1,2 @@ +Tianon Gravi (@tianon) +Aleksa Sarai (@cyphar) diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/lookup.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/lookup.go new file mode 100644 index 0000000000000..6f8a982ff72a6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/lookup.go @@ -0,0 +1,108 @@ +package user + +import ( + "errors" + "fmt" + "syscall" +) + +var ( + // The current operating system does not provide the required data for user lookups. + ErrUnsupported = errors.New("user lookup: operating system does not provide passwd-formatted data") +) + +func lookupUser(filter func(u User) bool) (User, error) { + // Get operating system-specific passwd reader-closer. + passwd, err := GetPasswd() + if err != nil { + return User{}, err + } + defer passwd.Close() + + // Get the users. + users, err := ParsePasswdFilter(passwd, filter) + if err != nil { + return User{}, err + } + + // No user entries found. + if len(users) == 0 { + return User{}, fmt.Errorf("no matching entries in passwd file") + } + + // Assume the first entry is the "correct" one. + return users[0], nil +} + +// CurrentUser looks up the current user by their user id in /etc/passwd. If the +// user cannot be found (or there is no /etc/passwd file on the filesystem), +// then CurrentUser returns an error. +func CurrentUser() (User, error) { + return LookupUid(syscall.Getuid()) +} + +// LookupUser looks up a user by their username in /etc/passwd. If the user +// cannot be found (or there is no /etc/passwd file on the filesystem), then +// LookupUser returns an error. +func LookupUser(username string) (User, error) { + return lookupUser(func(u User) bool { + return u.Name == username + }) +} + +// LookupUid looks up a user by their user id in /etc/passwd. If the user cannot +// be found (or there is no /etc/passwd file on the filesystem), then LookupId +// returns an error. +func LookupUid(uid int) (User, error) { + return lookupUser(func(u User) bool { + return u.Uid == uid + }) +} + +func lookupGroup(filter func(g Group) bool) (Group, error) { + // Get operating system-specific group reader-closer. + group, err := GetGroup() + if err != nil { + return Group{}, err + } + defer group.Close() + + // Get the users. + groups, err := ParseGroupFilter(group, filter) + if err != nil { + return Group{}, err + } + + // No user entries found. + if len(groups) == 0 { + return Group{}, fmt.Errorf("no matching entries in group file") + } + + // Assume the first entry is the "correct" one. + return groups[0], nil +} + +// CurrentGroup looks up the current user's group by their primary group id's +// entry in /etc/passwd. If the group cannot be found (or there is no +// /etc/group file on the filesystem), then CurrentGroup returns an error. +func CurrentGroup() (Group, error) { + return LookupGid(syscall.Getgid()) +} + +// LookupGroup looks up a group by its name in /etc/group. If the group cannot +// be found (or there is no /etc/group file on the filesystem), then LookupGroup +// returns an error. +func LookupGroup(groupname string) (Group, error) { + return lookupGroup(func(g Group) bool { + return g.Name == groupname + }) +} + +// LookupGid looks up a group by its group id in /etc/group. If the group cannot +// be found (or there is no /etc/group file on the filesystem), then LookupGid +// returns an error. +func LookupGid(gid int) (Group, error) { + return lookupGroup(func(g Group) bool { + return g.Gid == gid + }) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/lookup_unix.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/lookup_unix.go new file mode 100644 index 0000000000000..758b734c225ab --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/lookup_unix.go @@ -0,0 +1,30 @@ +// +build darwin dragonfly freebsd linux netbsd openbsd solaris + +package user + +import ( + "io" + "os" +) + +// Unix-specific path to the passwd and group formatted files. +const ( + unixPasswdPath = "/etc/passwd" + unixGroupPath = "/etc/group" +) + +func GetPasswdPath() (string, error) { + return unixPasswdPath, nil +} + +func GetPasswd() (io.ReadCloser, error) { + return os.Open(unixPasswdPath) +} + +func GetGroupPath() (string, error) { + return unixGroupPath, nil +} + +func GetGroup() (io.ReadCloser, error) { + return os.Open(unixGroupPath) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/lookup_unsupported.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/lookup_unsupported.go new file mode 100644 index 0000000000000..7217948870c51 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/lookup_unsupported.go @@ -0,0 +1,21 @@ +// +build !darwin,!dragonfly,!freebsd,!linux,!netbsd,!openbsd,!solaris + +package user + +import "io" + +func GetPasswdPath() (string, error) { + return "", ErrUnsupported +} + +func GetPasswd() (io.ReadCloser, error) { + return nil, ErrUnsupported +} + +func GetGroupPath() (string, error) { + return "", ErrUnsupported +} + +func GetGroup() (io.ReadCloser, error) { + return nil, ErrUnsupported +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/user.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/user.go new file mode 100644 index 0000000000000..13226dbfa77fe --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/user.go @@ -0,0 +1,407 @@ +package user + +import ( + "bufio" + "fmt" + "io" + "os" + "strconv" + "strings" +) + +const ( + minId = 0 + maxId = 1<<31 - 1 //for 32-bit systems compatibility +) + +var ( + ErrRange = fmt.Errorf("Uids and gids must be in range %d-%d", minId, maxId) +) + +type User struct { + Name string + Pass string + Uid int + Gid int + Gecos string + Home string + Shell string +} + +type Group struct { + Name string + Pass string + Gid int + List []string +} + +func parseLine(line string, v ...interface{}) { + if line == "" { + return + } + + parts := strings.Split(line, ":") + for i, p := range parts { + if len(v) <= i { + // if we have more "parts" than we have places to put them, bail for great "tolerance" of naughty configuration files + break + } + + switch e := v[i].(type) { + case *string: + // "root", "adm", "/bin/bash" + *e = p + case *int: + // "0", "4", "1000" + // ignore string to int conversion errors, for great "tolerance" of naughty configuration files + *e, _ = strconv.Atoi(p) + case *[]string: + // "", "root", "root,adm,daemon" + if p != "" { + *e = strings.Split(p, ",") + } else { + *e = []string{} + } + default: + // panic, because this is a programming/logic error, not a runtime one + panic("parseLine expects only pointers! argument " + strconv.Itoa(i) + " is not a pointer!") + } + } +} + +func ParsePasswdFile(path string) ([]User, error) { + passwd, err := os.Open(path) + if err != nil { + return nil, err + } + defer passwd.Close() + return ParsePasswd(passwd) +} + +func ParsePasswd(passwd io.Reader) ([]User, error) { + return ParsePasswdFilter(passwd, nil) +} + +func ParsePasswdFileFilter(path string, filter func(User) bool) ([]User, error) { + passwd, err := os.Open(path) + if err != nil { + return nil, err + } + defer passwd.Close() + return ParsePasswdFilter(passwd, filter) +} + +func ParsePasswdFilter(r io.Reader, filter func(User) bool) ([]User, error) { + if r == nil { + return nil, fmt.Errorf("nil source for passwd-formatted data") + } + + var ( + s = bufio.NewScanner(r) + out = []User{} + ) + + for s.Scan() { + if err := s.Err(); err != nil { + return nil, err + } + + text := strings.TrimSpace(s.Text()) + if text == "" { + continue + } + + // see: man 5 passwd + // name:password:UID:GID:GECOS:directory:shell + // Name:Pass:Uid:Gid:Gecos:Home:Shell + // root:x:0:0:root:/root:/bin/bash + // adm:x:3:4:adm:/var/adm:/bin/false + p := User{} + parseLine( + text, + &p.Name, &p.Pass, &p.Uid, &p.Gid, &p.Gecos, &p.Home, &p.Shell, + ) + + if filter == nil || filter(p) { + out = append(out, p) + } + } + + return out, nil +} + +func ParseGroupFile(path string) ([]Group, error) { + group, err := os.Open(path) + if err != nil { + return nil, err + } + defer group.Close() + return ParseGroup(group) +} + +func ParseGroup(group io.Reader) ([]Group, error) { + return ParseGroupFilter(group, nil) +} + +func ParseGroupFileFilter(path string, filter func(Group) bool) ([]Group, error) { + group, err := os.Open(path) + if err != nil { + return nil, err + } + defer group.Close() + return ParseGroupFilter(group, filter) +} + +func ParseGroupFilter(r io.Reader, filter func(Group) bool) ([]Group, error) { + if r == nil { + return nil, fmt.Errorf("nil source for group-formatted data") + } + + var ( + s = bufio.NewScanner(r) + out = []Group{} + ) + + for s.Scan() { + if err := s.Err(); err != nil { + return nil, err + } + + text := s.Text() + if text == "" { + continue + } + + // see: man 5 group + // group_name:password:GID:user_list + // Name:Pass:Gid:List + // root:x:0:root + // adm:x:4:root,adm,daemon + p := Group{} + parseLine( + text, + &p.Name, &p.Pass, &p.Gid, &p.List, + ) + + if filter == nil || filter(p) { + out = append(out, p) + } + } + + return out, nil +} + +type ExecUser struct { + Uid, Gid int + Sgids []int + Home string +} + +// GetExecUserPath is a wrapper for GetExecUser. It reads data from each of the +// given file paths and uses that data as the arguments to GetExecUser. If the +// files cannot be opened for any reason, the error is ignored and a nil +// io.Reader is passed instead. +func GetExecUserPath(userSpec string, defaults *ExecUser, passwdPath, groupPath string) (*ExecUser, error) { + passwd, err := os.Open(passwdPath) + if err != nil { + passwd = nil + } else { + defer passwd.Close() + } + + group, err := os.Open(groupPath) + if err != nil { + group = nil + } else { + defer group.Close() + } + + return GetExecUser(userSpec, defaults, passwd, group) +} + +// GetExecUser parses a user specification string (using the passwd and group +// readers as sources for /etc/passwd and /etc/group data, respectively). In +// the case of blank fields or missing data from the sources, the values in +// defaults is used. +// +// GetExecUser will return an error if a user or group literal could not be +// found in any entry in passwd and group respectively. +// +// Examples of valid user specifications are: +// * "" +// * "user" +// * "uid" +// * "user:group" +// * "uid:gid +// * "user:gid" +// * "uid:group" +func GetExecUser(userSpec string, defaults *ExecUser, passwd, group io.Reader) (*ExecUser, error) { + var ( + userArg, groupArg string + name string + ) + + if defaults == nil { + defaults = new(ExecUser) + } + + // Copy over defaults. + user := &ExecUser{ + Uid: defaults.Uid, + Gid: defaults.Gid, + Sgids: defaults.Sgids, + Home: defaults.Home, + } + + // Sgids slice *cannot* be nil. + if user.Sgids == nil { + user.Sgids = []int{} + } + + // allow for userArg to have either "user" syntax, or optionally "user:group" syntax + parseLine(userSpec, &userArg, &groupArg) + + users, err := ParsePasswdFilter(passwd, func(u User) bool { + if userArg == "" { + return u.Uid == user.Uid + } + return u.Name == userArg || strconv.Itoa(u.Uid) == userArg + }) + if err != nil && passwd != nil { + if userArg == "" { + userArg = strconv.Itoa(user.Uid) + } + return nil, fmt.Errorf("Unable to find user %v: %v", userArg, err) + } + + haveUser := users != nil && len(users) > 0 + if haveUser { + // if we found any user entries that matched our filter, let's take the first one as "correct" + name = users[0].Name + user.Uid = users[0].Uid + user.Gid = users[0].Gid + user.Home = users[0].Home + } else if userArg != "" { + // we asked for a user but didn't find them... let's check to see if we wanted a numeric user + user.Uid, err = strconv.Atoi(userArg) + if err != nil { + // not numeric - we have to bail + return nil, fmt.Errorf("Unable to find user %v", userArg) + } + + // Must be inside valid uid range. + if user.Uid < minId || user.Uid > maxId { + return nil, ErrRange + } + + // if userArg couldn't be found in /etc/passwd but is numeric, just roll with it - this is legit + } + + if groupArg != "" || name != "" { + groups, err := ParseGroupFilter(group, func(g Group) bool { + // Explicit group format takes precedence. + if groupArg != "" { + return g.Name == groupArg || strconv.Itoa(g.Gid) == groupArg + } + + // Check if user is a member. + for _, u := range g.List { + if u == name { + return true + } + } + + return false + }) + if err != nil && group != nil { + return nil, fmt.Errorf("Unable to find groups for user %v: %v", users[0].Name, err) + } + + haveGroup := groups != nil && len(groups) > 0 + if groupArg != "" { + if haveGroup { + // if we found any group entries that matched our filter, let's take the first one as "correct" + user.Gid = groups[0].Gid + } else { + // we asked for a group but didn't find id... let's check to see if we wanted a numeric group + user.Gid, err = strconv.Atoi(groupArg) + if err != nil { + // not numeric - we have to bail + return nil, fmt.Errorf("Unable to find group %v", groupArg) + } + + // Ensure gid is inside gid range. + if user.Gid < minId || user.Gid > maxId { + return nil, ErrRange + } + + // if groupArg couldn't be found in /etc/group but is numeric, just roll with it - this is legit + } + } else if haveGroup { + // If implicit group format, fill supplementary gids. + user.Sgids = make([]int, len(groups)) + for i, group := range groups { + user.Sgids[i] = group.Gid + } + } + } + + return user, nil +} + +// GetAdditionalGroupsPath looks up a list of groups by name or group id +// against the group file. If a group name cannot be found, an error will be +// returned. If a group id cannot be found, it will be returned as-is. +func GetAdditionalGroupsPath(additionalGroups []string, groupPath string) ([]int, error) { + groupReader, err := os.Open(groupPath) + if err != nil { + return nil, fmt.Errorf("Failed to open group file: %v", err) + } + defer groupReader.Close() + + groups, err := ParseGroupFilter(groupReader, func(g Group) bool { + for _, ag := range additionalGroups { + if g.Name == ag || strconv.Itoa(g.Gid) == ag { + return true + } + } + return false + }) + if err != nil { + return nil, fmt.Errorf("Unable to find additional groups %v: %v", additionalGroups, err) + } + + gidMap := make(map[int]struct{}) + for _, ag := range additionalGroups { + var found bool + for _, g := range groups { + // if we found a matched group either by name or gid, take the + // first matched as correct + if g.Name == ag || strconv.Itoa(g.Gid) == ag { + if _, ok := gidMap[g.Gid]; !ok { + gidMap[g.Gid] = struct{}{} + found = true + break + } + } + } + // we asked for a group but didn't find it. let's check to see + // if we wanted a numeric group + if !found { + gid, err := strconv.Atoi(ag) + if err != nil { + return nil, fmt.Errorf("Unable to find group %s", ag) + } + // Ensure gid is inside gid range. + if gid < minId || gid > maxId { + return nil, ErrRange + } + gidMap[gid] = struct{}{} + } + } + gids := []int{} + for gid := range gidMap { + gids = append(gids, gid) + } + return gids, nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/user_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/user_test.go new file mode 100644 index 0000000000000..ffb0760e220d9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/docker/libcontainer/user/user_test.go @@ -0,0 +1,443 @@ +package user + +import ( + "fmt" + "io" + "io/ioutil" + "reflect" + "sort" + "strconv" + "strings" + "testing" +) + +func TestUserParseLine(t *testing.T) { + var ( + a, b string + c []string + d int + ) + + parseLine("", &a, &b) + if a != "" || b != "" { + t.Fatalf("a and b should be empty ('%v', '%v')", a, b) + } + + parseLine("a", &a, &b) + if a != "a" || b != "" { + t.Fatalf("a should be 'a' and b should be empty ('%v', '%v')", a, b) + } + + parseLine("bad boys:corny cows", &a, &b) + if a != "bad boys" || b != "corny cows" { + t.Fatalf("a should be 'bad boys' and b should be 'corny cows' ('%v', '%v')", a, b) + } + + parseLine("", &c) + if len(c) != 0 { + t.Fatalf("c should be empty (%#v)", c) + } + + parseLine("d,e,f:g:h:i,j,k", &c, &a, &b, &c) + if a != "g" || b != "h" || len(c) != 3 || c[0] != "i" || c[1] != "j" || c[2] != "k" { + t.Fatalf("a should be 'g', b should be 'h', and c should be ['i','j','k'] ('%v', '%v', '%#v')", a, b, c) + } + + parseLine("::::::::::", &a, &b, &c) + if a != "" || b != "" || len(c) != 0 { + t.Fatalf("a, b, and c should all be empty ('%v', '%v', '%#v')", a, b, c) + } + + parseLine("not a number", &d) + if d != 0 { + t.Fatalf("d should be 0 (%v)", d) + } + + parseLine("b:12:c", &a, &d, &b) + if a != "b" || b != "c" || d != 12 { + t.Fatalf("a should be 'b' and b should be 'c', and d should be 12 ('%v', '%v', %v)", a, b, d) + } +} + +func TestUserParsePasswd(t *testing.T) { + users, err := ParsePasswdFilter(strings.NewReader(` +root:x:0:0:root:/root:/bin/bash +adm:x:3:4:adm:/var/adm:/bin/false +this is just some garbage data +`), nil) + if err != nil { + t.Fatalf("Unexpected error: %v", err) + } + if len(users) != 3 { + t.Fatalf("Expected 3 users, got %v", len(users)) + } + if users[0].Uid != 0 || users[0].Name != "root" { + t.Fatalf("Expected users[0] to be 0 - root, got %v - %v", users[0].Uid, users[0].Name) + } + if users[1].Uid != 3 || users[1].Name != "adm" { + t.Fatalf("Expected users[1] to be 3 - adm, got %v - %v", users[1].Uid, users[1].Name) + } +} + +func TestUserParseGroup(t *testing.T) { + groups, err := ParseGroupFilter(strings.NewReader(` +root:x:0:root +adm:x:4:root,adm,daemon +this is just some garbage data +`), nil) + if err != nil { + t.Fatalf("Unexpected error: %v", err) + } + if len(groups) != 3 { + t.Fatalf("Expected 3 groups, got %v", len(groups)) + } + if groups[0].Gid != 0 || groups[0].Name != "root" || len(groups[0].List) != 1 { + t.Fatalf("Expected groups[0] to be 0 - root - 1 member, got %v - %v - %v", groups[0].Gid, groups[0].Name, len(groups[0].List)) + } + if groups[1].Gid != 4 || groups[1].Name != "adm" || len(groups[1].List) != 3 { + t.Fatalf("Expected groups[1] to be 4 - adm - 3 members, got %v - %v - %v", groups[1].Gid, groups[1].Name, len(groups[1].List)) + } +} + +func TestValidGetExecUser(t *testing.T) { + const passwdContent = ` +root:x:0:0:root user:/root:/bin/bash +adm:x:42:43:adm:/var/adm:/bin/false +this is just some garbage data +` + const groupContent = ` +root:x:0:root +adm:x:43: +grp:x:1234:root,adm +this is just some garbage data +` + defaultExecUser := ExecUser{ + Uid: 8888, + Gid: 8888, + Sgids: []int{8888}, + Home: "/8888", + } + + tests := []struct { + ref string + expected ExecUser + }{ + { + ref: "root", + expected: ExecUser{ + Uid: 0, + Gid: 0, + Sgids: []int{0, 1234}, + Home: "/root", + }, + }, + { + ref: "adm", + expected: ExecUser{ + Uid: 42, + Gid: 43, + Sgids: []int{1234}, + Home: "/var/adm", + }, + }, + { + ref: "root:adm", + expected: ExecUser{ + Uid: 0, + Gid: 43, + Sgids: defaultExecUser.Sgids, + Home: "/root", + }, + }, + { + ref: "adm:1234", + expected: ExecUser{ + Uid: 42, + Gid: 1234, + Sgids: defaultExecUser.Sgids, + Home: "/var/adm", + }, + }, + { + ref: "42:1234", + expected: ExecUser{ + Uid: 42, + Gid: 1234, + Sgids: defaultExecUser.Sgids, + Home: "/var/adm", + }, + }, + { + ref: "1337:1234", + expected: ExecUser{ + Uid: 1337, + Gid: 1234, + Sgids: defaultExecUser.Sgids, + Home: defaultExecUser.Home, + }, + }, + { + ref: "1337", + expected: ExecUser{ + Uid: 1337, + Gid: defaultExecUser.Gid, + Sgids: defaultExecUser.Sgids, + Home: defaultExecUser.Home, + }, + }, + { + ref: "", + expected: ExecUser{ + Uid: defaultExecUser.Uid, + Gid: defaultExecUser.Gid, + Sgids: defaultExecUser.Sgids, + Home: defaultExecUser.Home, + }, + }, + } + + for _, test := range tests { + passwd := strings.NewReader(passwdContent) + group := strings.NewReader(groupContent) + + execUser, err := GetExecUser(test.ref, &defaultExecUser, passwd, group) + if err != nil { + t.Logf("got unexpected error when parsing '%s': %s", test.ref, err.Error()) + t.Fail() + continue + } + + if !reflect.DeepEqual(test.expected, *execUser) { + t.Logf("got: %#v", execUser) + t.Logf("expected: %#v", test.expected) + t.Fail() + continue + } + } +} + +func TestInvalidGetExecUser(t *testing.T) { + const passwdContent = ` +root:x:0:0:root user:/root:/bin/bash +adm:x:42:43:adm:/var/adm:/bin/false +this is just some garbage data +` + const groupContent = ` +root:x:0:root +adm:x:43: +grp:x:1234:root,adm +this is just some garbage data +` + + tests := []string{ + // No such user/group. + "notuser", + "notuser:notgroup", + "root:notgroup", + "notuser:adm", + "8888:notgroup", + "notuser:8888", + + // Invalid user/group values. + "-1:0", + "0:-3", + "-5:-2", + } + + for _, test := range tests { + passwd := strings.NewReader(passwdContent) + group := strings.NewReader(groupContent) + + execUser, err := GetExecUser(test, nil, passwd, group) + if err == nil { + t.Logf("got unexpected success when parsing '%s': %#v", test, execUser) + t.Fail() + continue + } + } +} + +func TestGetExecUserNilSources(t *testing.T) { + const passwdContent = ` +root:x:0:0:root user:/root:/bin/bash +adm:x:42:43:adm:/var/adm:/bin/false +this is just some garbage data +` + const groupContent = ` +root:x:0:root +adm:x:43: +grp:x:1234:root,adm +this is just some garbage data +` + + defaultExecUser := ExecUser{ + Uid: 8888, + Gid: 8888, + Sgids: []int{8888}, + Home: "/8888", + } + + tests := []struct { + ref string + passwd, group bool + expected ExecUser + }{ + { + ref: "", + passwd: false, + group: false, + expected: ExecUser{ + Uid: 8888, + Gid: 8888, + Sgids: []int{8888}, + Home: "/8888", + }, + }, + { + ref: "root", + passwd: true, + group: false, + expected: ExecUser{ + Uid: 0, + Gid: 0, + Sgids: []int{8888}, + Home: "/root", + }, + }, + { + ref: "0", + passwd: false, + group: false, + expected: ExecUser{ + Uid: 0, + Gid: 8888, + Sgids: []int{8888}, + Home: "/8888", + }, + }, + { + ref: "0:0", + passwd: false, + group: false, + expected: ExecUser{ + Uid: 0, + Gid: 0, + Sgids: []int{8888}, + Home: "/8888", + }, + }, + } + + for _, test := range tests { + var passwd, group io.Reader + + if test.passwd { + passwd = strings.NewReader(passwdContent) + } + + if test.group { + group = strings.NewReader(groupContent) + } + + execUser, err := GetExecUser(test.ref, &defaultExecUser, passwd, group) + if err != nil { + t.Logf("got unexpected error when parsing '%s': %s", test.ref, err.Error()) + t.Fail() + continue + } + + if !reflect.DeepEqual(test.expected, *execUser) { + t.Logf("got: %#v", execUser) + t.Logf("expected: %#v", test.expected) + t.Fail() + continue + } + } +} + +func TestGetAdditionalGroupsPath(t *testing.T) { + const groupContent = ` +root:x:0:root +adm:x:43: +grp:x:1234:root,adm +adm:x:4343:root,adm-duplicate +this is just some garbage data +` + tests := []struct { + groups []string + expected []int + hasError bool + }{ + { + // empty group + groups: []string{}, + expected: []int{}, + }, + { + // single group + groups: []string{"adm"}, + expected: []int{43}, + }, + { + // multiple groups + groups: []string{"adm", "grp"}, + expected: []int{43, 1234}, + }, + { + // invalid group + groups: []string{"adm", "grp", "not-exist"}, + expected: nil, + hasError: true, + }, + { + // group with numeric id + groups: []string{"43"}, + expected: []int{43}, + }, + { + // group with unknown numeric id + groups: []string{"adm", "10001"}, + expected: []int{43, 10001}, + }, + { + // groups specified twice with numeric and name + groups: []string{"adm", "43"}, + expected: []int{43}, + }, + { + // groups with too small id + groups: []string{"-1"}, + expected: nil, + hasError: true, + }, + { + // groups with too large id + groups: []string{strconv.Itoa(1 << 31)}, + expected: nil, + hasError: true, + }, + } + + for _, test := range tests { + tmpFile, err := ioutil.TempFile("", "get-additional-groups-path") + if err != nil { + t.Error(err) + } + fmt.Fprint(tmpFile, groupContent) + tmpFile.Close() + + gids, err := GetAdditionalGroupsPath(test.groups, tmpFile.Name()) + if test.hasError && err == nil { + t.Errorf("Parse(%#v) expects error but has none", test) + continue + } + if !test.hasError && err != nil { + t.Errorf("Parse(%#v) has error %v", test, err) + continue + } + sort.Sort(sort.IntSlice(gids)) + if !reflect.DeepEqual(gids, test.expected) { + t.Errorf("Gids(%v), expect %v from groups %v", gids, test.expected, test.groups) + } + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/LICENSE b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/LICENSE new file mode 100644 index 0000000000000..0e5fb872800da --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/LICENSE @@ -0,0 +1,27 @@ +Copyright (c) 2012 Rodrigo Moraes. All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are +met: + + * Redistributions of source code must retain the above copyright +notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above +copyright notice, this list of conditions and the following disclaimer +in the documentation and/or other materials provided with the +distribution. + * Neither the name of Google Inc. nor the names of its +contributors may be used to endorse or promote products derived from +this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/README.md b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/README.md new file mode 100644 index 0000000000000..c60a31b053bc4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/README.md @@ -0,0 +1,7 @@ +context +======= +[![Build Status](https://travis-ci.org/gorilla/context.png?branch=master)](https://travis-ci.org/gorilla/context) + +gorilla/context is a general purpose registry for global request variables. + +Read the full documentation here: http://www.gorillatoolkit.org/pkg/context diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go new file mode 100644 index 0000000000000..81cb128b19cad --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context.go @@ -0,0 +1,143 @@ +// Copyright 2012 The Gorilla Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package context + +import ( + "net/http" + "sync" + "time" +) + +var ( + mutex sync.RWMutex + data = make(map[*http.Request]map[interface{}]interface{}) + datat = make(map[*http.Request]int64) +) + +// Set stores a value for a given key in a given request. +func Set(r *http.Request, key, val interface{}) { + mutex.Lock() + if data[r] == nil { + data[r] = make(map[interface{}]interface{}) + datat[r] = time.Now().Unix() + } + data[r][key] = val + mutex.Unlock() +} + +// Get returns a value stored for a given key in a given request. +func Get(r *http.Request, key interface{}) interface{} { + mutex.RLock() + if ctx := data[r]; ctx != nil { + value := ctx[key] + mutex.RUnlock() + return value + } + mutex.RUnlock() + return nil +} + +// GetOk returns stored value and presence state like multi-value return of map access. +func GetOk(r *http.Request, key interface{}) (interface{}, bool) { + mutex.RLock() + if _, ok := data[r]; ok { + value, ok := data[r][key] + mutex.RUnlock() + return value, ok + } + mutex.RUnlock() + return nil, false +} + +// GetAll returns all stored values for the request as a map. Nil is returned for invalid requests. +func GetAll(r *http.Request) map[interface{}]interface{} { + mutex.RLock() + if context, ok := data[r]; ok { + result := make(map[interface{}]interface{}, len(context)) + for k, v := range context { + result[k] = v + } + mutex.RUnlock() + return result + } + mutex.RUnlock() + return nil +} + +// GetAllOk returns all stored values for the request as a map and a boolean value that indicates if +// the request was registered. +func GetAllOk(r *http.Request) (map[interface{}]interface{}, bool) { + mutex.RLock() + context, ok := data[r] + result := make(map[interface{}]interface{}, len(context)) + for k, v := range context { + result[k] = v + } + mutex.RUnlock() + return result, ok +} + +// Delete removes a value stored for a given key in a given request. +func Delete(r *http.Request, key interface{}) { + mutex.Lock() + if data[r] != nil { + delete(data[r], key) + } + mutex.Unlock() +} + +// Clear removes all values stored for a given request. +// +// This is usually called by a handler wrapper to clean up request +// variables at the end of a request lifetime. See ClearHandler(). +func Clear(r *http.Request) { + mutex.Lock() + clear(r) + mutex.Unlock() +} + +// clear is Clear without the lock. +func clear(r *http.Request) { + delete(data, r) + delete(datat, r) +} + +// Purge removes request data stored for longer than maxAge, in seconds. +// It returns the amount of requests removed. +// +// If maxAge <= 0, all request data is removed. +// +// This is only used for sanity check: in case context cleaning was not +// properly set some request data can be kept forever, consuming an increasing +// amount of memory. In case this is detected, Purge() must be called +// periodically until the problem is fixed. +func Purge(maxAge int) int { + mutex.Lock() + count := 0 + if maxAge <= 0 { + count = len(data) + data = make(map[*http.Request]map[interface{}]interface{}) + datat = make(map[*http.Request]int64) + } else { + min := time.Now().Unix() - int64(maxAge) + for r := range data { + if datat[r] < min { + clear(r) + count++ + } + } + } + mutex.Unlock() + return count +} + +// ClearHandler wraps an http.Handler and clears request values at the end +// of a request lifetime. +func ClearHandler(h http.Handler) http.Handler { + return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + defer Clear(r) + h.ServeHTTP(w, r) + }) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context_test.go new file mode 100644 index 0000000000000..9814c501e8aef --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/context_test.go @@ -0,0 +1,161 @@ +// Copyright 2012 The Gorilla Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package context + +import ( + "net/http" + "testing" +) + +type keyType int + +const ( + key1 keyType = iota + key2 +) + +func TestContext(t *testing.T) { + assertEqual := func(val interface{}, exp interface{}) { + if val != exp { + t.Errorf("Expected %v, got %v.", exp, val) + } + } + + r, _ := http.NewRequest("GET", "http://localhost:8080/", nil) + emptyR, _ := http.NewRequest("GET", "http://localhost:8080/", nil) + + // Get() + assertEqual(Get(r, key1), nil) + + // Set() + Set(r, key1, "1") + assertEqual(Get(r, key1), "1") + assertEqual(len(data[r]), 1) + + Set(r, key2, "2") + assertEqual(Get(r, key2), "2") + assertEqual(len(data[r]), 2) + + //GetOk + value, ok := GetOk(r, key1) + assertEqual(value, "1") + assertEqual(ok, true) + + value, ok = GetOk(r, "not exists") + assertEqual(value, nil) + assertEqual(ok, false) + + Set(r, "nil value", nil) + value, ok = GetOk(r, "nil value") + assertEqual(value, nil) + assertEqual(ok, true) + + // GetAll() + values := GetAll(r) + assertEqual(len(values), 3) + + // GetAll() for empty request + values = GetAll(emptyR) + if values != nil { + t.Error("GetAll didn't return nil value for invalid request") + } + + // GetAllOk() + values, ok = GetAllOk(r) + assertEqual(len(values), 3) + assertEqual(ok, true) + + // GetAllOk() for empty request + values, ok = GetAllOk(emptyR) + assertEqual(value, nil) + assertEqual(ok, false) + + // Delete() + Delete(r, key1) + assertEqual(Get(r, key1), nil) + assertEqual(len(data[r]), 2) + + Delete(r, key2) + assertEqual(Get(r, key2), nil) + assertEqual(len(data[r]), 1) + + // Clear() + Clear(r) + assertEqual(len(data), 0) +} + +func parallelReader(r *http.Request, key string, iterations int, wait, done chan struct{}) { + <-wait + for i := 0; i < iterations; i++ { + Get(r, key) + } + done <- struct{}{} + +} + +func parallelWriter(r *http.Request, key, value string, iterations int, wait, done chan struct{}) { + <-wait + for i := 0; i < iterations; i++ { + Set(r, key, value) + } + done <- struct{}{} + +} + +func benchmarkMutex(b *testing.B, numReaders, numWriters, iterations int) { + + b.StopTimer() + r, _ := http.NewRequest("GET", "http://localhost:8080/", nil) + done := make(chan struct{}) + b.StartTimer() + + for i := 0; i < b.N; i++ { + wait := make(chan struct{}) + + for i := 0; i < numReaders; i++ { + go parallelReader(r, "test", iterations, wait, done) + } + + for i := 0; i < numWriters; i++ { + go parallelWriter(r, "test", "123", iterations, wait, done) + } + + close(wait) + + for i := 0; i < numReaders+numWriters; i++ { + <-done + } + + } + +} + +func BenchmarkMutexSameReadWrite1(b *testing.B) { + benchmarkMutex(b, 1, 1, 32) +} +func BenchmarkMutexSameReadWrite2(b *testing.B) { + benchmarkMutex(b, 2, 2, 32) +} +func BenchmarkMutexSameReadWrite4(b *testing.B) { + benchmarkMutex(b, 4, 4, 32) +} +func BenchmarkMutex1(b *testing.B) { + benchmarkMutex(b, 2, 8, 32) +} +func BenchmarkMutex2(b *testing.B) { + benchmarkMutex(b, 16, 4, 64) +} +func BenchmarkMutex3(b *testing.B) { + benchmarkMutex(b, 1, 2, 128) +} +func BenchmarkMutex4(b *testing.B) { + benchmarkMutex(b, 128, 32, 256) +} +func BenchmarkMutex5(b *testing.B) { + benchmarkMutex(b, 1024, 2048, 64) +} +func BenchmarkMutex6(b *testing.B) { + benchmarkMutex(b, 2048, 1024, 512) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/doc.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/doc.go new file mode 100644 index 0000000000000..73c7400311e12 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/context/doc.go @@ -0,0 +1,82 @@ +// Copyright 2012 The Gorilla Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +/* +Package context stores values shared during a request lifetime. + +For example, a router can set variables extracted from the URL and later +application handlers can access those values, or it can be used to store +sessions values to be saved at the end of a request. There are several +others common uses. + +The idea was posted by Brad Fitzpatrick to the go-nuts mailing list: + + http://groups.google.com/group/golang-nuts/msg/e2d679d303aa5d53 + +Here's the basic usage: first define the keys that you will need. The key +type is interface{} so a key can be of any type that supports equality. +Here we define a key using a custom int type to avoid name collisions: + + package foo + + import ( + "github.com/gorilla/context" + ) + + type key int + + const MyKey key = 0 + +Then set a variable. Variables are bound to an http.Request object, so you +need a request instance to set a value: + + context.Set(r, MyKey, "bar") + +The application can later access the variable using the same key you provided: + + func MyHandler(w http.ResponseWriter, r *http.Request) { + // val is "bar". + val := context.Get(r, foo.MyKey) + + // returns ("bar", true) + val, ok := context.GetOk(r, foo.MyKey) + // ... + } + +And that's all about the basic usage. We discuss some other ideas below. + +Any type can be stored in the context. To enforce a given type, make the key +private and wrap Get() and Set() to accept and return values of a specific +type: + + type key int + + const mykey key = 0 + + // GetMyKey returns a value for this package from the request values. + func GetMyKey(r *http.Request) SomeType { + if rv := context.Get(r, mykey); rv != nil { + return rv.(SomeType) + } + return nil + } + + // SetMyKey sets a value for this package in the request values. + func SetMyKey(r *http.Request, val SomeType) { + context.Set(r, mykey, val) + } + +Variables must be cleared at the end of a request, to remove all values +that were stored. This can be done in an http.Handler, after a request was +served. Just call Clear() passing the request: + + context.Clear(r) + +...or use ClearHandler(), which conveniently wraps an http.Handler to clear +variables at the end of a request lifetime. + +The Routers from the packages gorilla/mux and gorilla/pat call Clear() +so if you are using either of them you don't need to clear the context manually. +*/ +package context diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/LICENSE b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/LICENSE new file mode 100644 index 0000000000000..0e5fb872800da --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/LICENSE @@ -0,0 +1,27 @@ +Copyright (c) 2012 Rodrigo Moraes. All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are +met: + + * Redistributions of source code must retain the above copyright +notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above +copyright notice, this list of conditions and the following disclaimer +in the documentation and/or other materials provided with the +distribution. + * Neither the name of Google Inc. nor the names of its +contributors may be used to endorse or promote products derived from +this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/README.md b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/README.md new file mode 100644 index 0000000000000..9a046ff97faa6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/README.md @@ -0,0 +1,235 @@ +mux +=== +[![GoDoc](https://godoc.org/github.com/gorilla/securecookie?status.svg)](https://godoc.org/github.com/gorilla/securecookie) +[![Build Status](https://travis-ci.org/gorilla/mux.png?branch=master)](https://travis-ci.org/gorilla/mux) + +Package gorilla/mux implements a request router and dispatcher. + +The name mux stands for "HTTP request multiplexer". Like the standard +http.ServeMux, mux.Router matches incoming requests against a list of +registered routes and calls a handler for the route that matches the URL +or other conditions. The main features are: + + * Requests can be matched based on URL host, path, path prefix, schemes, + header and query values, HTTP methods or using custom matchers. + * URL hosts and paths can have variables with an optional regular + expression. + * Registered URLs can be built, or "reversed", which helps maintaining + references to resources. + * Routes can be used as subrouters: nested routes are only tested if the + parent route matches. This is useful to define groups of routes that + share common conditions like a host, a path prefix or other repeated + attributes. As a bonus, this optimizes request matching. + * It implements the http.Handler interface so it is compatible with the + standard http.ServeMux. + +Let's start registering a couple of URL paths and handlers: + + func main() { + r := mux.NewRouter() + r.HandleFunc("/", HomeHandler) + r.HandleFunc("/products", ProductsHandler) + r.HandleFunc("/articles", ArticlesHandler) + http.Handle("/", r) + } + +Here we register three routes mapping URL paths to handlers. This is +equivalent to how http.HandleFunc() works: if an incoming request URL matches +one of the paths, the corresponding handler is called passing +(http.ResponseWriter, *http.Request) as parameters. + +Paths can have variables. They are defined using the format {name} or +{name:pattern}. If a regular expression pattern is not defined, the matched +variable will be anything until the next slash. For example: + + r := mux.NewRouter() + r.HandleFunc("/products/{key}", ProductHandler) + r.HandleFunc("/articles/{category}/", ArticlesCategoryHandler) + r.HandleFunc("/articles/{category}/{id:[0-9]+}", ArticleHandler) + +The names are used to create a map of route variables which can be retrieved +calling mux.Vars(): + + vars := mux.Vars(request) + category := vars["category"] + +And this is all you need to know about the basic usage. More advanced options +are explained below. + +Routes can also be restricted to a domain or subdomain. Just define a host +pattern to be matched. They can also have variables: + + r := mux.NewRouter() + // Only matches if domain is "www.example.com". + r.Host("www.example.com") + // Matches a dynamic subdomain. + r.Host("{subdomain:[a-z]+}.domain.com") + +There are several other matchers that can be added. To match path prefixes: + + r.PathPrefix("/products/") + +...or HTTP methods: + + r.Methods("GET", "POST") + +...or URL schemes: + + r.Schemes("https") + +...or header values: + + r.Headers("X-Requested-With", "XMLHttpRequest") + +...or query values: + + r.Queries("key", "value") + +...or to use a custom matcher function: + + r.MatcherFunc(func(r *http.Request, rm *RouteMatch) bool { + return r.ProtoMajor == 0 + }) + +...and finally, it is possible to combine several matchers in a single route: + + r.HandleFunc("/products", ProductsHandler). + Host("www.example.com"). + Methods("GET"). + Schemes("http") + +Setting the same matching conditions again and again can be boring, so we have +a way to group several routes that share the same requirements. +We call it "subrouting". + +For example, let's say we have several URLs that should only match when the +host is `www.example.com`. Create a route for that host and get a "subrouter" +from it: + + r := mux.NewRouter() + s := r.Host("www.example.com").Subrouter() + +Then register routes in the subrouter: + + s.HandleFunc("/products/", ProductsHandler) + s.HandleFunc("/products/{key}", ProductHandler) + s.HandleFunc("/articles/{category}/{id:[0-9]+}"), ArticleHandler) + +The three URL paths we registered above will only be tested if the domain is +`www.example.com`, because the subrouter is tested first. This is not +only convenient, but also optimizes request matching. You can create +subrouters combining any attribute matchers accepted by a route. + +Subrouters can be used to create domain or path "namespaces": you define +subrouters in a central place and then parts of the app can register its +paths relatively to a given subrouter. + +There's one more thing about subroutes. When a subrouter has a path prefix, +the inner routes use it as base for their paths: + + r := mux.NewRouter() + s := r.PathPrefix("/products").Subrouter() + // "/products/" + s.HandleFunc("/", ProductsHandler) + // "/products/{key}/" + s.HandleFunc("/{key}/", ProductHandler) + // "/products/{key}/details" + s.HandleFunc("/{key}/details", ProductDetailsHandler) + +Now let's see how to build registered URLs. + +Routes can be named. All routes that define a name can have their URLs built, +or "reversed". We define a name calling Name() on a route. For example: + + r := mux.NewRouter() + r.HandleFunc("/articles/{category}/{id:[0-9]+}", ArticleHandler). + Name("article") + +To build a URL, get the route and call the URL() method, passing a sequence of +key/value pairs for the route variables. For the previous route, we would do: + + url, err := r.Get("article").URL("category", "technology", "id", "42") + +...and the result will be a url.URL with the following path: + + "/articles/technology/42" + +This also works for host variables: + + r := mux.NewRouter() + r.Host("{subdomain}.domain.com"). + Path("/articles/{category}/{id:[0-9]+}"). + HandlerFunc(ArticleHandler). + Name("article") + + // url.String() will be "http://news.domain.com/articles/technology/42" + url, err := r.Get("article").URL("subdomain", "news", + "category", "technology", + "id", "42") + +All variables defined in the route are required, and their values must +conform to the corresponding patterns. These requirements guarantee that a +generated URL will always match a registered route -- the only exception is +for explicitly defined "build-only" routes which never match. + +Regex support also exists for matching Headers within a route. For example, we could do: + + r.HeadersRegexp("Content-Type", "application/(text|json)") + +...and the route will match both requests with a Content-Type of `application/json` as well as +`application/text` + +There's also a way to build only the URL host or path for a route: +use the methods URLHost() or URLPath() instead. For the previous route, +we would do: + + // "http://news.domain.com/" + host, err := r.Get("article").URLHost("subdomain", "news") + + // "/articles/technology/42" + path, err := r.Get("article").URLPath("category", "technology", "id", "42") + +And if you use subrouters, host and path defined separately can be built +as well: + + r := mux.NewRouter() + s := r.Host("{subdomain}.domain.com").Subrouter() + s.Path("/articles/{category}/{id:[0-9]+}"). + HandlerFunc(ArticleHandler). + Name("article") + + // "http://news.domain.com/articles/technology/42" + url, err := r.Get("article").URL("subdomain", "news", + "category", "technology", + "id", "42") + +## Full Example + +Here's a complete, runnable example of a small mux based server: + +```go +package main + +import ( + "net/http" + + "github.com/gorilla/mux" +) + +func YourHandler(w http.ResponseWriter, r *http.Request) { + w.Write([]byte("Gorilla!\n")) +} + +func main() { + r := mux.NewRouter() + // Routes consist of a path and a handler function. + r.HandleFunc("/", YourHandler) + + // Bind to a port and pass our router in + http.ListenAndServe(":8000", r) +} +``` + +## License + +BSD licensed. See the LICENSE file for details. diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/bench_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/bench_test.go new file mode 100644 index 0000000000000..c5f97b2b2a80b --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/bench_test.go @@ -0,0 +1,21 @@ +// Copyright 2012 The Gorilla Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package mux + +import ( + "net/http" + "testing" +) + +func BenchmarkMux(b *testing.B) { + router := new(Router) + handler := func(w http.ResponseWriter, r *http.Request) {} + router.HandleFunc("/v1/{v1}", handler) + + request, _ := http.NewRequest("GET", "/v1/anything", nil) + for i := 0; i < b.N; i++ { + router.ServeHTTP(nil, request) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/doc.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/doc.go new file mode 100644 index 0000000000000..49798cb5cf529 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/doc.go @@ -0,0 +1,206 @@ +// Copyright 2012 The Gorilla Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +/* +Package gorilla/mux implements a request router and dispatcher. + +The name mux stands for "HTTP request multiplexer". Like the standard +http.ServeMux, mux.Router matches incoming requests against a list of +registered routes and calls a handler for the route that matches the URL +or other conditions. The main features are: + + * Requests can be matched based on URL host, path, path prefix, schemes, + header and query values, HTTP methods or using custom matchers. + * URL hosts and paths can have variables with an optional regular + expression. + * Registered URLs can be built, or "reversed", which helps maintaining + references to resources. + * Routes can be used as subrouters: nested routes are only tested if the + parent route matches. This is useful to define groups of routes that + share common conditions like a host, a path prefix or other repeated + attributes. As a bonus, this optimizes request matching. + * It implements the http.Handler interface so it is compatible with the + standard http.ServeMux. + +Let's start registering a couple of URL paths and handlers: + + func main() { + r := mux.NewRouter() + r.HandleFunc("/", HomeHandler) + r.HandleFunc("/products", ProductsHandler) + r.HandleFunc("/articles", ArticlesHandler) + http.Handle("/", r) + } + +Here we register three routes mapping URL paths to handlers. This is +equivalent to how http.HandleFunc() works: if an incoming request URL matches +one of the paths, the corresponding handler is called passing +(http.ResponseWriter, *http.Request) as parameters. + +Paths can have variables. They are defined using the format {name} or +{name:pattern}. If a regular expression pattern is not defined, the matched +variable will be anything until the next slash. For example: + + r := mux.NewRouter() + r.HandleFunc("/products/{key}", ProductHandler) + r.HandleFunc("/articles/{category}/", ArticlesCategoryHandler) + r.HandleFunc("/articles/{category}/{id:[0-9]+}", ArticleHandler) + +The names are used to create a map of route variables which can be retrieved +calling mux.Vars(): + + vars := mux.Vars(request) + category := vars["category"] + +And this is all you need to know about the basic usage. More advanced options +are explained below. + +Routes can also be restricted to a domain or subdomain. Just define a host +pattern to be matched. They can also have variables: + + r := mux.NewRouter() + // Only matches if domain is "www.example.com". + r.Host("www.example.com") + // Matches a dynamic subdomain. + r.Host("{subdomain:[a-z]+}.domain.com") + +There are several other matchers that can be added. To match path prefixes: + + r.PathPrefix("/products/") + +...or HTTP methods: + + r.Methods("GET", "POST") + +...or URL schemes: + + r.Schemes("https") + +...or header values: + + r.Headers("X-Requested-With", "XMLHttpRequest") + +...or query values: + + r.Queries("key", "value") + +...or to use a custom matcher function: + + r.MatcherFunc(func(r *http.Request, rm *RouteMatch) bool { + return r.ProtoMajor == 0 + }) + +...and finally, it is possible to combine several matchers in a single route: + + r.HandleFunc("/products", ProductsHandler). + Host("www.example.com"). + Methods("GET"). + Schemes("http") + +Setting the same matching conditions again and again can be boring, so we have +a way to group several routes that share the same requirements. +We call it "subrouting". + +For example, let's say we have several URLs that should only match when the +host is "www.example.com". Create a route for that host and get a "subrouter" +from it: + + r := mux.NewRouter() + s := r.Host("www.example.com").Subrouter() + +Then register routes in the subrouter: + + s.HandleFunc("/products/", ProductsHandler) + s.HandleFunc("/products/{key}", ProductHandler) + s.HandleFunc("/articles/{category}/{id:[0-9]+}"), ArticleHandler) + +The three URL paths we registered above will only be tested if the domain is +"www.example.com", because the subrouter is tested first. This is not +only convenient, but also optimizes request matching. You can create +subrouters combining any attribute matchers accepted by a route. + +Subrouters can be used to create domain or path "namespaces": you define +subrouters in a central place and then parts of the app can register its +paths relatively to a given subrouter. + +There's one more thing about subroutes. When a subrouter has a path prefix, +the inner routes use it as base for their paths: + + r := mux.NewRouter() + s := r.PathPrefix("/products").Subrouter() + // "/products/" + s.HandleFunc("/", ProductsHandler) + // "/products/{key}/" + s.HandleFunc("/{key}/", ProductHandler) + // "/products/{key}/details" + s.HandleFunc("/{key}/details", ProductDetailsHandler) + +Now let's see how to build registered URLs. + +Routes can be named. All routes that define a name can have their URLs built, +or "reversed". We define a name calling Name() on a route. For example: + + r := mux.NewRouter() + r.HandleFunc("/articles/{category}/{id:[0-9]+}", ArticleHandler). + Name("article") + +To build a URL, get the route and call the URL() method, passing a sequence of +key/value pairs for the route variables. For the previous route, we would do: + + url, err := r.Get("article").URL("category", "technology", "id", "42") + +...and the result will be a url.URL with the following path: + + "/articles/technology/42" + +This also works for host variables: + + r := mux.NewRouter() + r.Host("{subdomain}.domain.com"). + Path("/articles/{category}/{id:[0-9]+}"). + HandlerFunc(ArticleHandler). + Name("article") + + // url.String() will be "http://news.domain.com/articles/technology/42" + url, err := r.Get("article").URL("subdomain", "news", + "category", "technology", + "id", "42") + +All variables defined in the route are required, and their values must +conform to the corresponding patterns. These requirements guarantee that a +generated URL will always match a registered route -- the only exception is +for explicitly defined "build-only" routes which never match. + +Regex support also exists for matching Headers within a route. For example, we could do: + + r.HeadersRegexp("Content-Type", "application/(text|json)") + +...and the route will match both requests with a Content-Type of `application/json` as well as +`application/text` + +There's also a way to build only the URL host or path for a route: +use the methods URLHost() or URLPath() instead. For the previous route, +we would do: + + // "http://news.domain.com/" + host, err := r.Get("article").URLHost("subdomain", "news") + + // "/articles/technology/42" + path, err := r.Get("article").URLPath("category", "technology", "id", "42") + +And if you use subrouters, host and path defined separately can be built +as well: + + r := mux.NewRouter() + s := r.Host("{subdomain}.domain.com").Subrouter() + s.Path("/articles/{category}/{id:[0-9]+}"). + HandlerFunc(ArticleHandler). + Name("article") + + // "http://news.domain.com/articles/technology/42" + url, err := r.Get("article").URL("subdomain", "news", + "category", "technology", + "id", "42") +*/ +package mux diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go new file mode 100644 index 0000000000000..b32e1a0512716 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux.go @@ -0,0 +1,469 @@ +// Copyright 2012 The Gorilla Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package mux + +import ( + "errors" + "fmt" + "net/http" + "path" + "regexp" + + "github.com/fsouza/go-dockerclient/external/github.com/gorilla/context" +) + +// NewRouter returns a new router instance. +func NewRouter() *Router { + return &Router{namedRoutes: make(map[string]*Route), KeepContext: false} +} + +// Router registers routes to be matched and dispatches a handler. +// +// It implements the http.Handler interface, so it can be registered to serve +// requests: +// +// var router = mux.NewRouter() +// +// func main() { +// http.Handle("/", router) +// } +// +// Or, for Google App Engine, register it in a init() function: +// +// func init() { +// http.Handle("/", router) +// } +// +// This will send all incoming requests to the router. +type Router struct { + // Configurable Handler to be used when no route matches. + NotFoundHandler http.Handler + // Parent route, if this is a subrouter. + parent parentRoute + // Routes to be matched, in order. + routes []*Route + // Routes by name for URL building. + namedRoutes map[string]*Route + // See Router.StrictSlash(). This defines the flag for new routes. + strictSlash bool + // If true, do not clear the request context after handling the request + KeepContext bool +} + +// Match matches registered routes against the request. +func (r *Router) Match(req *http.Request, match *RouteMatch) bool { + for _, route := range r.routes { + if route.Match(req, match) { + return true + } + } + return false +} + +// ServeHTTP dispatches the handler registered in the matched route. +// +// When there is a match, the route variables can be retrieved calling +// mux.Vars(request). +func (r *Router) ServeHTTP(w http.ResponseWriter, req *http.Request) { + // Clean path to canonical form and redirect. + if p := cleanPath(req.URL.Path); p != req.URL.Path { + + // Added 3 lines (Philip Schlump) - It was droping the query string and #whatever from query. + // This matches with fix in go 1.2 r.c. 4 for same problem. Go Issue: + // http://code.google.com/p/go/issues/detail?id=5252 + url := *req.URL + url.Path = p + p = url.String() + + w.Header().Set("Location", p) + w.WriteHeader(http.StatusMovedPermanently) + return + } + var match RouteMatch + var handler http.Handler + if r.Match(req, &match) { + handler = match.Handler + setVars(req, match.Vars) + setCurrentRoute(req, match.Route) + } + if handler == nil { + handler = r.NotFoundHandler + if handler == nil { + handler = http.NotFoundHandler() + } + } + if !r.KeepContext { + defer context.Clear(req) + } + handler.ServeHTTP(w, req) +} + +// Get returns a route registered with the given name. +func (r *Router) Get(name string) *Route { + return r.getNamedRoutes()[name] +} + +// GetRoute returns a route registered with the given name. This method +// was renamed to Get() and remains here for backwards compatibility. +func (r *Router) GetRoute(name string) *Route { + return r.getNamedRoutes()[name] +} + +// StrictSlash defines the trailing slash behavior for new routes. The initial +// value is false. +// +// When true, if the route path is "/path/", accessing "/path" will redirect +// to the former and vice versa. In other words, your application will always +// see the path as specified in the route. +// +// When false, if the route path is "/path", accessing "/path/" will not match +// this route and vice versa. +// +// Special case: when a route sets a path prefix using the PathPrefix() method, +// strict slash is ignored for that route because the redirect behavior can't +// be determined from a prefix alone. However, any subrouters created from that +// route inherit the original StrictSlash setting. +func (r *Router) StrictSlash(value bool) *Router { + r.strictSlash = value + return r +} + +// ---------------------------------------------------------------------------- +// parentRoute +// ---------------------------------------------------------------------------- + +// getNamedRoutes returns the map where named routes are registered. +func (r *Router) getNamedRoutes() map[string]*Route { + if r.namedRoutes == nil { + if r.parent != nil { + r.namedRoutes = r.parent.getNamedRoutes() + } else { + r.namedRoutes = make(map[string]*Route) + } + } + return r.namedRoutes +} + +// getRegexpGroup returns regexp definitions from the parent route, if any. +func (r *Router) getRegexpGroup() *routeRegexpGroup { + if r.parent != nil { + return r.parent.getRegexpGroup() + } + return nil +} + +func (r *Router) buildVars(m map[string]string) map[string]string { + if r.parent != nil { + m = r.parent.buildVars(m) + } + return m +} + +// ---------------------------------------------------------------------------- +// Route factories +// ---------------------------------------------------------------------------- + +// NewRoute registers an empty route. +func (r *Router) NewRoute() *Route { + route := &Route{parent: r, strictSlash: r.strictSlash} + r.routes = append(r.routes, route) + return route +} + +// Handle registers a new route with a matcher for the URL path. +// See Route.Path() and Route.Handler(). +func (r *Router) Handle(path string, handler http.Handler) *Route { + return r.NewRoute().Path(path).Handler(handler) +} + +// HandleFunc registers a new route with a matcher for the URL path. +// See Route.Path() and Route.HandlerFunc(). +func (r *Router) HandleFunc(path string, f func(http.ResponseWriter, + *http.Request)) *Route { + return r.NewRoute().Path(path).HandlerFunc(f) +} + +// Headers registers a new route with a matcher for request header values. +// See Route.Headers(). +func (r *Router) Headers(pairs ...string) *Route { + return r.NewRoute().Headers(pairs...) +} + +// Host registers a new route with a matcher for the URL host. +// See Route.Host(). +func (r *Router) Host(tpl string) *Route { + return r.NewRoute().Host(tpl) +} + +// MatcherFunc registers a new route with a custom matcher function. +// See Route.MatcherFunc(). +func (r *Router) MatcherFunc(f MatcherFunc) *Route { + return r.NewRoute().MatcherFunc(f) +} + +// Methods registers a new route with a matcher for HTTP methods. +// See Route.Methods(). +func (r *Router) Methods(methods ...string) *Route { + return r.NewRoute().Methods(methods...) +} + +// Path registers a new route with a matcher for the URL path. +// See Route.Path(). +func (r *Router) Path(tpl string) *Route { + return r.NewRoute().Path(tpl) +} + +// PathPrefix registers a new route with a matcher for the URL path prefix. +// See Route.PathPrefix(). +func (r *Router) PathPrefix(tpl string) *Route { + return r.NewRoute().PathPrefix(tpl) +} + +// Queries registers a new route with a matcher for URL query values. +// See Route.Queries(). +func (r *Router) Queries(pairs ...string) *Route { + return r.NewRoute().Queries(pairs...) +} + +// Schemes registers a new route with a matcher for URL schemes. +// See Route.Schemes(). +func (r *Router) Schemes(schemes ...string) *Route { + return r.NewRoute().Schemes(schemes...) +} + +// BuildVars registers a new route with a custom function for modifying +// route variables before building a URL. +func (r *Router) BuildVarsFunc(f BuildVarsFunc) *Route { + return r.NewRoute().BuildVarsFunc(f) +} + +// Walk walks the router and all its sub-routers, calling walkFn for each route +// in the tree. The routes are walked in the order they were added. Sub-routers +// are explored depth-first. +func (r *Router) Walk(walkFn WalkFunc) error { + return r.walk(walkFn, []*Route{}) +} + +// SkipRouter is used as a return value from WalkFuncs to indicate that the +// router that walk is about to descend down to should be skipped. +var SkipRouter = errors.New("skip this router") + +// WalkFunc is the type of the function called for each route visited by Walk. +// At every invocation, it is given the current route, and the current router, +// and a list of ancestor routes that lead to the current route. +type WalkFunc func(route *Route, router *Router, ancestors []*Route) error + +func (r *Router) walk(walkFn WalkFunc, ancestors []*Route) error { + for _, t := range r.routes { + if t.regexp == nil || t.regexp.path == nil || t.regexp.path.template == "" { + continue + } + + err := walkFn(t, r, ancestors) + if err == SkipRouter { + continue + } + for _, sr := range t.matchers { + if h, ok := sr.(*Router); ok { + err := h.walk(walkFn, ancestors) + if err != nil { + return err + } + } + } + if h, ok := t.handler.(*Router); ok { + ancestors = append(ancestors, t) + err := h.walk(walkFn, ancestors) + if err != nil { + return err + } + ancestors = ancestors[:len(ancestors)-1] + } + } + return nil +} + +// ---------------------------------------------------------------------------- +// Context +// ---------------------------------------------------------------------------- + +// RouteMatch stores information about a matched route. +type RouteMatch struct { + Route *Route + Handler http.Handler + Vars map[string]string +} + +type contextKey int + +const ( + varsKey contextKey = iota + routeKey +) + +// Vars returns the route variables for the current request, if any. +func Vars(r *http.Request) map[string]string { + if rv := context.Get(r, varsKey); rv != nil { + return rv.(map[string]string) + } + return nil +} + +// CurrentRoute returns the matched route for the current request, if any. +// This only works when called inside the handler of the matched route +// because the matched route is stored in the request context which is cleared +// after the handler returns, unless the KeepContext option is set on the +// Router. +func CurrentRoute(r *http.Request) *Route { + if rv := context.Get(r, routeKey); rv != nil { + return rv.(*Route) + } + return nil +} + +func setVars(r *http.Request, val interface{}) { + context.Set(r, varsKey, val) +} + +func setCurrentRoute(r *http.Request, val interface{}) { + context.Set(r, routeKey, val) +} + +// ---------------------------------------------------------------------------- +// Helpers +// ---------------------------------------------------------------------------- + +// cleanPath returns the canonical path for p, eliminating . and .. elements. +// Borrowed from the net/http package. +func cleanPath(p string) string { + if p == "" { + return "/" + } + if p[0] != '/' { + p = "/" + p + } + np := path.Clean(p) + // path.Clean removes trailing slash except for root; + // put the trailing slash back if necessary. + if p[len(p)-1] == '/' && np != "/" { + np += "/" + } + return np +} + +// uniqueVars returns an error if two slices contain duplicated strings. +func uniqueVars(s1, s2 []string) error { + for _, v1 := range s1 { + for _, v2 := range s2 { + if v1 == v2 { + return fmt.Errorf("mux: duplicated route variable %q", v2) + } + } + } + return nil +} + +func checkPairs(pairs ...string) (int, error) { + length := len(pairs) + if length%2 != 0 { + return length, fmt.Errorf( + "mux: number of parameters must be multiple of 2, got %v", pairs) + } + return length, nil +} + +// mapFromPairs converts variadic string parameters to a string map. +func mapFromPairsToString(pairs ...string) (map[string]string, error) { + length, err := checkPairs(pairs...) + if err != nil { + return nil, err + } + m := make(map[string]string, length/2) + for i := 0; i < length; i += 2 { + m[pairs[i]] = pairs[i+1] + } + return m, nil +} + +func mapFromPairsToRegex(pairs ...string) (map[string]*regexp.Regexp, error) { + length, err := checkPairs(pairs...) + if err != nil { + return nil, err + } + m := make(map[string]*regexp.Regexp, length/2) + for i := 0; i < length; i += 2 { + regex, err := regexp.Compile(pairs[i+1]) + if err != nil { + return nil, err + } + m[pairs[i]] = regex + } + return m, nil +} + +// matchInArray returns true if the given string value is in the array. +func matchInArray(arr []string, value string) bool { + for _, v := range arr { + if v == value { + return true + } + } + return false +} + +// matchMapWithString returns true if the given key/value pairs exist in a given map. +func matchMapWithString(toCheck map[string]string, toMatch map[string][]string, canonicalKey bool) bool { + for k, v := range toCheck { + // Check if key exists. + if canonicalKey { + k = http.CanonicalHeaderKey(k) + } + if values := toMatch[k]; values == nil { + return false + } else if v != "" { + // If value was defined as an empty string we only check that the + // key exists. Otherwise we also check for equality. + valueExists := false + for _, value := range values { + if v == value { + valueExists = true + break + } + } + if !valueExists { + return false + } + } + } + return true +} + +// matchMapWithRegex returns true if the given key/value pairs exist in a given map compiled against +// the given regex +func matchMapWithRegex(toCheck map[string]*regexp.Regexp, toMatch map[string][]string, canonicalKey bool) bool { + for k, v := range toCheck { + // Check if key exists. + if canonicalKey { + k = http.CanonicalHeaderKey(k) + } + if values := toMatch[k]; values == nil { + return false + } else if v != nil { + // If value was defined as an empty string we only check that the + // key exists. Otherwise we also check for equality. + valueExists := false + for _, value := range values { + if v.MatchString(value) { + valueExists = true + break + } + } + if !valueExists { + return false + } + } + } + return true +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux_test.go new file mode 100644 index 0000000000000..74cb98b8329cc --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/mux_test.go @@ -0,0 +1,1334 @@ +// Copyright 2012 The Gorilla Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package mux + +import ( + "fmt" + "net/http" + "strings" + "testing" + + "github.com/fsouza/go-dockerclient/external/github.com/gorilla/context" +) + +func (r *Route) GoString() string { + matchers := make([]string, len(r.matchers)) + for i, m := range r.matchers { + matchers[i] = fmt.Sprintf("%#v", m) + } + return fmt.Sprintf("&Route{matchers:[]matcher{%s}}", strings.Join(matchers, ", ")) +} + +func (r *routeRegexp) GoString() string { + return fmt.Sprintf("&routeRegexp{template: %q, matchHost: %t, matchQuery: %t, strictSlash: %t, regexp: regexp.MustCompile(%q), reverse: %q, varsN: %v, varsR: %v", r.template, r.matchHost, r.matchQuery, r.strictSlash, r.regexp.String(), r.reverse, r.varsN, r.varsR) +} + +type routeTest struct { + title string // title of the test + route *Route // the route being tested + request *http.Request // a request to test the route + vars map[string]string // the expected vars of the match + host string // the expected host of the match + path string // the expected path of the match + shouldMatch bool // whether the request is expected to match the route at all + shouldRedirect bool // whether the request should result in a redirect +} + +func TestHost(t *testing.T) { + // newRequestHost a new request with a method, url, and host header + newRequestHost := func(method, url, host string) *http.Request { + req, err := http.NewRequest(method, url, nil) + if err != nil { + panic(err) + } + req.Host = host + return req + } + + tests := []routeTest{ + { + title: "Host route match", + route: new(Route).Host("aaa.bbb.ccc"), + request: newRequest("GET", "http://aaa.bbb.ccc/111/222/333"), + vars: map[string]string{}, + host: "aaa.bbb.ccc", + path: "", + shouldMatch: true, + }, + { + title: "Host route, wrong host in request URL", + route: new(Route).Host("aaa.bbb.ccc"), + request: newRequest("GET", "http://aaa.222.ccc/111/222/333"), + vars: map[string]string{}, + host: "aaa.bbb.ccc", + path: "", + shouldMatch: false, + }, + { + title: "Host route with port, match", + route: new(Route).Host("aaa.bbb.ccc:1234"), + request: newRequest("GET", "http://aaa.bbb.ccc:1234/111/222/333"), + vars: map[string]string{}, + host: "aaa.bbb.ccc:1234", + path: "", + shouldMatch: true, + }, + { + title: "Host route with port, wrong port in request URL", + route: new(Route).Host("aaa.bbb.ccc:1234"), + request: newRequest("GET", "http://aaa.bbb.ccc:9999/111/222/333"), + vars: map[string]string{}, + host: "aaa.bbb.ccc:1234", + path: "", + shouldMatch: false, + }, + { + title: "Host route, match with host in request header", + route: new(Route).Host("aaa.bbb.ccc"), + request: newRequestHost("GET", "/111/222/333", "aaa.bbb.ccc"), + vars: map[string]string{}, + host: "aaa.bbb.ccc", + path: "", + shouldMatch: true, + }, + { + title: "Host route, wrong host in request header", + route: new(Route).Host("aaa.bbb.ccc"), + request: newRequestHost("GET", "/111/222/333", "aaa.222.ccc"), + vars: map[string]string{}, + host: "aaa.bbb.ccc", + path: "", + shouldMatch: false, + }, + // BUG {new(Route).Host("aaa.bbb.ccc:1234"), newRequestHost("GET", "/111/222/333", "aaa.bbb.ccc:1234"), map[string]string{}, "aaa.bbb.ccc:1234", "", true}, + { + title: "Host route with port, wrong host in request header", + route: new(Route).Host("aaa.bbb.ccc:1234"), + request: newRequestHost("GET", "/111/222/333", "aaa.bbb.ccc:9999"), + vars: map[string]string{}, + host: "aaa.bbb.ccc:1234", + path: "", + shouldMatch: false, + }, + { + title: "Host route with pattern, match", + route: new(Route).Host("aaa.{v1:[a-z]{3}}.ccc"), + request: newRequest("GET", "http://aaa.bbb.ccc/111/222/333"), + vars: map[string]string{"v1": "bbb"}, + host: "aaa.bbb.ccc", + path: "", + shouldMatch: true, + }, + { + title: "Host route with pattern, additional capturing group, match", + route: new(Route).Host("aaa.{v1:[a-z]{2}(b|c)}.ccc"), + request: newRequest("GET", "http://aaa.bbb.ccc/111/222/333"), + vars: map[string]string{"v1": "bbb"}, + host: "aaa.bbb.ccc", + path: "", + shouldMatch: true, + }, + { + title: "Host route with pattern, wrong host in request URL", + route: new(Route).Host("aaa.{v1:[a-z]{3}}.ccc"), + request: newRequest("GET", "http://aaa.222.ccc/111/222/333"), + vars: map[string]string{"v1": "bbb"}, + host: "aaa.bbb.ccc", + path: "", + shouldMatch: false, + }, + { + title: "Host route with multiple patterns, match", + route: new(Route).Host("{v1:[a-z]{3}}.{v2:[a-z]{3}}.{v3:[a-z]{3}}"), + request: newRequest("GET", "http://aaa.bbb.ccc/111/222/333"), + vars: map[string]string{"v1": "aaa", "v2": "bbb", "v3": "ccc"}, + host: "aaa.bbb.ccc", + path: "", + shouldMatch: true, + }, + { + title: "Host route with multiple patterns, wrong host in request URL", + route: new(Route).Host("{v1:[a-z]{3}}.{v2:[a-z]{3}}.{v3:[a-z]{3}}"), + request: newRequest("GET", "http://aaa.222.ccc/111/222/333"), + vars: map[string]string{"v1": "aaa", "v2": "bbb", "v3": "ccc"}, + host: "aaa.bbb.ccc", + path: "", + shouldMatch: false, + }, + { + title: "Host route with hyphenated name and pattern, match", + route: new(Route).Host("aaa.{v-1:[a-z]{3}}.ccc"), + request: newRequest("GET", "http://aaa.bbb.ccc/111/222/333"), + vars: map[string]string{"v-1": "bbb"}, + host: "aaa.bbb.ccc", + path: "", + shouldMatch: true, + }, + { + title: "Host route with hyphenated name and pattern, additional capturing group, match", + route: new(Route).Host("aaa.{v-1:[a-z]{2}(b|c)}.ccc"), + request: newRequest("GET", "http://aaa.bbb.ccc/111/222/333"), + vars: map[string]string{"v-1": "bbb"}, + host: "aaa.bbb.ccc", + path: "", + shouldMatch: true, + }, + { + title: "Host route with multiple hyphenated names and patterns, match", + route: new(Route).Host("{v-1:[a-z]{3}}.{v-2:[a-z]{3}}.{v-3:[a-z]{3}}"), + request: newRequest("GET", "http://aaa.bbb.ccc/111/222/333"), + vars: map[string]string{"v-1": "aaa", "v-2": "bbb", "v-3": "ccc"}, + host: "aaa.bbb.ccc", + path: "", + shouldMatch: true, + }, + { + title: "Path route with single pattern with pipe, match", + route: new(Route).Path("/{category:a|b/c}"), + request: newRequest("GET", "http://localhost/a"), + vars: map[string]string{"category": "a"}, + host: "", + path: "/a", + shouldMatch: true, + }, + { + title: "Path route with single pattern with pipe, match", + route: new(Route).Path("/{category:a|b/c}"), + request: newRequest("GET", "http://localhost/b/c"), + vars: map[string]string{"category": "b/c"}, + host: "", + path: "/b/c", + shouldMatch: true, + }, + { + title: "Path route with multiple patterns with pipe, match", + route: new(Route).Path("/{category:a|b/c}/{product}/{id:[0-9]+}"), + request: newRequest("GET", "http://localhost/a/product_name/1"), + vars: map[string]string{"category": "a", "product": "product_name", "id": "1"}, + host: "", + path: "/a/product_name/1", + shouldMatch: true, + }, + { + title: "Path route with multiple patterns with pipe, match", + route: new(Route).Path("/{category:a|b/c}/{product}/{id:[0-9]+}"), + request: newRequest("GET", "http://localhost/b/c/product_name/1"), + vars: map[string]string{"category": "b/c", "product": "product_name", "id": "1"}, + host: "", + path: "/b/c/product_name/1", + shouldMatch: true, + }, + } + for _, test := range tests { + testRoute(t, test) + } +} + +func TestPath(t *testing.T) { + tests := []routeTest{ + { + title: "Path route, match", + route: new(Route).Path("/111/222/333"), + request: newRequest("GET", "http://localhost/111/222/333"), + vars: map[string]string{}, + host: "", + path: "/111/222/333", + shouldMatch: true, + }, + { + title: "Path route, match with trailing slash in request and path", + route: new(Route).Path("/111/"), + request: newRequest("GET", "http://localhost/111/"), + vars: map[string]string{}, + host: "", + path: "/111/", + shouldMatch: true, + }, + { + title: "Path route, do not match with trailing slash in path", + route: new(Route).Path("/111/"), + request: newRequest("GET", "http://localhost/111"), + vars: map[string]string{}, + host: "", + path: "/111", + shouldMatch: false, + }, + { + title: "Path route, do not match with trailing slash in request", + route: new(Route).Path("/111"), + request: newRequest("GET", "http://localhost/111/"), + vars: map[string]string{}, + host: "", + path: "/111/", + shouldMatch: false, + }, + { + title: "Path route, wrong path in request in request URL", + route: new(Route).Path("/111/222/333"), + request: newRequest("GET", "http://localhost/1/2/3"), + vars: map[string]string{}, + host: "", + path: "/111/222/333", + shouldMatch: false, + }, + { + title: "Path route with pattern, match", + route: new(Route).Path("/111/{v1:[0-9]{3}}/333"), + request: newRequest("GET", "http://localhost/111/222/333"), + vars: map[string]string{"v1": "222"}, + host: "", + path: "/111/222/333", + shouldMatch: true, + }, + { + title: "Path route with pattern, URL in request does not match", + route: new(Route).Path("/111/{v1:[0-9]{3}}/333"), + request: newRequest("GET", "http://localhost/111/aaa/333"), + vars: map[string]string{"v1": "222"}, + host: "", + path: "/111/222/333", + shouldMatch: false, + }, + { + title: "Path route with multiple patterns, match", + route: new(Route).Path("/{v1:[0-9]{3}}/{v2:[0-9]{3}}/{v3:[0-9]{3}}"), + request: newRequest("GET", "http://localhost/111/222/333"), + vars: map[string]string{"v1": "111", "v2": "222", "v3": "333"}, + host: "", + path: "/111/222/333", + shouldMatch: true, + }, + { + title: "Path route with multiple patterns, URL in request does not match", + route: new(Route).Path("/{v1:[0-9]{3}}/{v2:[0-9]{3}}/{v3:[0-9]{3}}"), + request: newRequest("GET", "http://localhost/111/aaa/333"), + vars: map[string]string{"v1": "111", "v2": "222", "v3": "333"}, + host: "", + path: "/111/222/333", + shouldMatch: false, + }, + { + title: "Path route with multiple patterns with pipe, match", + route: new(Route).Path("/{category:a|(b/c)}/{product}/{id:[0-9]+}"), + request: newRequest("GET", "http://localhost/a/product_name/1"), + vars: map[string]string{"category": "a", "product": "product_name", "id": "1"}, + host: "", + path: "/a/product_name/1", + shouldMatch: true, + }, + { + title: "Path route with hyphenated name and pattern, match", + route: new(Route).Path("/111/{v-1:[0-9]{3}}/333"), + request: newRequest("GET", "http://localhost/111/222/333"), + vars: map[string]string{"v-1": "222"}, + host: "", + path: "/111/222/333", + shouldMatch: true, + }, + { + title: "Path route with multiple hyphenated names and patterns, match", + route: new(Route).Path("/{v-1:[0-9]{3}}/{v-2:[0-9]{3}}/{v-3:[0-9]{3}}"), + request: newRequest("GET", "http://localhost/111/222/333"), + vars: map[string]string{"v-1": "111", "v-2": "222", "v-3": "333"}, + host: "", + path: "/111/222/333", + shouldMatch: true, + }, + { + title: "Path route with multiple hyphenated names and patterns with pipe, match", + route: new(Route).Path("/{product-category:a|(b/c)}/{product-name}/{product-id:[0-9]+}"), + request: newRequest("GET", "http://localhost/a/product_name/1"), + vars: map[string]string{"product-category": "a", "product-name": "product_name", "product-id": "1"}, + host: "", + path: "/a/product_name/1", + shouldMatch: true, + }, + } + + for _, test := range tests { + testRoute(t, test) + } +} + +func TestPathPrefix(t *testing.T) { + tests := []routeTest{ + { + title: "PathPrefix route, match", + route: new(Route).PathPrefix("/111"), + request: newRequest("GET", "http://localhost/111/222/333"), + vars: map[string]string{}, + host: "", + path: "/111", + shouldMatch: true, + }, + { + title: "PathPrefix route, match substring", + route: new(Route).PathPrefix("/1"), + request: newRequest("GET", "http://localhost/111/222/333"), + vars: map[string]string{}, + host: "", + path: "/1", + shouldMatch: true, + }, + { + title: "PathPrefix route, URL prefix in request does not match", + route: new(Route).PathPrefix("/111"), + request: newRequest("GET", "http://localhost/1/2/3"), + vars: map[string]string{}, + host: "", + path: "/111", + shouldMatch: false, + }, + { + title: "PathPrefix route with pattern, match", + route: new(Route).PathPrefix("/111/{v1:[0-9]{3}}"), + request: newRequest("GET", "http://localhost/111/222/333"), + vars: map[string]string{"v1": "222"}, + host: "", + path: "/111/222", + shouldMatch: true, + }, + { + title: "PathPrefix route with pattern, URL prefix in request does not match", + route: new(Route).PathPrefix("/111/{v1:[0-9]{3}}"), + request: newRequest("GET", "http://localhost/111/aaa/333"), + vars: map[string]string{"v1": "222"}, + host: "", + path: "/111/222", + shouldMatch: false, + }, + { + title: "PathPrefix route with multiple patterns, match", + route: new(Route).PathPrefix("/{v1:[0-9]{3}}/{v2:[0-9]{3}}"), + request: newRequest("GET", "http://localhost/111/222/333"), + vars: map[string]string{"v1": "111", "v2": "222"}, + host: "", + path: "/111/222", + shouldMatch: true, + }, + { + title: "PathPrefix route with multiple patterns, URL prefix in request does not match", + route: new(Route).PathPrefix("/{v1:[0-9]{3}}/{v2:[0-9]{3}}"), + request: newRequest("GET", "http://localhost/111/aaa/333"), + vars: map[string]string{"v1": "111", "v2": "222"}, + host: "", + path: "/111/222", + shouldMatch: false, + }, + } + + for _, test := range tests { + testRoute(t, test) + } +} + +func TestHostPath(t *testing.T) { + tests := []routeTest{ + { + title: "Host and Path route, match", + route: new(Route).Host("aaa.bbb.ccc").Path("/111/222/333"), + request: newRequest("GET", "http://aaa.bbb.ccc/111/222/333"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Host and Path route, wrong host in request URL", + route: new(Route).Host("aaa.bbb.ccc").Path("/111/222/333"), + request: newRequest("GET", "http://aaa.222.ccc/111/222/333"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: false, + }, + { + title: "Host and Path route with pattern, match", + route: new(Route).Host("aaa.{v1:[a-z]{3}}.ccc").Path("/111/{v2:[0-9]{3}}/333"), + request: newRequest("GET", "http://aaa.bbb.ccc/111/222/333"), + vars: map[string]string{"v1": "bbb", "v2": "222"}, + host: "aaa.bbb.ccc", + path: "/111/222/333", + shouldMatch: true, + }, + { + title: "Host and Path route with pattern, URL in request does not match", + route: new(Route).Host("aaa.{v1:[a-z]{3}}.ccc").Path("/111/{v2:[0-9]{3}}/333"), + request: newRequest("GET", "http://aaa.222.ccc/111/222/333"), + vars: map[string]string{"v1": "bbb", "v2": "222"}, + host: "aaa.bbb.ccc", + path: "/111/222/333", + shouldMatch: false, + }, + { + title: "Host and Path route with multiple patterns, match", + route: new(Route).Host("{v1:[a-z]{3}}.{v2:[a-z]{3}}.{v3:[a-z]{3}}").Path("/{v4:[0-9]{3}}/{v5:[0-9]{3}}/{v6:[0-9]{3}}"), + request: newRequest("GET", "http://aaa.bbb.ccc/111/222/333"), + vars: map[string]string{"v1": "aaa", "v2": "bbb", "v3": "ccc", "v4": "111", "v5": "222", "v6": "333"}, + host: "aaa.bbb.ccc", + path: "/111/222/333", + shouldMatch: true, + }, + { + title: "Host and Path route with multiple patterns, URL in request does not match", + route: new(Route).Host("{v1:[a-z]{3}}.{v2:[a-z]{3}}.{v3:[a-z]{3}}").Path("/{v4:[0-9]{3}}/{v5:[0-9]{3}}/{v6:[0-9]{3}}"), + request: newRequest("GET", "http://aaa.222.ccc/111/222/333"), + vars: map[string]string{"v1": "aaa", "v2": "bbb", "v3": "ccc", "v4": "111", "v5": "222", "v6": "333"}, + host: "aaa.bbb.ccc", + path: "/111/222/333", + shouldMatch: false, + }, + } + + for _, test := range tests { + testRoute(t, test) + } +} + +func TestHeaders(t *testing.T) { + // newRequestHeaders creates a new request with a method, url, and headers + newRequestHeaders := func(method, url string, headers map[string]string) *http.Request { + req, err := http.NewRequest(method, url, nil) + if err != nil { + panic(err) + } + for k, v := range headers { + req.Header.Add(k, v) + } + return req + } + + tests := []routeTest{ + { + title: "Headers route, match", + route: new(Route).Headers("foo", "bar", "baz", "ding"), + request: newRequestHeaders("GET", "http://localhost", map[string]string{"foo": "bar", "baz": "ding"}), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Headers route, bad header values", + route: new(Route).Headers("foo", "bar", "baz", "ding"), + request: newRequestHeaders("GET", "http://localhost", map[string]string{"foo": "bar", "baz": "dong"}), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: false, + }, + { + title: "Headers route, regex header values to match", + route: new(Route).Headers("foo", "ba[zr]"), + request: newRequestHeaders("GET", "http://localhost", map[string]string{"foo": "bar"}), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: false, + }, + { + title: "Headers route, regex header values to match", + route: new(Route).HeadersRegexp("foo", "ba[zr]"), + request: newRequestHeaders("GET", "http://localhost", map[string]string{"foo": "baz"}), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: true, + }, + } + + for _, test := range tests { + testRoute(t, test) + } + +} + +func TestMethods(t *testing.T) { + tests := []routeTest{ + { + title: "Methods route, match GET", + route: new(Route).Methods("GET", "POST"), + request: newRequest("GET", "http://localhost"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Methods route, match POST", + route: new(Route).Methods("GET", "POST"), + request: newRequest("POST", "http://localhost"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Methods route, bad method", + route: new(Route).Methods("GET", "POST"), + request: newRequest("PUT", "http://localhost"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: false, + }, + } + + for _, test := range tests { + testRoute(t, test) + } +} + +func TestQueries(t *testing.T) { + tests := []routeTest{ + { + title: "Queries route, match", + route: new(Route).Queries("foo", "bar", "baz", "ding"), + request: newRequest("GET", "http://localhost?foo=bar&baz=ding"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route, match with a query string", + route: new(Route).Host("www.example.com").Path("/api").Queries("foo", "bar", "baz", "ding"), + request: newRequest("GET", "http://www.example.com/api?foo=bar&baz=ding"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route, match with a query string out of order", + route: new(Route).Host("www.example.com").Path("/api").Queries("foo", "bar", "baz", "ding"), + request: newRequest("GET", "http://www.example.com/api?baz=ding&foo=bar"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route, bad query", + route: new(Route).Queries("foo", "bar", "baz", "ding"), + request: newRequest("GET", "http://localhost?foo=bar&baz=dong"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: false, + }, + { + title: "Queries route with pattern, match", + route: new(Route).Queries("foo", "{v1}"), + request: newRequest("GET", "http://localhost?foo=bar"), + vars: map[string]string{"v1": "bar"}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route with multiple patterns, match", + route: new(Route).Queries("foo", "{v1}", "baz", "{v2}"), + request: newRequest("GET", "http://localhost?foo=bar&baz=ding"), + vars: map[string]string{"v1": "bar", "v2": "ding"}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route with regexp pattern, match", + route: new(Route).Queries("foo", "{v1:[0-9]+}"), + request: newRequest("GET", "http://localhost?foo=10"), + vars: map[string]string{"v1": "10"}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route with regexp pattern, regexp does not match", + route: new(Route).Queries("foo", "{v1:[0-9]+}"), + request: newRequest("GET", "http://localhost?foo=a"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: false, + }, + { + title: "Queries route with regexp pattern with quantifier, match", + route: new(Route).Queries("foo", "{v1:[0-9]{1}}"), + request: newRequest("GET", "http://localhost?foo=1"), + vars: map[string]string{"v1": "1"}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route with regexp pattern with quantifier, additional variable in query string, match", + route: new(Route).Queries("foo", "{v1:[0-9]{1}}"), + request: newRequest("GET", "http://localhost?bar=2&foo=1"), + vars: map[string]string{"v1": "1"}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route with regexp pattern with quantifier, regexp does not match", + route: new(Route).Queries("foo", "{v1:[0-9]{1}}"), + request: newRequest("GET", "http://localhost?foo=12"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: false, + }, + { + title: "Queries route with regexp pattern with quantifier, additional capturing group", + route: new(Route).Queries("foo", "{v1:[0-9]{1}(a|b)}"), + request: newRequest("GET", "http://localhost?foo=1a"), + vars: map[string]string{"v1": "1a"}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route with regexp pattern with quantifier, additional variable in query string, regexp does not match", + route: new(Route).Queries("foo", "{v1:[0-9]{1}}"), + request: newRequest("GET", "http://localhost?foo=12"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: false, + }, + { + title: "Queries route with hyphenated name, match", + route: new(Route).Queries("foo", "{v-1}"), + request: newRequest("GET", "http://localhost?foo=bar"), + vars: map[string]string{"v-1": "bar"}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route with multiple hyphenated names, match", + route: new(Route).Queries("foo", "{v-1}", "baz", "{v-2}"), + request: newRequest("GET", "http://localhost?foo=bar&baz=ding"), + vars: map[string]string{"v-1": "bar", "v-2": "ding"}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route with hyphenate name and pattern, match", + route: new(Route).Queries("foo", "{v-1:[0-9]+}"), + request: newRequest("GET", "http://localhost?foo=10"), + vars: map[string]string{"v-1": "10"}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route with hyphenated name and pattern with quantifier, additional capturing group", + route: new(Route).Queries("foo", "{v-1:[0-9]{1}(a|b)}"), + request: newRequest("GET", "http://localhost?foo=1a"), + vars: map[string]string{"v-1": "1a"}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route with empty value, should match", + route: new(Route).Queries("foo", ""), + request: newRequest("GET", "http://localhost?foo=bar"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route with empty value and no parameter in request, should not match", + route: new(Route).Queries("foo", ""), + request: newRequest("GET", "http://localhost"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: false, + }, + { + title: "Queries route with empty value and empty parameter in request, should match", + route: new(Route).Queries("foo", ""), + request: newRequest("GET", "http://localhost?foo="), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route with overlapping value, should not match", + route: new(Route).Queries("foo", "bar"), + request: newRequest("GET", "http://localhost?foo=barfoo"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: false, + }, + { + title: "Queries route with no parameter in request, should not match", + route: new(Route).Queries("foo", "{bar}"), + request: newRequest("GET", "http://localhost"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: false, + }, + { + title: "Queries route with empty parameter in request, should match", + route: new(Route).Queries("foo", "{bar}"), + request: newRequest("GET", "http://localhost?foo="), + vars: map[string]string{"foo": ""}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Queries route, bad submatch", + route: new(Route).Queries("foo", "bar", "baz", "ding"), + request: newRequest("GET", "http://localhost?fffoo=bar&baz=dingggg"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: false, + }, + } + + for _, test := range tests { + testRoute(t, test) + } +} + +func TestSchemes(t *testing.T) { + tests := []routeTest{ + // Schemes + { + title: "Schemes route, match https", + route: new(Route).Schemes("https", "ftp"), + request: newRequest("GET", "https://localhost"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Schemes route, match ftp", + route: new(Route).Schemes("https", "ftp"), + request: newRequest("GET", "ftp://localhost"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "Schemes route, bad scheme", + route: new(Route).Schemes("https", "ftp"), + request: newRequest("GET", "http://localhost"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: false, + }, + } + for _, test := range tests { + testRoute(t, test) + } +} + +func TestMatcherFunc(t *testing.T) { + m := func(r *http.Request, m *RouteMatch) bool { + if r.URL.Host == "aaa.bbb.ccc" { + return true + } + return false + } + + tests := []routeTest{ + { + title: "MatchFunc route, match", + route: new(Route).MatcherFunc(m), + request: newRequest("GET", "http://aaa.bbb.ccc"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: true, + }, + { + title: "MatchFunc route, non-match", + route: new(Route).MatcherFunc(m), + request: newRequest("GET", "http://aaa.222.ccc"), + vars: map[string]string{}, + host: "", + path: "", + shouldMatch: false, + }, + } + + for _, test := range tests { + testRoute(t, test) + } +} + +func TestBuildVarsFunc(t *testing.T) { + tests := []routeTest{ + { + title: "BuildVarsFunc set on route", + route: new(Route).Path(`/111/{v1:\d}{v2:.*}`).BuildVarsFunc(func(vars map[string]string) map[string]string { + vars["v1"] = "3" + vars["v2"] = "a" + return vars + }), + request: newRequest("GET", "http://localhost/111/2"), + path: "/111/3a", + shouldMatch: true, + }, + { + title: "BuildVarsFunc set on route and parent route", + route: new(Route).PathPrefix(`/{v1:\d}`).BuildVarsFunc(func(vars map[string]string) map[string]string { + vars["v1"] = "2" + return vars + }).Subrouter().Path(`/{v2:\w}`).BuildVarsFunc(func(vars map[string]string) map[string]string { + vars["v2"] = "b" + return vars + }), + request: newRequest("GET", "http://localhost/1/a"), + path: "/2/b", + shouldMatch: true, + }, + } + + for _, test := range tests { + testRoute(t, test) + } +} + +func TestSubRouter(t *testing.T) { + subrouter1 := new(Route).Host("{v1:[a-z]+}.google.com").Subrouter() + subrouter2 := new(Route).PathPrefix("/foo/{v1}").Subrouter() + + tests := []routeTest{ + { + route: subrouter1.Path("/{v2:[a-z]+}"), + request: newRequest("GET", "http://aaa.google.com/bbb"), + vars: map[string]string{"v1": "aaa", "v2": "bbb"}, + host: "aaa.google.com", + path: "/bbb", + shouldMatch: true, + }, + { + route: subrouter1.Path("/{v2:[a-z]+}"), + request: newRequest("GET", "http://111.google.com/111"), + vars: map[string]string{"v1": "aaa", "v2": "bbb"}, + host: "aaa.google.com", + path: "/bbb", + shouldMatch: false, + }, + { + route: subrouter2.Path("/baz/{v2}"), + request: newRequest("GET", "http://localhost/foo/bar/baz/ding"), + vars: map[string]string{"v1": "bar", "v2": "ding"}, + host: "", + path: "/foo/bar/baz/ding", + shouldMatch: true, + }, + { + route: subrouter2.Path("/baz/{v2}"), + request: newRequest("GET", "http://localhost/foo/bar"), + vars: map[string]string{"v1": "bar", "v2": "ding"}, + host: "", + path: "/foo/bar/baz/ding", + shouldMatch: false, + }, + } + + for _, test := range tests { + testRoute(t, test) + } +} + +func TestNamedRoutes(t *testing.T) { + r1 := NewRouter() + r1.NewRoute().Name("a") + r1.NewRoute().Name("b") + r1.NewRoute().Name("c") + + r2 := r1.NewRoute().Subrouter() + r2.NewRoute().Name("d") + r2.NewRoute().Name("e") + r2.NewRoute().Name("f") + + r3 := r2.NewRoute().Subrouter() + r3.NewRoute().Name("g") + r3.NewRoute().Name("h") + r3.NewRoute().Name("i") + + if r1.namedRoutes == nil || len(r1.namedRoutes) != 9 { + t.Errorf("Expected 9 named routes, got %v", r1.namedRoutes) + } else if r1.Get("i") == nil { + t.Errorf("Subroute name not registered") + } +} + +func TestStrictSlash(t *testing.T) { + r := NewRouter() + r.StrictSlash(true) + + tests := []routeTest{ + { + title: "Redirect path without slash", + route: r.NewRoute().Path("/111/"), + request: newRequest("GET", "http://localhost/111"), + vars: map[string]string{}, + host: "", + path: "/111/", + shouldMatch: true, + shouldRedirect: true, + }, + { + title: "Do not redirect path with slash", + route: r.NewRoute().Path("/111/"), + request: newRequest("GET", "http://localhost/111/"), + vars: map[string]string{}, + host: "", + path: "/111/", + shouldMatch: true, + shouldRedirect: false, + }, + { + title: "Redirect path with slash", + route: r.NewRoute().Path("/111"), + request: newRequest("GET", "http://localhost/111/"), + vars: map[string]string{}, + host: "", + path: "/111", + shouldMatch: true, + shouldRedirect: true, + }, + { + title: "Do not redirect path without slash", + route: r.NewRoute().Path("/111"), + request: newRequest("GET", "http://localhost/111"), + vars: map[string]string{}, + host: "", + path: "/111", + shouldMatch: true, + shouldRedirect: false, + }, + { + title: "Propagate StrictSlash to subrouters", + route: r.NewRoute().PathPrefix("/static/").Subrouter().Path("/images/"), + request: newRequest("GET", "http://localhost/static/images"), + vars: map[string]string{}, + host: "", + path: "/static/images/", + shouldMatch: true, + shouldRedirect: true, + }, + { + title: "Ignore StrictSlash for path prefix", + route: r.NewRoute().PathPrefix("/static/"), + request: newRequest("GET", "http://localhost/static/logo.png"), + vars: map[string]string{}, + host: "", + path: "/static/", + shouldMatch: true, + shouldRedirect: false, + }, + } + + for _, test := range tests { + testRoute(t, test) + } +} + +func TestWalkSingleDepth(t *testing.T) { + r0 := NewRouter() + r1 := NewRouter() + r2 := NewRouter() + + r0.Path("/g") + r0.Path("/o") + r0.Path("/d").Handler(r1) + r0.Path("/r").Handler(r2) + r0.Path("/a") + + r1.Path("/z") + r1.Path("/i") + r1.Path("/l") + r1.Path("/l") + + r2.Path("/i") + r2.Path("/l") + r2.Path("/l") + + paths := []string{"g", "o", "r", "i", "l", "l", "a"} + depths := []int{0, 0, 0, 1, 1, 1, 0} + i := 0 + err := r0.Walk(func(route *Route, router *Router, ancestors []*Route) error { + matcher := route.matchers[0].(*routeRegexp) + if matcher.template == "/d" { + return SkipRouter + } + if len(ancestors) != depths[i] { + t.Errorf(`Expected depth of %d at i = %d; got "%s"`, depths[i], i, len(ancestors)) + } + if matcher.template != "/"+paths[i] { + t.Errorf(`Expected "/%s" at i = %d; got "%s"`, paths[i], i, matcher.template) + } + i++ + return nil + }) + if err != nil { + panic(err) + } + if i != len(paths) { + t.Errorf("Expected %d routes, found %d", len(paths), i) + } +} + +func TestWalkNested(t *testing.T) { + router := NewRouter() + + g := router.Path("/g").Subrouter() + o := g.PathPrefix("/o").Subrouter() + r := o.PathPrefix("/r").Subrouter() + i := r.PathPrefix("/i").Subrouter() + l1 := i.PathPrefix("/l").Subrouter() + l2 := l1.PathPrefix("/l").Subrouter() + l2.Path("/a") + + paths := []string{"/g", "/g/o", "/g/o/r", "/g/o/r/i", "/g/o/r/i/l", "/g/o/r/i/l/l", "/g/o/r/i/l/l/a"} + idx := 0 + err := router.Walk(func(route *Route, router *Router, ancestors []*Route) error { + path := paths[idx] + tpl := route.regexp.path.template + if tpl != path { + t.Errorf(`Expected %s got %s`, path, tpl) + } + idx++ + return nil + }) + if err != nil { + panic(err) + } + if idx != len(paths) { + t.Errorf("Expected %d routes, found %d", len(paths), idx) + } +} + +// ---------------------------------------------------------------------------- +// Helpers +// ---------------------------------------------------------------------------- + +func getRouteTemplate(route *Route) string { + host, path := "none", "none" + if route.regexp != nil { + if route.regexp.host != nil { + host = route.regexp.host.template + } + if route.regexp.path != nil { + path = route.regexp.path.template + } + } + return fmt.Sprintf("Host: %v, Path: %v", host, path) +} + +func testRoute(t *testing.T, test routeTest) { + request := test.request + route := test.route + vars := test.vars + shouldMatch := test.shouldMatch + host := test.host + path := test.path + url := test.host + test.path + shouldRedirect := test.shouldRedirect + + var match RouteMatch + ok := route.Match(request, &match) + if ok != shouldMatch { + msg := "Should match" + if !shouldMatch { + msg = "Should not match" + } + t.Errorf("(%v) %v:\nRoute: %#v\nRequest: %#v\nVars: %v\n", test.title, msg, route, request, vars) + return + } + if shouldMatch { + if test.vars != nil && !stringMapEqual(test.vars, match.Vars) { + t.Errorf("(%v) Vars not equal: expected %v, got %v", test.title, vars, match.Vars) + return + } + if host != "" { + u, _ := test.route.URLHost(mapToPairs(match.Vars)...) + if host != u.Host { + t.Errorf("(%v) URLHost not equal: expected %v, got %v -- %v", test.title, host, u.Host, getRouteTemplate(route)) + return + } + } + if path != "" { + u, _ := route.URLPath(mapToPairs(match.Vars)...) + if path != u.Path { + t.Errorf("(%v) URLPath not equal: expected %v, got %v -- %v", test.title, path, u.Path, getRouteTemplate(route)) + return + } + } + if url != "" { + u, _ := route.URL(mapToPairs(match.Vars)...) + if url != u.Host+u.Path { + t.Errorf("(%v) URL not equal: expected %v, got %v -- %v", test.title, url, u.Host+u.Path, getRouteTemplate(route)) + return + } + } + if shouldRedirect && match.Handler == nil { + t.Errorf("(%v) Did not redirect", test.title) + return + } + if !shouldRedirect && match.Handler != nil { + t.Errorf("(%v) Unexpected redirect", test.title) + return + } + } +} + +// Tests that the context is cleared or not cleared properly depending on +// the configuration of the router +func TestKeepContext(t *testing.T) { + func1 := func(w http.ResponseWriter, r *http.Request) {} + + r := NewRouter() + r.HandleFunc("/", func1).Name("func1") + + req, _ := http.NewRequest("GET", "http://localhost/", nil) + context.Set(req, "t", 1) + + res := new(http.ResponseWriter) + r.ServeHTTP(*res, req) + + if _, ok := context.GetOk(req, "t"); ok { + t.Error("Context should have been cleared at end of request") + } + + r.KeepContext = true + + req, _ = http.NewRequest("GET", "http://localhost/", nil) + context.Set(req, "t", 1) + + r.ServeHTTP(*res, req) + if _, ok := context.GetOk(req, "t"); !ok { + t.Error("Context should NOT have been cleared at end of request") + } + +} + +type TestA301ResponseWriter struct { + hh http.Header + status int +} + +func (ho TestA301ResponseWriter) Header() http.Header { + return http.Header(ho.hh) +} + +func (ho TestA301ResponseWriter) Write(b []byte) (int, error) { + return 0, nil +} + +func (ho TestA301ResponseWriter) WriteHeader(code int) { + ho.status = code +} + +func Test301Redirect(t *testing.T) { + m := make(http.Header) + + func1 := func(w http.ResponseWriter, r *http.Request) {} + func2 := func(w http.ResponseWriter, r *http.Request) {} + + r := NewRouter() + r.HandleFunc("/api/", func2).Name("func2") + r.HandleFunc("/", func1).Name("func1") + + req, _ := http.NewRequest("GET", "http://localhost//api/?abc=def", nil) + + res := TestA301ResponseWriter{ + hh: m, + status: 0, + } + r.ServeHTTP(&res, req) + + if "http://localhost/api/?abc=def" != res.hh["Location"][0] { + t.Errorf("Should have complete URL with query string") + } +} + +// https://plus.google.com/101022900381697718949/posts/eWy6DjFJ6uW +func TestSubrouterHeader(t *testing.T) { + expected := "func1 response" + func1 := func(w http.ResponseWriter, r *http.Request) { + fmt.Fprint(w, expected) + } + func2 := func(http.ResponseWriter, *http.Request) {} + + r := NewRouter() + s := r.Headers("SomeSpecialHeader", "").Subrouter() + s.HandleFunc("/", func1).Name("func1") + r.HandleFunc("/", func2).Name("func2") + + req, _ := http.NewRequest("GET", "http://localhost/", nil) + req.Header.Add("SomeSpecialHeader", "foo") + match := new(RouteMatch) + matched := r.Match(req, match) + if !matched { + t.Errorf("Should match request") + } + if match.Route.GetName() != "func1" { + t.Errorf("Expecting func1 handler, got %s", match.Route.GetName()) + } + resp := NewRecorder() + match.Handler.ServeHTTP(resp, req) + if resp.Body.String() != expected { + t.Errorf("Expecting %q", expected) + } +} + +// mapToPairs converts a string map to a slice of string pairs +func mapToPairs(m map[string]string) []string { + var i int + p := make([]string, len(m)*2) + for k, v := range m { + p[i] = k + p[i+1] = v + i += 2 + } + return p +} + +// stringMapEqual checks the equality of two string maps +func stringMapEqual(m1, m2 map[string]string) bool { + nil1 := m1 == nil + nil2 := m2 == nil + if nil1 != nil2 || len(m1) != len(m2) { + return false + } + for k, v := range m1 { + if v != m2[k] { + return false + } + } + return true +} + +// newRequest is a helper function to create a new request with a method and url +func newRequest(method, url string) *http.Request { + req, err := http.NewRequest(method, url, nil) + if err != nil { + panic(err) + } + return req +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/old_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/old_test.go new file mode 100644 index 0000000000000..755db483e8f38 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/old_test.go @@ -0,0 +1,714 @@ +// Old tests ported to Go1. This is a mess. Want to drop it one day. + +// Copyright 2011 Gorilla Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package mux + +import ( + "bytes" + "net/http" + "testing" +) + +// ---------------------------------------------------------------------------- +// ResponseRecorder +// ---------------------------------------------------------------------------- +// Copyright 2009 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// ResponseRecorder is an implementation of http.ResponseWriter that +// records its mutations for later inspection in tests. +type ResponseRecorder struct { + Code int // the HTTP response code from WriteHeader + HeaderMap http.Header // the HTTP response headers + Body *bytes.Buffer // if non-nil, the bytes.Buffer to append written data to + Flushed bool +} + +// NewRecorder returns an initialized ResponseRecorder. +func NewRecorder() *ResponseRecorder { + return &ResponseRecorder{ + HeaderMap: make(http.Header), + Body: new(bytes.Buffer), + } +} + +// DefaultRemoteAddr is the default remote address to return in RemoteAddr if +// an explicit DefaultRemoteAddr isn't set on ResponseRecorder. +const DefaultRemoteAddr = "1.2.3.4" + +// Header returns the response headers. +func (rw *ResponseRecorder) Header() http.Header { + return rw.HeaderMap +} + +// Write always succeeds and writes to rw.Body, if not nil. +func (rw *ResponseRecorder) Write(buf []byte) (int, error) { + if rw.Body != nil { + rw.Body.Write(buf) + } + if rw.Code == 0 { + rw.Code = http.StatusOK + } + return len(buf), nil +} + +// WriteHeader sets rw.Code. +func (rw *ResponseRecorder) WriteHeader(code int) { + rw.Code = code +} + +// Flush sets rw.Flushed to true. +func (rw *ResponseRecorder) Flush() { + rw.Flushed = true +} + +// ---------------------------------------------------------------------------- + +func TestRouteMatchers(t *testing.T) { + var scheme, host, path, query, method string + var headers map[string]string + var resultVars map[bool]map[string]string + + router := NewRouter() + router.NewRoute().Host("{var1}.google.com"). + Path("/{var2:[a-z]+}/{var3:[0-9]+}"). + Queries("foo", "bar"). + Methods("GET"). + Schemes("https"). + Headers("x-requested-with", "XMLHttpRequest") + router.NewRoute().Host("www.{var4}.com"). + PathPrefix("/foo/{var5:[a-z]+}/{var6:[0-9]+}"). + Queries("baz", "ding"). + Methods("POST"). + Schemes("http"). + Headers("Content-Type", "application/json") + + reset := func() { + // Everything match. + scheme = "https" + host = "www.google.com" + path = "/product/42" + query = "?foo=bar" + method = "GET" + headers = map[string]string{"X-Requested-With": "XMLHttpRequest"} + resultVars = map[bool]map[string]string{ + true: {"var1": "www", "var2": "product", "var3": "42"}, + false: {}, + } + } + + reset2 := func() { + // Everything match. + scheme = "http" + host = "www.google.com" + path = "/foo/product/42/path/that/is/ignored" + query = "?baz=ding" + method = "POST" + headers = map[string]string{"Content-Type": "application/json"} + resultVars = map[bool]map[string]string{ + true: {"var4": "google", "var5": "product", "var6": "42"}, + false: {}, + } + } + + match := func(shouldMatch bool) { + url := scheme + "://" + host + path + query + request, _ := http.NewRequest(method, url, nil) + for key, value := range headers { + request.Header.Add(key, value) + } + + var routeMatch RouteMatch + matched := router.Match(request, &routeMatch) + if matched != shouldMatch { + // Need better messages. :) + if matched { + t.Errorf("Should match.") + } else { + t.Errorf("Should not match.") + } + } + + if matched { + currentRoute := routeMatch.Route + if currentRoute == nil { + t.Errorf("Expected a current route.") + } + vars := routeMatch.Vars + expectedVars := resultVars[shouldMatch] + if len(vars) != len(expectedVars) { + t.Errorf("Expected vars: %v Got: %v.", expectedVars, vars) + } + for name, value := range vars { + if expectedVars[name] != value { + t.Errorf("Expected vars: %v Got: %v.", expectedVars, vars) + } + } + } + } + + // 1st route -------------------------------------------------------------- + + // Everything match. + reset() + match(true) + + // Scheme doesn't match. + reset() + scheme = "http" + match(false) + + // Host doesn't match. + reset() + host = "www.mygoogle.com" + match(false) + + // Path doesn't match. + reset() + path = "/product/notdigits" + match(false) + + // Query doesn't match. + reset() + query = "?foo=baz" + match(false) + + // Method doesn't match. + reset() + method = "POST" + match(false) + + // Header doesn't match. + reset() + headers = map[string]string{} + match(false) + + // Everything match, again. + reset() + match(true) + + // 2nd route -------------------------------------------------------------- + + // Everything match. + reset2() + match(true) + + // Scheme doesn't match. + reset2() + scheme = "https" + match(false) + + // Host doesn't match. + reset2() + host = "sub.google.com" + match(false) + + // Path doesn't match. + reset2() + path = "/bar/product/42" + match(false) + + // Query doesn't match. + reset2() + query = "?foo=baz" + match(false) + + // Method doesn't match. + reset2() + method = "GET" + match(false) + + // Header doesn't match. + reset2() + headers = map[string]string{} + match(false) + + // Everything match, again. + reset2() + match(true) +} + +type headerMatcherTest struct { + matcher headerMatcher + headers map[string]string + result bool +} + +var headerMatcherTests = []headerMatcherTest{ + { + matcher: headerMatcher(map[string]string{"x-requested-with": "XMLHttpRequest"}), + headers: map[string]string{"X-Requested-With": "XMLHttpRequest"}, + result: true, + }, + { + matcher: headerMatcher(map[string]string{"x-requested-with": ""}), + headers: map[string]string{"X-Requested-With": "anything"}, + result: true, + }, + { + matcher: headerMatcher(map[string]string{"x-requested-with": "XMLHttpRequest"}), + headers: map[string]string{}, + result: false, + }, +} + +type hostMatcherTest struct { + matcher *Route + url string + vars map[string]string + result bool +} + +var hostMatcherTests = []hostMatcherTest{ + { + matcher: NewRouter().NewRoute().Host("{foo:[a-z][a-z][a-z]}.{bar:[a-z][a-z][a-z]}.{baz:[a-z][a-z][a-z]}"), + url: "http://abc.def.ghi/", + vars: map[string]string{"foo": "abc", "bar": "def", "baz": "ghi"}, + result: true, + }, + { + matcher: NewRouter().NewRoute().Host("{foo:[a-z][a-z][a-z]}.{bar:[a-z][a-z][a-z]}.{baz:[a-z][a-z][a-z]}"), + url: "http://a.b.c/", + vars: map[string]string{"foo": "abc", "bar": "def", "baz": "ghi"}, + result: false, + }, +} + +type methodMatcherTest struct { + matcher methodMatcher + method string + result bool +} + +var methodMatcherTests = []methodMatcherTest{ + { + matcher: methodMatcher([]string{"GET", "POST", "PUT"}), + method: "GET", + result: true, + }, + { + matcher: methodMatcher([]string{"GET", "POST", "PUT"}), + method: "POST", + result: true, + }, + { + matcher: methodMatcher([]string{"GET", "POST", "PUT"}), + method: "PUT", + result: true, + }, + { + matcher: methodMatcher([]string{"GET", "POST", "PUT"}), + method: "DELETE", + result: false, + }, +} + +type pathMatcherTest struct { + matcher *Route + url string + vars map[string]string + result bool +} + +var pathMatcherTests = []pathMatcherTest{ + { + matcher: NewRouter().NewRoute().Path("/{foo:[0-9][0-9][0-9]}/{bar:[0-9][0-9][0-9]}/{baz:[0-9][0-9][0-9]}"), + url: "http://localhost:8080/123/456/789", + vars: map[string]string{"foo": "123", "bar": "456", "baz": "789"}, + result: true, + }, + { + matcher: NewRouter().NewRoute().Path("/{foo:[0-9][0-9][0-9]}/{bar:[0-9][0-9][0-9]}/{baz:[0-9][0-9][0-9]}"), + url: "http://localhost:8080/1/2/3", + vars: map[string]string{"foo": "123", "bar": "456", "baz": "789"}, + result: false, + }, +} + +type schemeMatcherTest struct { + matcher schemeMatcher + url string + result bool +} + +var schemeMatcherTests = []schemeMatcherTest{ + { + matcher: schemeMatcher([]string{"http", "https"}), + url: "http://localhost:8080/", + result: true, + }, + { + matcher: schemeMatcher([]string{"http", "https"}), + url: "https://localhost:8080/", + result: true, + }, + { + matcher: schemeMatcher([]string{"https"}), + url: "http://localhost:8080/", + result: false, + }, + { + matcher: schemeMatcher([]string{"http"}), + url: "https://localhost:8080/", + result: false, + }, +} + +type urlBuildingTest struct { + route *Route + vars []string + url string +} + +var urlBuildingTests = []urlBuildingTest{ + { + route: new(Route).Host("foo.domain.com"), + vars: []string{}, + url: "http://foo.domain.com", + }, + { + route: new(Route).Host("{subdomain}.domain.com"), + vars: []string{"subdomain", "bar"}, + url: "http://bar.domain.com", + }, + { + route: new(Route).Host("foo.domain.com").Path("/articles"), + vars: []string{}, + url: "http://foo.domain.com/articles", + }, + { + route: new(Route).Path("/articles"), + vars: []string{}, + url: "/articles", + }, + { + route: new(Route).Path("/articles/{category}/{id:[0-9]+}"), + vars: []string{"category", "technology", "id", "42"}, + url: "/articles/technology/42", + }, + { + route: new(Route).Host("{subdomain}.domain.com").Path("/articles/{category}/{id:[0-9]+}"), + vars: []string{"subdomain", "foo", "category", "technology", "id", "42"}, + url: "http://foo.domain.com/articles/technology/42", + }, +} + +func TestHeaderMatcher(t *testing.T) { + for _, v := range headerMatcherTests { + request, _ := http.NewRequest("GET", "http://localhost:8080/", nil) + for key, value := range v.headers { + request.Header.Add(key, value) + } + var routeMatch RouteMatch + result := v.matcher.Match(request, &routeMatch) + if result != v.result { + if v.result { + t.Errorf("%#v: should match %v.", v.matcher, request.Header) + } else { + t.Errorf("%#v: should not match %v.", v.matcher, request.Header) + } + } + } +} + +func TestHostMatcher(t *testing.T) { + for _, v := range hostMatcherTests { + request, _ := http.NewRequest("GET", v.url, nil) + var routeMatch RouteMatch + result := v.matcher.Match(request, &routeMatch) + vars := routeMatch.Vars + if result != v.result { + if v.result { + t.Errorf("%#v: should match %v.", v.matcher, v.url) + } else { + t.Errorf("%#v: should not match %v.", v.matcher, v.url) + } + } + if result { + if len(vars) != len(v.vars) { + t.Errorf("%#v: vars length should be %v, got %v.", v.matcher, len(v.vars), len(vars)) + } + for name, value := range vars { + if v.vars[name] != value { + t.Errorf("%#v: expected value %v for key %v, got %v.", v.matcher, v.vars[name], name, value) + } + } + } else { + if len(vars) != 0 { + t.Errorf("%#v: vars length should be 0, got %v.", v.matcher, len(vars)) + } + } + } +} + +func TestMethodMatcher(t *testing.T) { + for _, v := range methodMatcherTests { + request, _ := http.NewRequest(v.method, "http://localhost:8080/", nil) + var routeMatch RouteMatch + result := v.matcher.Match(request, &routeMatch) + if result != v.result { + if v.result { + t.Errorf("%#v: should match %v.", v.matcher, v.method) + } else { + t.Errorf("%#v: should not match %v.", v.matcher, v.method) + } + } + } +} + +func TestPathMatcher(t *testing.T) { + for _, v := range pathMatcherTests { + request, _ := http.NewRequest("GET", v.url, nil) + var routeMatch RouteMatch + result := v.matcher.Match(request, &routeMatch) + vars := routeMatch.Vars + if result != v.result { + if v.result { + t.Errorf("%#v: should match %v.", v.matcher, v.url) + } else { + t.Errorf("%#v: should not match %v.", v.matcher, v.url) + } + } + if result { + if len(vars) != len(v.vars) { + t.Errorf("%#v: vars length should be %v, got %v.", v.matcher, len(v.vars), len(vars)) + } + for name, value := range vars { + if v.vars[name] != value { + t.Errorf("%#v: expected value %v for key %v, got %v.", v.matcher, v.vars[name], name, value) + } + } + } else { + if len(vars) != 0 { + t.Errorf("%#v: vars length should be 0, got %v.", v.matcher, len(vars)) + } + } + } +} + +func TestSchemeMatcher(t *testing.T) { + for _, v := range schemeMatcherTests { + request, _ := http.NewRequest("GET", v.url, nil) + var routeMatch RouteMatch + result := v.matcher.Match(request, &routeMatch) + if result != v.result { + if v.result { + t.Errorf("%#v: should match %v.", v.matcher, v.url) + } else { + t.Errorf("%#v: should not match %v.", v.matcher, v.url) + } + } + } +} + +func TestUrlBuilding(t *testing.T) { + + for _, v := range urlBuildingTests { + u, _ := v.route.URL(v.vars...) + url := u.String() + if url != v.url { + t.Errorf("expected %v, got %v", v.url, url) + /* + reversePath := "" + reverseHost := "" + if v.route.pathTemplate != nil { + reversePath = v.route.pathTemplate.Reverse + } + if v.route.hostTemplate != nil { + reverseHost = v.route.hostTemplate.Reverse + } + + t.Errorf("%#v:\nexpected: %q\ngot: %q\nreverse path: %q\nreverse host: %q", v.route, v.url, url, reversePath, reverseHost) + */ + } + } + + ArticleHandler := func(w http.ResponseWriter, r *http.Request) { + } + + router := NewRouter() + router.HandleFunc("/articles/{category}/{id:[0-9]+}", ArticleHandler).Name("article") + + url, _ := router.Get("article").URL("category", "technology", "id", "42") + expected := "/articles/technology/42" + if url.String() != expected { + t.Errorf("Expected %v, got %v", expected, url.String()) + } +} + +func TestMatchedRouteName(t *testing.T) { + routeName := "stock" + router := NewRouter() + route := router.NewRoute().Path("/products/").Name(routeName) + + url := "http://www.example.com/products/" + request, _ := http.NewRequest("GET", url, nil) + var rv RouteMatch + ok := router.Match(request, &rv) + + if !ok || rv.Route != route { + t.Errorf("Expected same route, got %+v.", rv.Route) + } + + retName := rv.Route.GetName() + if retName != routeName { + t.Errorf("Expected %q, got %q.", routeName, retName) + } +} + +func TestSubRouting(t *testing.T) { + // Example from docs. + router := NewRouter() + subrouter := router.NewRoute().Host("www.example.com").Subrouter() + route := subrouter.NewRoute().Path("/products/").Name("products") + + url := "http://www.example.com/products/" + request, _ := http.NewRequest("GET", url, nil) + var rv RouteMatch + ok := router.Match(request, &rv) + + if !ok || rv.Route != route { + t.Errorf("Expected same route, got %+v.", rv.Route) + } + + u, _ := router.Get("products").URL() + builtUrl := u.String() + // Yay, subroute aware of the domain when building! + if builtUrl != url { + t.Errorf("Expected %q, got %q.", url, builtUrl) + } +} + +func TestVariableNames(t *testing.T) { + route := new(Route).Host("{arg1}.domain.com").Path("/{arg1}/{arg2:[0-9]+}") + if route.err == nil { + t.Errorf("Expected error for duplicated variable names") + } +} + +func TestRedirectSlash(t *testing.T) { + var route *Route + var routeMatch RouteMatch + r := NewRouter() + + r.StrictSlash(false) + route = r.NewRoute() + if route.strictSlash != false { + t.Errorf("Expected false redirectSlash.") + } + + r.StrictSlash(true) + route = r.NewRoute() + if route.strictSlash != true { + t.Errorf("Expected true redirectSlash.") + } + + route = new(Route) + route.strictSlash = true + route.Path("/{arg1}/{arg2:[0-9]+}/") + request, _ := http.NewRequest("GET", "http://localhost/foo/123", nil) + routeMatch = RouteMatch{} + _ = route.Match(request, &routeMatch) + vars := routeMatch.Vars + if vars["arg1"] != "foo" { + t.Errorf("Expected foo.") + } + if vars["arg2"] != "123" { + t.Errorf("Expected 123.") + } + rsp := NewRecorder() + routeMatch.Handler.ServeHTTP(rsp, request) + if rsp.HeaderMap.Get("Location") != "http://localhost/foo/123/" { + t.Errorf("Expected redirect header.") + } + + route = new(Route) + route.strictSlash = true + route.Path("/{arg1}/{arg2:[0-9]+}") + request, _ = http.NewRequest("GET", "http://localhost/foo/123/", nil) + routeMatch = RouteMatch{} + _ = route.Match(request, &routeMatch) + vars = routeMatch.Vars + if vars["arg1"] != "foo" { + t.Errorf("Expected foo.") + } + if vars["arg2"] != "123" { + t.Errorf("Expected 123.") + } + rsp = NewRecorder() + routeMatch.Handler.ServeHTTP(rsp, request) + if rsp.HeaderMap.Get("Location") != "http://localhost/foo/123" { + t.Errorf("Expected redirect header.") + } +} + +// Test for the new regexp library, still not available in stable Go. +func TestNewRegexp(t *testing.T) { + var p *routeRegexp + var matches []string + + tests := map[string]map[string][]string{ + "/{foo:a{2}}": { + "/a": nil, + "/aa": {"aa"}, + "/aaa": nil, + "/aaaa": nil, + }, + "/{foo:a{2,}}": { + "/a": nil, + "/aa": {"aa"}, + "/aaa": {"aaa"}, + "/aaaa": {"aaaa"}, + }, + "/{foo:a{2,3}}": { + "/a": nil, + "/aa": {"aa"}, + "/aaa": {"aaa"}, + "/aaaa": nil, + }, + "/{foo:[a-z]{3}}/{bar:[a-z]{2}}": { + "/a": nil, + "/ab": nil, + "/abc": nil, + "/abcd": nil, + "/abc/ab": {"abc", "ab"}, + "/abc/abc": nil, + "/abcd/ab": nil, + }, + `/{foo:\w{3,}}/{bar:\d{2,}}`: { + "/a": nil, + "/ab": nil, + "/abc": nil, + "/abc/1": nil, + "/abc/12": {"abc", "12"}, + "/abcd/12": {"abcd", "12"}, + "/abcd/123": {"abcd", "123"}, + }, + } + + for pattern, paths := range tests { + p, _ = newRouteRegexp(pattern, false, false, false, false) + for path, result := range paths { + matches = p.regexp.FindStringSubmatch(path) + if result == nil { + if matches != nil { + t.Errorf("%v should not match %v.", pattern, path) + } + } else { + if len(matches) != len(result)+1 { + t.Errorf("Expected %v matches, got %v.", len(result)+1, len(matches)) + } else { + for k, v := range result { + if matches[k+1] != v { + t.Errorf("Expected %v, got %v.", v, matches[k+1]) + } + } + } + } + } + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/regexp.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/regexp.go new file mode 100644 index 0000000000000..06728dd545e02 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/regexp.go @@ -0,0 +1,317 @@ +// Copyright 2012 The Gorilla Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package mux + +import ( + "bytes" + "fmt" + "net/http" + "net/url" + "regexp" + "strconv" + "strings" +) + +// newRouteRegexp parses a route template and returns a routeRegexp, +// used to match a host, a path or a query string. +// +// It will extract named variables, assemble a regexp to be matched, create +// a "reverse" template to build URLs and compile regexps to validate variable +// values used in URL building. +// +// Previously we accepted only Python-like identifiers for variable +// names ([a-zA-Z_][a-zA-Z0-9_]*), but currently the only restriction is that +// name and pattern can't be empty, and names can't contain a colon. +func newRouteRegexp(tpl string, matchHost, matchPrefix, matchQuery, strictSlash bool) (*routeRegexp, error) { + // Check if it is well-formed. + idxs, errBraces := braceIndices(tpl) + if errBraces != nil { + return nil, errBraces + } + // Backup the original. + template := tpl + // Now let's parse it. + defaultPattern := "[^/]+" + if matchQuery { + defaultPattern = "[^?&]*" + } else if matchHost { + defaultPattern = "[^.]+" + matchPrefix = false + } + // Only match strict slash if not matching + if matchPrefix || matchHost || matchQuery { + strictSlash = false + } + // Set a flag for strictSlash. + endSlash := false + if strictSlash && strings.HasSuffix(tpl, "/") { + tpl = tpl[:len(tpl)-1] + endSlash = true + } + varsN := make([]string, len(idxs)/2) + varsR := make([]*regexp.Regexp, len(idxs)/2) + pattern := bytes.NewBufferString("") + pattern.WriteByte('^') + reverse := bytes.NewBufferString("") + var end int + var err error + for i := 0; i < len(idxs); i += 2 { + // Set all values we are interested in. + raw := tpl[end:idxs[i]] + end = idxs[i+1] + parts := strings.SplitN(tpl[idxs[i]+1:end-1], ":", 2) + name := parts[0] + patt := defaultPattern + if len(parts) == 2 { + patt = parts[1] + } + // Name or pattern can't be empty. + if name == "" || patt == "" { + return nil, fmt.Errorf("mux: missing name or pattern in %q", + tpl[idxs[i]:end]) + } + // Build the regexp pattern. + varIdx := i / 2 + fmt.Fprintf(pattern, "%s(?P<%s>%s)", regexp.QuoteMeta(raw), varGroupName(varIdx), patt) + // Build the reverse template. + fmt.Fprintf(reverse, "%s%%s", raw) + + // Append variable name and compiled pattern. + varsN[varIdx] = name + varsR[varIdx], err = regexp.Compile(fmt.Sprintf("^%s$", patt)) + if err != nil { + return nil, err + } + } + // Add the remaining. + raw := tpl[end:] + pattern.WriteString(regexp.QuoteMeta(raw)) + if strictSlash { + pattern.WriteString("[/]?") + } + if matchQuery { + // Add the default pattern if the query value is empty + if queryVal := strings.SplitN(template, "=", 2)[1]; queryVal == "" { + pattern.WriteString(defaultPattern) + } + } + if !matchPrefix { + pattern.WriteByte('$') + } + reverse.WriteString(raw) + if endSlash { + reverse.WriteByte('/') + } + // Compile full regexp. + reg, errCompile := regexp.Compile(pattern.String()) + if errCompile != nil { + return nil, errCompile + } + // Done! + return &routeRegexp{ + template: template, + matchHost: matchHost, + matchQuery: matchQuery, + strictSlash: strictSlash, + regexp: reg, + reverse: reverse.String(), + varsN: varsN, + varsR: varsR, + }, nil +} + +// routeRegexp stores a regexp to match a host or path and information to +// collect and validate route variables. +type routeRegexp struct { + // The unmodified template. + template string + // True for host match, false for path or query string match. + matchHost bool + // True for query string match, false for path and host match. + matchQuery bool + // The strictSlash value defined on the route, but disabled if PathPrefix was used. + strictSlash bool + // Expanded regexp. + regexp *regexp.Regexp + // Reverse template. + reverse string + // Variable names. + varsN []string + // Variable regexps (validators). + varsR []*regexp.Regexp +} + +// Match matches the regexp against the URL host or path. +func (r *routeRegexp) Match(req *http.Request, match *RouteMatch) bool { + if !r.matchHost { + if r.matchQuery { + return r.matchQueryString(req) + } else { + return r.regexp.MatchString(req.URL.Path) + } + } + return r.regexp.MatchString(getHost(req)) +} + +// url builds a URL part using the given values. +func (r *routeRegexp) url(values map[string]string) (string, error) { + urlValues := make([]interface{}, len(r.varsN)) + for k, v := range r.varsN { + value, ok := values[v] + if !ok { + return "", fmt.Errorf("mux: missing route variable %q", v) + } + urlValues[k] = value + } + rv := fmt.Sprintf(r.reverse, urlValues...) + if !r.regexp.MatchString(rv) { + // The URL is checked against the full regexp, instead of checking + // individual variables. This is faster but to provide a good error + // message, we check individual regexps if the URL doesn't match. + for k, v := range r.varsN { + if !r.varsR[k].MatchString(values[v]) { + return "", fmt.Errorf( + "mux: variable %q doesn't match, expected %q", values[v], + r.varsR[k].String()) + } + } + } + return rv, nil +} + +// getUrlQuery returns a single query parameter from a request URL. +// For a URL with foo=bar&baz=ding, we return only the relevant key +// value pair for the routeRegexp. +func (r *routeRegexp) getUrlQuery(req *http.Request) string { + if !r.matchQuery { + return "" + } + templateKey := strings.SplitN(r.template, "=", 2)[0] + for key, vals := range req.URL.Query() { + if key == templateKey && len(vals) > 0 { + return key + "=" + vals[0] + } + } + return "" +} + +func (r *routeRegexp) matchQueryString(req *http.Request) bool { + return r.regexp.MatchString(r.getUrlQuery(req)) +} + +// braceIndices returns the first level curly brace indices from a string. +// It returns an error in case of unbalanced braces. +func braceIndices(s string) ([]int, error) { + var level, idx int + idxs := make([]int, 0) + for i := 0; i < len(s); i++ { + switch s[i] { + case '{': + if level++; level == 1 { + idx = i + } + case '}': + if level--; level == 0 { + idxs = append(idxs, idx, i+1) + } else if level < 0 { + return nil, fmt.Errorf("mux: unbalanced braces in %q", s) + } + } + } + if level != 0 { + return nil, fmt.Errorf("mux: unbalanced braces in %q", s) + } + return idxs, nil +} + +// varGroupName builds a capturing group name for the indexed variable. +func varGroupName(idx int) string { + return "v" + strconv.Itoa(idx) +} + +// ---------------------------------------------------------------------------- +// routeRegexpGroup +// ---------------------------------------------------------------------------- + +// routeRegexpGroup groups the route matchers that carry variables. +type routeRegexpGroup struct { + host *routeRegexp + path *routeRegexp + queries []*routeRegexp +} + +// setMatch extracts the variables from the URL once a route matches. +func (v *routeRegexpGroup) setMatch(req *http.Request, m *RouteMatch, r *Route) { + // Store host variables. + if v.host != nil { + hostVars := v.host.regexp.FindStringSubmatch(getHost(req)) + if hostVars != nil { + subexpNames := v.host.regexp.SubexpNames() + varName := 0 + for i, name := range subexpNames[1:] { + if name != "" && name == varGroupName(varName) { + m.Vars[v.host.varsN[varName]] = hostVars[i+1] + varName++ + } + } + } + } + // Store path variables. + if v.path != nil { + pathVars := v.path.regexp.FindStringSubmatch(req.URL.Path) + if pathVars != nil { + subexpNames := v.path.regexp.SubexpNames() + varName := 0 + for i, name := range subexpNames[1:] { + if name != "" && name == varGroupName(varName) { + m.Vars[v.path.varsN[varName]] = pathVars[i+1] + varName++ + } + } + // Check if we should redirect. + if v.path.strictSlash { + p1 := strings.HasSuffix(req.URL.Path, "/") + p2 := strings.HasSuffix(v.path.template, "/") + if p1 != p2 { + u, _ := url.Parse(req.URL.String()) + if p1 { + u.Path = u.Path[:len(u.Path)-1] + } else { + u.Path += "/" + } + m.Handler = http.RedirectHandler(u.String(), 301) + } + } + } + } + // Store query string variables. + for _, q := range v.queries { + queryVars := q.regexp.FindStringSubmatch(q.getUrlQuery(req)) + if queryVars != nil { + subexpNames := q.regexp.SubexpNames() + varName := 0 + for i, name := range subexpNames[1:] { + if name != "" && name == varGroupName(varName) { + m.Vars[q.varsN[varName]] = queryVars[i+1] + varName++ + } + } + } + } +} + +// getHost tries its best to return the request host. +func getHost(r *http.Request) string { + if r.URL.IsAbs() { + return r.URL.Host + } + host := r.Host + // Slice off any port information. + if i := strings.Index(host, ":"); i != -1 { + host = host[:i] + } + return host + +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go new file mode 100644 index 0000000000000..890130460c9c0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux/route.go @@ -0,0 +1,603 @@ +// Copyright 2012 The Gorilla Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package mux + +import ( + "errors" + "fmt" + "net/http" + "net/url" + "regexp" + "strings" +) + +// Route stores information to match a request and build URLs. +type Route struct { + // Parent where the route was registered (a Router). + parent parentRoute + // Request handler for the route. + handler http.Handler + // List of matchers. + matchers []matcher + // Manager for the variables from host and path. + regexp *routeRegexpGroup + // If true, when the path pattern is "/path/", accessing "/path" will + // redirect to the former and vice versa. + strictSlash bool + // If true, this route never matches: it is only used to build URLs. + buildOnly bool + // The name used to build URLs. + name string + // Error resulted from building a route. + err error + + buildVarsFunc BuildVarsFunc +} + +// Match matches the route against the request. +func (r *Route) Match(req *http.Request, match *RouteMatch) bool { + if r.buildOnly || r.err != nil { + return false + } + // Match everything. + for _, m := range r.matchers { + if matched := m.Match(req, match); !matched { + return false + } + } + // Yay, we have a match. Let's collect some info about it. + if match.Route == nil { + match.Route = r + } + if match.Handler == nil { + match.Handler = r.handler + } + if match.Vars == nil { + match.Vars = make(map[string]string) + } + // Set variables. + if r.regexp != nil { + r.regexp.setMatch(req, match, r) + } + return true +} + +// ---------------------------------------------------------------------------- +// Route attributes +// ---------------------------------------------------------------------------- + +// GetError returns an error resulted from building the route, if any. +func (r *Route) GetError() error { + return r.err +} + +// BuildOnly sets the route to never match: it is only used to build URLs. +func (r *Route) BuildOnly() *Route { + r.buildOnly = true + return r +} + +// Handler -------------------------------------------------------------------- + +// Handler sets a handler for the route. +func (r *Route) Handler(handler http.Handler) *Route { + if r.err == nil { + r.handler = handler + } + return r +} + +// HandlerFunc sets a handler function for the route. +func (r *Route) HandlerFunc(f func(http.ResponseWriter, *http.Request)) *Route { + return r.Handler(http.HandlerFunc(f)) +} + +// GetHandler returns the handler for the route, if any. +func (r *Route) GetHandler() http.Handler { + return r.handler +} + +// Name ----------------------------------------------------------------------- + +// Name sets the name for the route, used to build URLs. +// If the name was registered already it will be overwritten. +func (r *Route) Name(name string) *Route { + if r.name != "" { + r.err = fmt.Errorf("mux: route already has name %q, can't set %q", + r.name, name) + } + if r.err == nil { + r.name = name + r.getNamedRoutes()[name] = r + } + return r +} + +// GetName returns the name for the route, if any. +func (r *Route) GetName() string { + return r.name +} + +// ---------------------------------------------------------------------------- +// Matchers +// ---------------------------------------------------------------------------- + +// matcher types try to match a request. +type matcher interface { + Match(*http.Request, *RouteMatch) bool +} + +// addMatcher adds a matcher to the route. +func (r *Route) addMatcher(m matcher) *Route { + if r.err == nil { + r.matchers = append(r.matchers, m) + } + return r +} + +// addRegexpMatcher adds a host or path matcher and builder to a route. +func (r *Route) addRegexpMatcher(tpl string, matchHost, matchPrefix, matchQuery bool) error { + if r.err != nil { + return r.err + } + r.regexp = r.getRegexpGroup() + if !matchHost && !matchQuery { + if len(tpl) == 0 || tpl[0] != '/' { + return fmt.Errorf("mux: path must start with a slash, got %q", tpl) + } + if r.regexp.path != nil { + tpl = strings.TrimRight(r.regexp.path.template, "/") + tpl + } + } + rr, err := newRouteRegexp(tpl, matchHost, matchPrefix, matchQuery, r.strictSlash) + if err != nil { + return err + } + for _, q := range r.regexp.queries { + if err = uniqueVars(rr.varsN, q.varsN); err != nil { + return err + } + } + if matchHost { + if r.regexp.path != nil { + if err = uniqueVars(rr.varsN, r.regexp.path.varsN); err != nil { + return err + } + } + r.regexp.host = rr + } else { + if r.regexp.host != nil { + if err = uniqueVars(rr.varsN, r.regexp.host.varsN); err != nil { + return err + } + } + if matchQuery { + r.regexp.queries = append(r.regexp.queries, rr) + } else { + r.regexp.path = rr + } + } + r.addMatcher(rr) + return nil +} + +// Headers -------------------------------------------------------------------- + +// headerMatcher matches the request against header values. +type headerMatcher map[string]string + +func (m headerMatcher) Match(r *http.Request, match *RouteMatch) bool { + return matchMapWithString(m, r.Header, true) +} + +// Headers adds a matcher for request header values. +// It accepts a sequence of key/value pairs to be matched. For example: +// +// r := mux.NewRouter() +// r.Headers("Content-Type", "application/json", +// "X-Requested-With", "XMLHttpRequest") +// +// The above route will only match if both request header values match. +// Alternatively, you can provide a regular expression and match the header as follows: +// +// r.Headers("Content-Type", "application/(text|json)", +// "X-Requested-With", "XMLHttpRequest") +// +// The above route will the same as the previous example, with the addition of matching +// application/text as well. +// +// It the value is an empty string, it will match any value if the key is set. +func (r *Route) Headers(pairs ...string) *Route { + if r.err == nil { + var headers map[string]string + headers, r.err = mapFromPairsToString(pairs...) + return r.addMatcher(headerMatcher(headers)) + } + return r +} + +// headerRegexMatcher matches the request against the route given a regex for the header +type headerRegexMatcher map[string]*regexp.Regexp + +func (m headerRegexMatcher) Match(r *http.Request, match *RouteMatch) bool { + return matchMapWithRegex(m, r.Header, true) +} + +// Regular expressions can be used with headers as well. +// It accepts a sequence of key/value pairs, where the value has regex support. For example +// r := mux.NewRouter() +// r.HeadersRegexp("Content-Type", "application/(text|json)", +// "X-Requested-With", "XMLHttpRequest") +// +// The above route will only match if both the request header matches both regular expressions. +// It the value is an empty string, it will match any value if the key is set. +func (r *Route) HeadersRegexp(pairs ...string) *Route { + if r.err == nil { + var headers map[string]*regexp.Regexp + headers, r.err = mapFromPairsToRegex(pairs...) + return r.addMatcher(headerRegexMatcher(headers)) + } + return r +} + +// Host ----------------------------------------------------------------------- + +// Host adds a matcher for the URL host. +// It accepts a template with zero or more URL variables enclosed by {}. +// Variables can define an optional regexp pattern to be matched: +// +// - {name} matches anything until the next dot. +// +// - {name:pattern} matches the given regexp pattern. +// +// For example: +// +// r := mux.NewRouter() +// r.Host("www.example.com") +// r.Host("{subdomain}.domain.com") +// r.Host("{subdomain:[a-z]+}.domain.com") +// +// Variable names must be unique in a given route. They can be retrieved +// calling mux.Vars(request). +func (r *Route) Host(tpl string) *Route { + r.err = r.addRegexpMatcher(tpl, true, false, false) + return r +} + +// MatcherFunc ---------------------------------------------------------------- + +// MatcherFunc is the function signature used by custom matchers. +type MatcherFunc func(*http.Request, *RouteMatch) bool + +func (m MatcherFunc) Match(r *http.Request, match *RouteMatch) bool { + return m(r, match) +} + +// MatcherFunc adds a custom function to be used as request matcher. +func (r *Route) MatcherFunc(f MatcherFunc) *Route { + return r.addMatcher(f) +} + +// Methods -------------------------------------------------------------------- + +// methodMatcher matches the request against HTTP methods. +type methodMatcher []string + +func (m methodMatcher) Match(r *http.Request, match *RouteMatch) bool { + return matchInArray(m, r.Method) +} + +// Methods adds a matcher for HTTP methods. +// It accepts a sequence of one or more methods to be matched, e.g.: +// "GET", "POST", "PUT". +func (r *Route) Methods(methods ...string) *Route { + for k, v := range methods { + methods[k] = strings.ToUpper(v) + } + return r.addMatcher(methodMatcher(methods)) +} + +// Path ----------------------------------------------------------------------- + +// Path adds a matcher for the URL path. +// It accepts a template with zero or more URL variables enclosed by {}. The +// template must start with a "/". +// Variables can define an optional regexp pattern to be matched: +// +// - {name} matches anything until the next slash. +// +// - {name:pattern} matches the given regexp pattern. +// +// For example: +// +// r := mux.NewRouter() +// r.Path("/products/").Handler(ProductsHandler) +// r.Path("/products/{key}").Handler(ProductsHandler) +// r.Path("/articles/{category}/{id:[0-9]+}"). +// Handler(ArticleHandler) +// +// Variable names must be unique in a given route. They can be retrieved +// calling mux.Vars(request). +func (r *Route) Path(tpl string) *Route { + r.err = r.addRegexpMatcher(tpl, false, false, false) + return r +} + +// PathPrefix ----------------------------------------------------------------- + +// PathPrefix adds a matcher for the URL path prefix. This matches if the given +// template is a prefix of the full URL path. See Route.Path() for details on +// the tpl argument. +// +// Note that it does not treat slashes specially ("/foobar/" will be matched by +// the prefix "/foo") so you may want to use a trailing slash here. +// +// Also note that the setting of Router.StrictSlash() has no effect on routes +// with a PathPrefix matcher. +func (r *Route) PathPrefix(tpl string) *Route { + r.err = r.addRegexpMatcher(tpl, false, true, false) + return r +} + +// Query ---------------------------------------------------------------------- + +// Queries adds a matcher for URL query values. +// It accepts a sequence of key/value pairs. Values may define variables. +// For example: +// +// r := mux.NewRouter() +// r.Queries("foo", "bar", "id", "{id:[0-9]+}") +// +// The above route will only match if the URL contains the defined queries +// values, e.g.: ?foo=bar&id=42. +// +// It the value is an empty string, it will match any value if the key is set. +// +// Variables can define an optional regexp pattern to be matched: +// +// - {name} matches anything until the next slash. +// +// - {name:pattern} matches the given regexp pattern. +func (r *Route) Queries(pairs ...string) *Route { + length := len(pairs) + if length%2 != 0 { + r.err = fmt.Errorf( + "mux: number of parameters must be multiple of 2, got %v", pairs) + return nil + } + for i := 0; i < length; i += 2 { + if r.err = r.addRegexpMatcher(pairs[i]+"="+pairs[i+1], false, false, true); r.err != nil { + return r + } + } + + return r +} + +// Schemes -------------------------------------------------------------------- + +// schemeMatcher matches the request against URL schemes. +type schemeMatcher []string + +func (m schemeMatcher) Match(r *http.Request, match *RouteMatch) bool { + return matchInArray(m, r.URL.Scheme) +} + +// Schemes adds a matcher for URL schemes. +// It accepts a sequence of schemes to be matched, e.g.: "http", "https". +func (r *Route) Schemes(schemes ...string) *Route { + for k, v := range schemes { + schemes[k] = strings.ToLower(v) + } + return r.addMatcher(schemeMatcher(schemes)) +} + +// BuildVarsFunc -------------------------------------------------------------- + +// BuildVarsFunc is the function signature used by custom build variable +// functions (which can modify route variables before a route's URL is built). +type BuildVarsFunc func(map[string]string) map[string]string + +// BuildVarsFunc adds a custom function to be used to modify build variables +// before a route's URL is built. +func (r *Route) BuildVarsFunc(f BuildVarsFunc) *Route { + r.buildVarsFunc = f + return r +} + +// Subrouter ------------------------------------------------------------------ + +// Subrouter creates a subrouter for the route. +// +// It will test the inner routes only if the parent route matched. For example: +// +// r := mux.NewRouter() +// s := r.Host("www.example.com").Subrouter() +// s.HandleFunc("/products/", ProductsHandler) +// s.HandleFunc("/products/{key}", ProductHandler) +// s.HandleFunc("/articles/{category}/{id:[0-9]+}"), ArticleHandler) +// +// Here, the routes registered in the subrouter won't be tested if the host +// doesn't match. +func (r *Route) Subrouter() *Router { + router := &Router{parent: r, strictSlash: r.strictSlash} + r.addMatcher(router) + return router +} + +// ---------------------------------------------------------------------------- +// URL building +// ---------------------------------------------------------------------------- + +// URL builds a URL for the route. +// +// It accepts a sequence of key/value pairs for the route variables. For +// example, given this route: +// +// r := mux.NewRouter() +// r.HandleFunc("/articles/{category}/{id:[0-9]+}", ArticleHandler). +// Name("article") +// +// ...a URL for it can be built using: +// +// url, err := r.Get("article").URL("category", "technology", "id", "42") +// +// ...which will return an url.URL with the following path: +// +// "/articles/technology/42" +// +// This also works for host variables: +// +// r := mux.NewRouter() +// r.Host("{subdomain}.domain.com"). +// HandleFunc("/articles/{category}/{id:[0-9]+}", ArticleHandler). +// Name("article") +// +// // url.String() will be "http://news.domain.com/articles/technology/42" +// url, err := r.Get("article").URL("subdomain", "news", +// "category", "technology", +// "id", "42") +// +// All variables defined in the route are required, and their values must +// conform to the corresponding patterns. +func (r *Route) URL(pairs ...string) (*url.URL, error) { + if r.err != nil { + return nil, r.err + } + if r.regexp == nil { + return nil, errors.New("mux: route doesn't have a host or path") + } + values, err := r.prepareVars(pairs...) + if err != nil { + return nil, err + } + var scheme, host, path string + if r.regexp.host != nil { + // Set a default scheme. + scheme = "http" + if host, err = r.regexp.host.url(values); err != nil { + return nil, err + } + } + if r.regexp.path != nil { + if path, err = r.regexp.path.url(values); err != nil { + return nil, err + } + } + return &url.URL{ + Scheme: scheme, + Host: host, + Path: path, + }, nil +} + +// URLHost builds the host part of the URL for a route. See Route.URL(). +// +// The route must have a host defined. +func (r *Route) URLHost(pairs ...string) (*url.URL, error) { + if r.err != nil { + return nil, r.err + } + if r.regexp == nil || r.regexp.host == nil { + return nil, errors.New("mux: route doesn't have a host") + } + values, err := r.prepareVars(pairs...) + if err != nil { + return nil, err + } + host, err := r.regexp.host.url(values) + if err != nil { + return nil, err + } + return &url.URL{ + Scheme: "http", + Host: host, + }, nil +} + +// URLPath builds the path part of the URL for a route. See Route.URL(). +// +// The route must have a path defined. +func (r *Route) URLPath(pairs ...string) (*url.URL, error) { + if r.err != nil { + return nil, r.err + } + if r.regexp == nil || r.regexp.path == nil { + return nil, errors.New("mux: route doesn't have a path") + } + values, err := r.prepareVars(pairs...) + if err != nil { + return nil, err + } + path, err := r.regexp.path.url(values) + if err != nil { + return nil, err + } + return &url.URL{ + Path: path, + }, nil +} + +// prepareVars converts the route variable pairs into a map. If the route has a +// BuildVarsFunc, it is invoked. +func (r *Route) prepareVars(pairs ...string) (map[string]string, error) { + m, err := mapFromPairsToString(pairs...) + if err != nil { + return nil, err + } + return r.buildVars(m), nil +} + +func (r *Route) buildVars(m map[string]string) map[string]string { + if r.parent != nil { + m = r.parent.buildVars(m) + } + if r.buildVarsFunc != nil { + m = r.buildVarsFunc(m) + } + return m +} + +// ---------------------------------------------------------------------------- +// parentRoute +// ---------------------------------------------------------------------------- + +// parentRoute allows routes to know about parent host and path definitions. +type parentRoute interface { + getNamedRoutes() map[string]*Route + getRegexpGroup() *routeRegexpGroup + buildVars(map[string]string) map[string]string +} + +// getNamedRoutes returns the map where named routes are registered. +func (r *Route) getNamedRoutes() map[string]*Route { + if r.parent == nil { + // During tests router is not always set. + r.parent = NewRouter() + } + return r.parent.getNamedRoutes() +} + +// getRegexpGroup returns regexp definitions from this route. +func (r *Route) getRegexpGroup() *routeRegexpGroup { + if r.regexp == nil { + if r.parent == nil { + // During tests router is not always set. + r.parent = NewRouter() + } + regexp := r.parent.getRegexpGroup() + if regexp == nil { + r.regexp = new(routeRegexpGroup) + } else { + // Copy. + r.regexp = &routeRegexpGroup{ + host: regexp.host, + path: regexp.path, + queries: regexp.queries, + } + } + } + return r.regexp +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/MAINTAINERS b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/MAINTAINERS new file mode 100644 index 0000000000000..edbe200669461 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/MAINTAINERS @@ -0,0 +1,2 @@ +Tianon Gravi (@tianon) +Aleksa Sarai (@cyphar) diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/lookup.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/lookup.go new file mode 100644 index 0000000000000..6f8a982ff72a6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/lookup.go @@ -0,0 +1,108 @@ +package user + +import ( + "errors" + "fmt" + "syscall" +) + +var ( + // The current operating system does not provide the required data for user lookups. + ErrUnsupported = errors.New("user lookup: operating system does not provide passwd-formatted data") +) + +func lookupUser(filter func(u User) bool) (User, error) { + // Get operating system-specific passwd reader-closer. + passwd, err := GetPasswd() + if err != nil { + return User{}, err + } + defer passwd.Close() + + // Get the users. + users, err := ParsePasswdFilter(passwd, filter) + if err != nil { + return User{}, err + } + + // No user entries found. + if len(users) == 0 { + return User{}, fmt.Errorf("no matching entries in passwd file") + } + + // Assume the first entry is the "correct" one. + return users[0], nil +} + +// CurrentUser looks up the current user by their user id in /etc/passwd. If the +// user cannot be found (or there is no /etc/passwd file on the filesystem), +// then CurrentUser returns an error. +func CurrentUser() (User, error) { + return LookupUid(syscall.Getuid()) +} + +// LookupUser looks up a user by their username in /etc/passwd. If the user +// cannot be found (or there is no /etc/passwd file on the filesystem), then +// LookupUser returns an error. +func LookupUser(username string) (User, error) { + return lookupUser(func(u User) bool { + return u.Name == username + }) +} + +// LookupUid looks up a user by their user id in /etc/passwd. If the user cannot +// be found (or there is no /etc/passwd file on the filesystem), then LookupId +// returns an error. +func LookupUid(uid int) (User, error) { + return lookupUser(func(u User) bool { + return u.Uid == uid + }) +} + +func lookupGroup(filter func(g Group) bool) (Group, error) { + // Get operating system-specific group reader-closer. + group, err := GetGroup() + if err != nil { + return Group{}, err + } + defer group.Close() + + // Get the users. + groups, err := ParseGroupFilter(group, filter) + if err != nil { + return Group{}, err + } + + // No user entries found. + if len(groups) == 0 { + return Group{}, fmt.Errorf("no matching entries in group file") + } + + // Assume the first entry is the "correct" one. + return groups[0], nil +} + +// CurrentGroup looks up the current user's group by their primary group id's +// entry in /etc/passwd. If the group cannot be found (or there is no +// /etc/group file on the filesystem), then CurrentGroup returns an error. +func CurrentGroup() (Group, error) { + return LookupGid(syscall.Getgid()) +} + +// LookupGroup looks up a group by its name in /etc/group. If the group cannot +// be found (or there is no /etc/group file on the filesystem), then LookupGroup +// returns an error. +func LookupGroup(groupname string) (Group, error) { + return lookupGroup(func(g Group) bool { + return g.Name == groupname + }) +} + +// LookupGid looks up a group by its group id in /etc/group. If the group cannot +// be found (or there is no /etc/group file on the filesystem), then LookupGid +// returns an error. +func LookupGid(gid int) (Group, error) { + return lookupGroup(func(g Group) bool { + return g.Gid == gid + }) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/lookup_unix.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/lookup_unix.go new file mode 100644 index 0000000000000..758b734c225ab --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/lookup_unix.go @@ -0,0 +1,30 @@ +// +build darwin dragonfly freebsd linux netbsd openbsd solaris + +package user + +import ( + "io" + "os" +) + +// Unix-specific path to the passwd and group formatted files. +const ( + unixPasswdPath = "/etc/passwd" + unixGroupPath = "/etc/group" +) + +func GetPasswdPath() (string, error) { + return unixPasswdPath, nil +} + +func GetPasswd() (io.ReadCloser, error) { + return os.Open(unixPasswdPath) +} + +func GetGroupPath() (string, error) { + return unixGroupPath, nil +} + +func GetGroup() (io.ReadCloser, error) { + return os.Open(unixGroupPath) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/lookup_unsupported.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/lookup_unsupported.go new file mode 100644 index 0000000000000..7217948870c51 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/lookup_unsupported.go @@ -0,0 +1,21 @@ +// +build !darwin,!dragonfly,!freebsd,!linux,!netbsd,!openbsd,!solaris + +package user + +import "io" + +func GetPasswdPath() (string, error) { + return "", ErrUnsupported +} + +func GetPasswd() (io.ReadCloser, error) { + return nil, ErrUnsupported +} + +func GetGroupPath() (string, error) { + return "", ErrUnsupported +} + +func GetGroup() (io.ReadCloser, error) { + return nil, ErrUnsupported +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/user.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/user.go new file mode 100644 index 0000000000000..964e31bfd4ad9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/user.go @@ -0,0 +1,413 @@ +package user + +import ( + "bufio" + "fmt" + "io" + "os" + "strconv" + "strings" +) + +const ( + minId = 0 + maxId = 1<<31 - 1 //for 32-bit systems compatibility +) + +var ( + ErrRange = fmt.Errorf("Uids and gids must be in range %d-%d", minId, maxId) +) + +type User struct { + Name string + Pass string + Uid int + Gid int + Gecos string + Home string + Shell string +} + +type Group struct { + Name string + Pass string + Gid int + List []string +} + +func parseLine(line string, v ...interface{}) { + if line == "" { + return + } + + parts := strings.Split(line, ":") + for i, p := range parts { + if len(v) <= i { + // if we have more "parts" than we have places to put them, bail for great "tolerance" of naughty configuration files + break + } + + switch e := v[i].(type) { + case *string: + // "root", "adm", "/bin/bash" + *e = p + case *int: + // "0", "4", "1000" + // ignore string to int conversion errors, for great "tolerance" of naughty configuration files + *e, _ = strconv.Atoi(p) + case *[]string: + // "", "root", "root,adm,daemon" + if p != "" { + *e = strings.Split(p, ",") + } else { + *e = []string{} + } + default: + // panic, because this is a programming/logic error, not a runtime one + panic("parseLine expects only pointers! argument " + strconv.Itoa(i) + " is not a pointer!") + } + } +} + +func ParsePasswdFile(path string) ([]User, error) { + passwd, err := os.Open(path) + if err != nil { + return nil, err + } + defer passwd.Close() + return ParsePasswd(passwd) +} + +func ParsePasswd(passwd io.Reader) ([]User, error) { + return ParsePasswdFilter(passwd, nil) +} + +func ParsePasswdFileFilter(path string, filter func(User) bool) ([]User, error) { + passwd, err := os.Open(path) + if err != nil { + return nil, err + } + defer passwd.Close() + return ParsePasswdFilter(passwd, filter) +} + +func ParsePasswdFilter(r io.Reader, filter func(User) bool) ([]User, error) { + if r == nil { + return nil, fmt.Errorf("nil source for passwd-formatted data") + } + + var ( + s = bufio.NewScanner(r) + out = []User{} + ) + + for s.Scan() { + if err := s.Err(); err != nil { + return nil, err + } + + text := strings.TrimSpace(s.Text()) + if text == "" { + continue + } + + // see: man 5 passwd + // name:password:UID:GID:GECOS:directory:shell + // Name:Pass:Uid:Gid:Gecos:Home:Shell + // root:x:0:0:root:/root:/bin/bash + // adm:x:3:4:adm:/var/adm:/bin/false + p := User{} + parseLine( + text, + &p.Name, &p.Pass, &p.Uid, &p.Gid, &p.Gecos, &p.Home, &p.Shell, + ) + + if filter == nil || filter(p) { + out = append(out, p) + } + } + + return out, nil +} + +func ParseGroupFile(path string) ([]Group, error) { + group, err := os.Open(path) + if err != nil { + return nil, err + } + defer group.Close() + return ParseGroup(group) +} + +func ParseGroup(group io.Reader) ([]Group, error) { + return ParseGroupFilter(group, nil) +} + +func ParseGroupFileFilter(path string, filter func(Group) bool) ([]Group, error) { + group, err := os.Open(path) + if err != nil { + return nil, err + } + defer group.Close() + return ParseGroupFilter(group, filter) +} + +func ParseGroupFilter(r io.Reader, filter func(Group) bool) ([]Group, error) { + if r == nil { + return nil, fmt.Errorf("nil source for group-formatted data") + } + + var ( + s = bufio.NewScanner(r) + out = []Group{} + ) + + for s.Scan() { + if err := s.Err(); err != nil { + return nil, err + } + + text := s.Text() + if text == "" { + continue + } + + // see: man 5 group + // group_name:password:GID:user_list + // Name:Pass:Gid:List + // root:x:0:root + // adm:x:4:root,adm,daemon + p := Group{} + parseLine( + text, + &p.Name, &p.Pass, &p.Gid, &p.List, + ) + + if filter == nil || filter(p) { + out = append(out, p) + } + } + + return out, nil +} + +type ExecUser struct { + Uid, Gid int + Sgids []int + Home string +} + +// GetExecUserPath is a wrapper for GetExecUser. It reads data from each of the +// given file paths and uses that data as the arguments to GetExecUser. If the +// files cannot be opened for any reason, the error is ignored and a nil +// io.Reader is passed instead. +func GetExecUserPath(userSpec string, defaults *ExecUser, passwdPath, groupPath string) (*ExecUser, error) { + passwd, err := os.Open(passwdPath) + if err != nil { + passwd = nil + } else { + defer passwd.Close() + } + + group, err := os.Open(groupPath) + if err != nil { + group = nil + } else { + defer group.Close() + } + + return GetExecUser(userSpec, defaults, passwd, group) +} + +// GetExecUser parses a user specification string (using the passwd and group +// readers as sources for /etc/passwd and /etc/group data, respectively). In +// the case of blank fields or missing data from the sources, the values in +// defaults is used. +// +// GetExecUser will return an error if a user or group literal could not be +// found in any entry in passwd and group respectively. +// +// Examples of valid user specifications are: +// * "" +// * "user" +// * "uid" +// * "user:group" +// * "uid:gid +// * "user:gid" +// * "uid:group" +func GetExecUser(userSpec string, defaults *ExecUser, passwd, group io.Reader) (*ExecUser, error) { + var ( + userArg, groupArg string + name string + ) + + if defaults == nil { + defaults = new(ExecUser) + } + + // Copy over defaults. + user := &ExecUser{ + Uid: defaults.Uid, + Gid: defaults.Gid, + Sgids: defaults.Sgids, + Home: defaults.Home, + } + + // Sgids slice *cannot* be nil. + if user.Sgids == nil { + user.Sgids = []int{} + } + + // allow for userArg to have either "user" syntax, or optionally "user:group" syntax + parseLine(userSpec, &userArg, &groupArg) + + users, err := ParsePasswdFilter(passwd, func(u User) bool { + if userArg == "" { + return u.Uid == user.Uid + } + return u.Name == userArg || strconv.Itoa(u.Uid) == userArg + }) + if err != nil && passwd != nil { + if userArg == "" { + userArg = strconv.Itoa(user.Uid) + } + return nil, fmt.Errorf("Unable to find user %v: %v", userArg, err) + } + + haveUser := users != nil && len(users) > 0 + if haveUser { + // if we found any user entries that matched our filter, let's take the first one as "correct" + name = users[0].Name + user.Uid = users[0].Uid + user.Gid = users[0].Gid + user.Home = users[0].Home + } else if userArg != "" { + // we asked for a user but didn't find them... let's check to see if we wanted a numeric user + user.Uid, err = strconv.Atoi(userArg) + if err != nil { + // not numeric - we have to bail + return nil, fmt.Errorf("Unable to find user %v", userArg) + } + + // Must be inside valid uid range. + if user.Uid < minId || user.Uid > maxId { + return nil, ErrRange + } + + // if userArg couldn't be found in /etc/passwd but is numeric, just roll with it - this is legit + } + + if groupArg != "" || name != "" { + groups, err := ParseGroupFilter(group, func(g Group) bool { + // Explicit group format takes precedence. + if groupArg != "" { + return g.Name == groupArg || strconv.Itoa(g.Gid) == groupArg + } + + // Check if user is a member. + for _, u := range g.List { + if u == name { + return true + } + } + + return false + }) + if err != nil && group != nil { + return nil, fmt.Errorf("Unable to find groups for user %v: %v", users[0].Name, err) + } + + haveGroup := groups != nil && len(groups) > 0 + if groupArg != "" { + if haveGroup { + // if we found any group entries that matched our filter, let's take the first one as "correct" + user.Gid = groups[0].Gid + } else { + // we asked for a group but didn't find id... let's check to see if we wanted a numeric group + user.Gid, err = strconv.Atoi(groupArg) + if err != nil { + // not numeric - we have to bail + return nil, fmt.Errorf("Unable to find group %v", groupArg) + } + + // Ensure gid is inside gid range. + if user.Gid < minId || user.Gid > maxId { + return nil, ErrRange + } + + // if groupArg couldn't be found in /etc/group but is numeric, just roll with it - this is legit + } + } else if haveGroup { + // If implicit group format, fill supplementary gids. + user.Sgids = make([]int, len(groups)) + for i, group := range groups { + user.Sgids[i] = group.Gid + } + } + } + + return user, nil +} + +// GetAdditionalGroups looks up a list of groups by name or group id against +// against the given /etc/group formatted data. If a group name cannot be found, +// an error will be returned. If a group id cannot be found, it will be returned +// as-is. +func GetAdditionalGroups(additionalGroups []string, group io.Reader) ([]int, error) { + groups, err := ParseGroupFilter(group, func(g Group) bool { + for _, ag := range additionalGroups { + if g.Name == ag || strconv.Itoa(g.Gid) == ag { + return true + } + } + return false + }) + if err != nil { + return nil, fmt.Errorf("Unable to find additional groups %v: %v", additionalGroups, err) + } + + gidMap := make(map[int]struct{}) + for _, ag := range additionalGroups { + var found bool + for _, g := range groups { + // if we found a matched group either by name or gid, take the + // first matched as correct + if g.Name == ag || strconv.Itoa(g.Gid) == ag { + if _, ok := gidMap[g.Gid]; !ok { + gidMap[g.Gid] = struct{}{} + found = true + break + } + } + } + // we asked for a group but didn't find it. let's check to see + // if we wanted a numeric group + if !found { + gid, err := strconv.Atoi(ag) + if err != nil { + return nil, fmt.Errorf("Unable to find group %s", ag) + } + // Ensure gid is inside gid range. + if gid < minId || gid > maxId { + return nil, ErrRange + } + gidMap[gid] = struct{}{} + } + } + gids := []int{} + for gid := range gidMap { + gids = append(gids, gid) + } + return gids, nil +} + +// Wrapper around GetAdditionalGroups that opens the groupPath given and gives +// it as an argument to GetAdditionalGroups. +func GetAdditionalGroupsPath(additionalGroups []string, groupPath string) ([]int, error) { + group, err := os.Open(groupPath) + if err != nil { + return nil, fmt.Errorf("Failed to open group file: %v", err) + } + defer group.Close() + return GetAdditionalGroups(additionalGroups, group) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/user_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/user_test.go new file mode 100644 index 0000000000000..0e37ac3dd2e32 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/external/github.com/opencontainers/runc/libcontainer/user/user_test.go @@ -0,0 +1,436 @@ +package user + +import ( + "io" + "reflect" + "sort" + "strconv" + "strings" + "testing" +) + +func TestUserParseLine(t *testing.T) { + var ( + a, b string + c []string + d int + ) + + parseLine("", &a, &b) + if a != "" || b != "" { + t.Fatalf("a and b should be empty ('%v', '%v')", a, b) + } + + parseLine("a", &a, &b) + if a != "a" || b != "" { + t.Fatalf("a should be 'a' and b should be empty ('%v', '%v')", a, b) + } + + parseLine("bad boys:corny cows", &a, &b) + if a != "bad boys" || b != "corny cows" { + t.Fatalf("a should be 'bad boys' and b should be 'corny cows' ('%v', '%v')", a, b) + } + + parseLine("", &c) + if len(c) != 0 { + t.Fatalf("c should be empty (%#v)", c) + } + + parseLine("d,e,f:g:h:i,j,k", &c, &a, &b, &c) + if a != "g" || b != "h" || len(c) != 3 || c[0] != "i" || c[1] != "j" || c[2] != "k" { + t.Fatalf("a should be 'g', b should be 'h', and c should be ['i','j','k'] ('%v', '%v', '%#v')", a, b, c) + } + + parseLine("::::::::::", &a, &b, &c) + if a != "" || b != "" || len(c) != 0 { + t.Fatalf("a, b, and c should all be empty ('%v', '%v', '%#v')", a, b, c) + } + + parseLine("not a number", &d) + if d != 0 { + t.Fatalf("d should be 0 (%v)", d) + } + + parseLine("b:12:c", &a, &d, &b) + if a != "b" || b != "c" || d != 12 { + t.Fatalf("a should be 'b' and b should be 'c', and d should be 12 ('%v', '%v', %v)", a, b, d) + } +} + +func TestUserParsePasswd(t *testing.T) { + users, err := ParsePasswdFilter(strings.NewReader(` +root:x:0:0:root:/root:/bin/bash +adm:x:3:4:adm:/var/adm:/bin/false +this is just some garbage data +`), nil) + if err != nil { + t.Fatalf("Unexpected error: %v", err) + } + if len(users) != 3 { + t.Fatalf("Expected 3 users, got %v", len(users)) + } + if users[0].Uid != 0 || users[0].Name != "root" { + t.Fatalf("Expected users[0] to be 0 - root, got %v - %v", users[0].Uid, users[0].Name) + } + if users[1].Uid != 3 || users[1].Name != "adm" { + t.Fatalf("Expected users[1] to be 3 - adm, got %v - %v", users[1].Uid, users[1].Name) + } +} + +func TestUserParseGroup(t *testing.T) { + groups, err := ParseGroupFilter(strings.NewReader(` +root:x:0:root +adm:x:4:root,adm,daemon +this is just some garbage data +`), nil) + if err != nil { + t.Fatalf("Unexpected error: %v", err) + } + if len(groups) != 3 { + t.Fatalf("Expected 3 groups, got %v", len(groups)) + } + if groups[0].Gid != 0 || groups[0].Name != "root" || len(groups[0].List) != 1 { + t.Fatalf("Expected groups[0] to be 0 - root - 1 member, got %v - %v - %v", groups[0].Gid, groups[0].Name, len(groups[0].List)) + } + if groups[1].Gid != 4 || groups[1].Name != "adm" || len(groups[1].List) != 3 { + t.Fatalf("Expected groups[1] to be 4 - adm - 3 members, got %v - %v - %v", groups[1].Gid, groups[1].Name, len(groups[1].List)) + } +} + +func TestValidGetExecUser(t *testing.T) { + const passwdContent = ` +root:x:0:0:root user:/root:/bin/bash +adm:x:42:43:adm:/var/adm:/bin/false +this is just some garbage data +` + const groupContent = ` +root:x:0:root +adm:x:43: +grp:x:1234:root,adm +this is just some garbage data +` + defaultExecUser := ExecUser{ + Uid: 8888, + Gid: 8888, + Sgids: []int{8888}, + Home: "/8888", + } + + tests := []struct { + ref string + expected ExecUser + }{ + { + ref: "root", + expected: ExecUser{ + Uid: 0, + Gid: 0, + Sgids: []int{0, 1234}, + Home: "/root", + }, + }, + { + ref: "adm", + expected: ExecUser{ + Uid: 42, + Gid: 43, + Sgids: []int{1234}, + Home: "/var/adm", + }, + }, + { + ref: "root:adm", + expected: ExecUser{ + Uid: 0, + Gid: 43, + Sgids: defaultExecUser.Sgids, + Home: "/root", + }, + }, + { + ref: "adm:1234", + expected: ExecUser{ + Uid: 42, + Gid: 1234, + Sgids: defaultExecUser.Sgids, + Home: "/var/adm", + }, + }, + { + ref: "42:1234", + expected: ExecUser{ + Uid: 42, + Gid: 1234, + Sgids: defaultExecUser.Sgids, + Home: "/var/adm", + }, + }, + { + ref: "1337:1234", + expected: ExecUser{ + Uid: 1337, + Gid: 1234, + Sgids: defaultExecUser.Sgids, + Home: defaultExecUser.Home, + }, + }, + { + ref: "1337", + expected: ExecUser{ + Uid: 1337, + Gid: defaultExecUser.Gid, + Sgids: defaultExecUser.Sgids, + Home: defaultExecUser.Home, + }, + }, + { + ref: "", + expected: ExecUser{ + Uid: defaultExecUser.Uid, + Gid: defaultExecUser.Gid, + Sgids: defaultExecUser.Sgids, + Home: defaultExecUser.Home, + }, + }, + } + + for _, test := range tests { + passwd := strings.NewReader(passwdContent) + group := strings.NewReader(groupContent) + + execUser, err := GetExecUser(test.ref, &defaultExecUser, passwd, group) + if err != nil { + t.Logf("got unexpected error when parsing '%s': %s", test.ref, err.Error()) + t.Fail() + continue + } + + if !reflect.DeepEqual(test.expected, *execUser) { + t.Logf("got: %#v", execUser) + t.Logf("expected: %#v", test.expected) + t.Fail() + continue + } + } +} + +func TestInvalidGetExecUser(t *testing.T) { + const passwdContent = ` +root:x:0:0:root user:/root:/bin/bash +adm:x:42:43:adm:/var/adm:/bin/false +this is just some garbage data +` + const groupContent = ` +root:x:0:root +adm:x:43: +grp:x:1234:root,adm +this is just some garbage data +` + + tests := []string{ + // No such user/group. + "notuser", + "notuser:notgroup", + "root:notgroup", + "notuser:adm", + "8888:notgroup", + "notuser:8888", + + // Invalid user/group values. + "-1:0", + "0:-3", + "-5:-2", + } + + for _, test := range tests { + passwd := strings.NewReader(passwdContent) + group := strings.NewReader(groupContent) + + execUser, err := GetExecUser(test, nil, passwd, group) + if err == nil { + t.Logf("got unexpected success when parsing '%s': %#v", test, execUser) + t.Fail() + continue + } + } +} + +func TestGetExecUserNilSources(t *testing.T) { + const passwdContent = ` +root:x:0:0:root user:/root:/bin/bash +adm:x:42:43:adm:/var/adm:/bin/false +this is just some garbage data +` + const groupContent = ` +root:x:0:root +adm:x:43: +grp:x:1234:root,adm +this is just some garbage data +` + + defaultExecUser := ExecUser{ + Uid: 8888, + Gid: 8888, + Sgids: []int{8888}, + Home: "/8888", + } + + tests := []struct { + ref string + passwd, group bool + expected ExecUser + }{ + { + ref: "", + passwd: false, + group: false, + expected: ExecUser{ + Uid: 8888, + Gid: 8888, + Sgids: []int{8888}, + Home: "/8888", + }, + }, + { + ref: "root", + passwd: true, + group: false, + expected: ExecUser{ + Uid: 0, + Gid: 0, + Sgids: []int{8888}, + Home: "/root", + }, + }, + { + ref: "0", + passwd: false, + group: false, + expected: ExecUser{ + Uid: 0, + Gid: 8888, + Sgids: []int{8888}, + Home: "/8888", + }, + }, + { + ref: "0:0", + passwd: false, + group: false, + expected: ExecUser{ + Uid: 0, + Gid: 0, + Sgids: []int{8888}, + Home: "/8888", + }, + }, + } + + for _, test := range tests { + var passwd, group io.Reader + + if test.passwd { + passwd = strings.NewReader(passwdContent) + } + + if test.group { + group = strings.NewReader(groupContent) + } + + execUser, err := GetExecUser(test.ref, &defaultExecUser, passwd, group) + if err != nil { + t.Logf("got unexpected error when parsing '%s': %s", test.ref, err.Error()) + t.Fail() + continue + } + + if !reflect.DeepEqual(test.expected, *execUser) { + t.Logf("got: %#v", execUser) + t.Logf("expected: %#v", test.expected) + t.Fail() + continue + } + } +} + +func TestGetAdditionalGroups(t *testing.T) { + const groupContent = ` +root:x:0:root +adm:x:43: +grp:x:1234:root,adm +adm:x:4343:root,adm-duplicate +this is just some garbage data +` + tests := []struct { + groups []string + expected []int + hasError bool + }{ + { + // empty group + groups: []string{}, + expected: []int{}, + }, + { + // single group + groups: []string{"adm"}, + expected: []int{43}, + }, + { + // multiple groups + groups: []string{"adm", "grp"}, + expected: []int{43, 1234}, + }, + { + // invalid group + groups: []string{"adm", "grp", "not-exist"}, + expected: nil, + hasError: true, + }, + { + // group with numeric id + groups: []string{"43"}, + expected: []int{43}, + }, + { + // group with unknown numeric id + groups: []string{"adm", "10001"}, + expected: []int{43, 10001}, + }, + { + // groups specified twice with numeric and name + groups: []string{"adm", "43"}, + expected: []int{43}, + }, + { + // groups with too small id + groups: []string{"-1"}, + expected: nil, + hasError: true, + }, + { + // groups with too large id + groups: []string{strconv.Itoa(1 << 31)}, + expected: nil, + hasError: true, + }, + } + + for _, test := range tests { + group := strings.NewReader(groupContent) + + gids, err := GetAdditionalGroups(test.groups, group) + if test.hasError && err == nil { + t.Errorf("Parse(%#v) expects error but has none", test) + continue + } + if !test.hasError && err != nil { + t.Errorf("Parse(%#v) has error %v", test, err) + continue + } + sort.Sort(sort.IntSlice(gids)) + if !reflect.DeepEqual(gids, test.expected) { + t.Errorf("Gids(%v), expect %v from groups %v", gids, test.expected, test.groups) + } + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image.go new file mode 100644 index 0000000000000..4bceb0d3de167 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image.go @@ -0,0 +1,537 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "bytes" + "encoding/base64" + "encoding/json" + "errors" + "fmt" + "io" + "io/ioutil" + "net/http" + "net/url" + "os" + "time" +) + +// APIImages represent an image returned in the ListImages call. +type APIImages struct { + ID string `json:"Id" yaml:"Id"` + RepoTags []string `json:"RepoTags,omitempty" yaml:"RepoTags,omitempty"` + Created int64 `json:"Created,omitempty" yaml:"Created,omitempty"` + Size int64 `json:"Size,omitempty" yaml:"Size,omitempty"` + VirtualSize int64 `json:"VirtualSize,omitempty" yaml:"VirtualSize,omitempty"` + ParentID string `json:"ParentId,omitempty" yaml:"ParentId,omitempty"` + RepoDigests []string `json:"RepoDigests,omitempty" yaml:"RepoDigests,omitempty"` + Labels map[string]string `json:"Labels,omitempty" yaml:"Labels,omitempty"` +} + +// Image is the type representing a docker image and its various properties +type Image struct { + ID string `json:"Id" yaml:"Id"` + Parent string `json:"Parent,omitempty" yaml:"Parent,omitempty"` + Comment string `json:"Comment,omitempty" yaml:"Comment,omitempty"` + Created time.Time `json:"Created,omitempty" yaml:"Created,omitempty"` + Container string `json:"Container,omitempty" yaml:"Container,omitempty"` + ContainerConfig Config `json:"ContainerConfig,omitempty" yaml:"ContainerConfig,omitempty"` + DockerVersion string `json:"DockerVersion,omitempty" yaml:"DockerVersion,omitempty"` + Author string `json:"Author,omitempty" yaml:"Author,omitempty"` + Config *Config `json:"Config,omitempty" yaml:"Config,omitempty"` + Architecture string `json:"Architecture,omitempty" yaml:"Architecture,omitempty"` + Size int64 `json:"Size,omitempty" yaml:"Size,omitempty"` + VirtualSize int64 `json:"VirtualSize,omitempty" yaml:"VirtualSize,omitempty"` +} + +// ImageHistory represent a layer in an image's history returned by the +// ImageHistory call. +type ImageHistory struct { + ID string `json:"Id" yaml:"Id"` + Tags []string `json:"Tags,omitempty" yaml:"Tags,omitempty"` + Created int64 `json:"Created,omitempty" yaml:"Created,omitempty"` + CreatedBy string `json:"CreatedBy,omitempty" yaml:"CreatedBy,omitempty"` + Size int64 `json:"Size,omitempty" yaml:"Size,omitempty"` +} + +// ImagePre012 serves the same purpose as the Image type except that it is for +// earlier versions of the Docker API (pre-012 to be specific) +type ImagePre012 struct { + ID string `json:"id"` + Parent string `json:"parent,omitempty"` + Comment string `json:"comment,omitempty"` + Created time.Time `json:"created"` + Container string `json:"container,omitempty"` + ContainerConfig Config `json:"container_config,omitempty"` + DockerVersion string `json:"docker_version,omitempty"` + Author string `json:"author,omitempty"` + Config *Config `json:"config,omitempty"` + Architecture string `json:"architecture,omitempty"` + Size int64 `json:"size,omitempty"` +} + +// ListImagesOptions specify parameters to the ListImages function. +// +// See http://goo.gl/HRVN1Z for more details. +type ListImagesOptions struct { + All bool + Filters map[string][]string + Digests bool +} + +var ( + // ErrNoSuchImage is the error returned when the image does not exist. + ErrNoSuchImage = errors.New("no such image") + + // ErrMissingRepo is the error returned when the remote repository is + // missing. + ErrMissingRepo = errors.New("missing remote repository e.g. 'github.com/user/repo'") + + // ErrMissingOutputStream is the error returned when no output stream + // is provided to some calls, like BuildImage. + ErrMissingOutputStream = errors.New("missing output stream") + + // ErrMultipleContexts is the error returned when both a ContextDir and + // InputStream are provided in BuildImageOptions + ErrMultipleContexts = errors.New("image build may not be provided BOTH context dir and input stream") + + // ErrMustSpecifyNames is the error rreturned when the Names field on + // ExportImagesOptions is nil or empty + ErrMustSpecifyNames = errors.New("must specify at least one name to export") +) + +// ListImages returns the list of available images in the server. +// +// See http://goo.gl/HRVN1Z for more details. +func (c *Client) ListImages(opts ListImagesOptions) ([]APIImages, error) { + path := "/images/json?" + queryString(opts) + body, _, err := c.do("GET", path, doOptions{}) + if err != nil { + return nil, err + } + var images []APIImages + err = json.Unmarshal(body, &images) + if err != nil { + return nil, err + } + return images, nil +} + +// ImageHistory returns the history of the image by its name or ID. +// +// See http://goo.gl/2oJmNs for more details. +func (c *Client) ImageHistory(name string) ([]ImageHistory, error) { + body, status, err := c.do("GET", "/images/"+name+"/history", doOptions{}) + if status == http.StatusNotFound { + return nil, ErrNoSuchImage + } + if err != nil { + return nil, err + } + var history []ImageHistory + err = json.Unmarshal(body, &history) + if err != nil { + return nil, err + } + return history, nil +} + +// RemoveImage removes an image by its name or ID. +// +// See http://goo.gl/znj0wM for more details. +func (c *Client) RemoveImage(name string) error { + _, status, err := c.do("DELETE", "/images/"+name, doOptions{}) + if status == http.StatusNotFound { + return ErrNoSuchImage + } + return err +} + +// RemoveImageOptions present the set of options available for removing an image +// from a registry. +// +// See http://goo.gl/6V48bF for more details. +type RemoveImageOptions struct { + Force bool `qs:"force"` + NoPrune bool `qs:"noprune"` +} + +// RemoveImageExtended removes an image by its name or ID. +// Extra params can be passed, see RemoveImageOptions +// +// See http://goo.gl/znj0wM for more details. +func (c *Client) RemoveImageExtended(name string, opts RemoveImageOptions) error { + uri := fmt.Sprintf("/images/%s?%s", name, queryString(&opts)) + _, status, err := c.do("DELETE", uri, doOptions{}) + if status == http.StatusNotFound { + return ErrNoSuchImage + } + return err +} + +// InspectImage returns an image by its name or ID. +// +// See http://goo.gl/Q112NY for more details. +func (c *Client) InspectImage(name string) (*Image, error) { + body, status, err := c.do("GET", "/images/"+name+"/json", doOptions{}) + if status == http.StatusNotFound { + return nil, ErrNoSuchImage + } + if err != nil { + return nil, err + } + + var image Image + + // if the caller elected to skip checking the server's version, assume it's the latest + if c.SkipServerVersionCheck || c.expectedAPIVersion.GreaterThanOrEqualTo(apiVersion112) { + err = json.Unmarshal(body, &image) + if err != nil { + return nil, err + } + } else { + var imagePre012 ImagePre012 + err = json.Unmarshal(body, &imagePre012) + if err != nil { + return nil, err + } + + image.ID = imagePre012.ID + image.Parent = imagePre012.Parent + image.Comment = imagePre012.Comment + image.Created = imagePre012.Created + image.Container = imagePre012.Container + image.ContainerConfig = imagePre012.ContainerConfig + image.DockerVersion = imagePre012.DockerVersion + image.Author = imagePre012.Author + image.Config = imagePre012.Config + image.Architecture = imagePre012.Architecture + image.Size = imagePre012.Size + } + + return &image, nil +} + +// PushImageOptions represents options to use in the PushImage method. +// +// See http://goo.gl/pN8A3P for more details. +type PushImageOptions struct { + // Name of the image + Name string + + // Tag of the image + Tag string + + // Registry server to push the image + Registry string + + OutputStream io.Writer `qs:"-"` + RawJSONStream bool `qs:"-"` +} + +// PushImage pushes an image to a remote registry, logging progress to w. +// +// An empty instance of AuthConfiguration may be used for unauthenticated +// pushes. +// +// See http://goo.gl/pN8A3P for more details. +func (c *Client) PushImage(opts PushImageOptions, auth AuthConfiguration) error { + if opts.Name == "" { + return ErrNoSuchImage + } + headers, err := headersWithAuth(auth) + if err != nil { + return err + } + name := opts.Name + opts.Name = "" + path := "/images/" + name + "/push?" + queryString(&opts) + return c.stream("POST", path, streamOptions{ + setRawTerminal: true, + rawJSONStream: opts.RawJSONStream, + headers: headers, + stdout: opts.OutputStream, + }) +} + +// PullImageOptions present the set of options available for pulling an image +// from a registry. +// +// See http://goo.gl/ACyYNS for more details. +type PullImageOptions struct { + Repository string `qs:"fromImage"` + Registry string + Tag string + OutputStream io.Writer `qs:"-"` + RawJSONStream bool `qs:"-"` +} + +// PullImage pulls an image from a remote registry, logging progress to opts.OutputStream. +// +// See http://goo.gl/ACyYNS for more details. +func (c *Client) PullImage(opts PullImageOptions, auth AuthConfiguration) error { + if opts.Repository == "" { + return ErrNoSuchImage + } + + headers, err := headersWithAuth(auth) + if err != nil { + return err + } + return c.createImage(queryString(&opts), headers, nil, opts.OutputStream, opts.RawJSONStream) +} + +func (c *Client) createImage(qs string, headers map[string]string, in io.Reader, w io.Writer, rawJSONStream bool) error { + path := "/images/create?" + qs + return c.stream("POST", path, streamOptions{ + setRawTerminal: true, + rawJSONStream: rawJSONStream, + headers: headers, + in: in, + stdout: w, + }) +} + +// LoadImageOptions represents the options for LoadImage Docker API Call +// +// See http://goo.gl/Y8NNCq for more details. +type LoadImageOptions struct { + InputStream io.Reader +} + +// LoadImage imports a tarball docker image +// +// See http://goo.gl/Y8NNCq for more details. +func (c *Client) LoadImage(opts LoadImageOptions) error { + return c.stream("POST", "/images/load", streamOptions{ + setRawTerminal: true, + in: opts.InputStream, + }) +} + +// ExportImageOptions represent the options for ExportImage Docker API call +// +// See http://goo.gl/mi6kvk for more details. +type ExportImageOptions struct { + Name string + OutputStream io.Writer +} + +// ExportImage exports an image (as a tar file) into the stream +// +// See http://goo.gl/mi6kvk for more details. +func (c *Client) ExportImage(opts ExportImageOptions) error { + return c.stream("GET", fmt.Sprintf("/images/%s/get", opts.Name), streamOptions{ + setRawTerminal: true, + stdout: opts.OutputStream, + }) +} + +// ExportImagesOptions represent the options for ExportImages Docker API call +// +// See http://goo.gl/YeZzQK for more details. +type ExportImagesOptions struct { + Names []string + OutputStream io.Writer `qs:"-"` +} + +// ExportImages exports one or more images (as a tar file) into the stream +// +// See http://goo.gl/YeZzQK for more details. +func (c *Client) ExportImages(opts ExportImagesOptions) error { + if opts.Names == nil || len(opts.Names) == 0 { + return ErrMustSpecifyNames + } + return c.stream("GET", "/images/get?"+queryString(&opts), streamOptions{ + setRawTerminal: true, + stdout: opts.OutputStream, + }) +} + +// ImportImageOptions present the set of informations available for importing +// an image from a source file or the stdin. +// +// See http://goo.gl/PhBKnS for more details. +type ImportImageOptions struct { + Repository string `qs:"repo"` + Source string `qs:"fromSrc"` + Tag string `qs:"tag"` + + InputStream io.Reader `qs:"-"` + OutputStream io.Writer `qs:"-"` + RawJSONStream bool `qs:"-"` +} + +// ImportImage imports an image from a url, a file or stdin +// +// See http://goo.gl/PhBKnS for more details. +func (c *Client) ImportImage(opts ImportImageOptions) error { + if opts.Repository == "" { + return ErrNoSuchImage + } + if opts.Source != "-" { + opts.InputStream = nil + } + if opts.Source != "-" && !isURL(opts.Source) { + f, err := os.Open(opts.Source) + if err != nil { + return err + } + b, err := ioutil.ReadAll(f) + opts.InputStream = bytes.NewBuffer(b) + opts.Source = "-" + } + return c.createImage(queryString(&opts), nil, opts.InputStream, opts.OutputStream, opts.RawJSONStream) +} + +// BuildImageOptions present the set of informations available for building an +// image from a tarfile with a Dockerfile in it. +// +// For more details about the Docker building process, see +// http://goo.gl/tlPXPu. +type BuildImageOptions struct { + Name string `qs:"t"` + Dockerfile string `qs:"dockerfile"` + NoCache bool `qs:"nocache"` + SuppressOutput bool `qs:"q"` + Pull bool `qs:"pull"` + RmTmpContainer bool `qs:"rm"` + ForceRmTmpContainer bool `qs:"forcerm"` + Memory int64 `qs:"memory"` + Memswap int64 `qs:"memswap"` + CPUShares int64 `qs:"cpushares"` + CPUSetCPUs string `qs:"cpusetcpus"` + InputStream io.Reader `qs:"-"` + OutputStream io.Writer `qs:"-"` + RawJSONStream bool `qs:"-"` + Remote string `qs:"remote"` + Auth AuthConfiguration `qs:"-"` // for older docker X-Registry-Auth header + AuthConfigs AuthConfigurations `qs:"-"` // for newer docker X-Registry-Config header + ContextDir string `qs:"-"` +} + +// BuildImage builds an image from a tarball's url or a Dockerfile in the input +// stream. +// +// See http://goo.gl/7nuGXa for more details. +func (c *Client) BuildImage(opts BuildImageOptions) error { + if opts.OutputStream == nil { + return ErrMissingOutputStream + } + headers, err := headersWithAuth(opts.Auth, opts.AuthConfigs) + if err != nil { + return err + } + + if opts.Remote != "" && opts.Name == "" { + opts.Name = opts.Remote + } + if opts.InputStream != nil || opts.ContextDir != "" { + headers["Content-Type"] = "application/tar" + } else if opts.Remote == "" { + return ErrMissingRepo + } + if opts.ContextDir != "" { + if opts.InputStream != nil { + return ErrMultipleContexts + } + var err error + if opts.InputStream, err = createTarStream(opts.ContextDir, opts.Dockerfile); err != nil { + return err + } + } + + return c.stream("POST", fmt.Sprintf("/build?%s", queryString(&opts)), streamOptions{ + setRawTerminal: true, + rawJSONStream: opts.RawJSONStream, + headers: headers, + in: opts.InputStream, + stdout: opts.OutputStream, + }) +} + +// TagImageOptions present the set of options to tag an image. +// +// See http://goo.gl/5g6qFy for more details. +type TagImageOptions struct { + Repo string + Tag string + Force bool +} + +// TagImage adds a tag to the image identified by the given name. +// +// See http://goo.gl/5g6qFy for more details. +func (c *Client) TagImage(name string, opts TagImageOptions) error { + if name == "" { + return ErrNoSuchImage + } + _, status, err := c.do("POST", fmt.Sprintf("/images/"+name+"/tag?%s", + queryString(&opts)), doOptions{}) + + if status == http.StatusNotFound { + return ErrNoSuchImage + } + + return err +} + +func isURL(u string) bool { + p, err := url.Parse(u) + if err != nil { + return false + } + return p.Scheme == "http" || p.Scheme == "https" +} + +func headersWithAuth(auths ...interface{}) (map[string]string, error) { + var headers = make(map[string]string) + + for _, auth := range auths { + switch auth.(type) { + case AuthConfiguration: + var buf bytes.Buffer + if err := json.NewEncoder(&buf).Encode(auth); err != nil { + return nil, err + } + headers["X-Registry-Auth"] = base64.URLEncoding.EncodeToString(buf.Bytes()) + case AuthConfigurations: + var buf bytes.Buffer + if err := json.NewEncoder(&buf).Encode(auth); err != nil { + return nil, err + } + headers["X-Registry-Config"] = base64.URLEncoding.EncodeToString(buf.Bytes()) + } + } + + return headers, nil +} + +// APIImageSearch reflect the result of a search on the dockerHub +// +// See http://goo.gl/xI5lLZ for more details. +type APIImageSearch struct { + Description string `json:"description,omitempty" yaml:"description,omitempty"` + IsOfficial bool `json:"is_official,omitempty" yaml:"is_official,omitempty"` + IsAutomated bool `json:"is_automated,omitempty" yaml:"is_automated,omitempty"` + Name string `json:"name,omitempty" yaml:"name,omitempty"` + StarCount int `json:"star_count,omitempty" yaml:"star_count,omitempty"` +} + +// SearchImages search the docker hub with a specific given term. +// +// See http://goo.gl/xI5lLZ for more details. +func (c *Client) SearchImages(term string) ([]APIImageSearch, error) { + body, _, err := c.do("GET", "/images/search?term="+term, doOptions{}) + if err != nil { + return nil, err + } + var searchResult []APIImageSearch + err = json.Unmarshal(body, &searchResult) + if err != nil { + return nil, err + } + return searchResult, nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image_test.go new file mode 100644 index 0000000000000..d6bce64cfc842 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/image_test.go @@ -0,0 +1,967 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "bytes" + "encoding/base64" + "encoding/json" + "io/ioutil" + "net/http" + "net/url" + "os" + "reflect" + "strings" + "testing" + "time" +) + +func newTestClient(rt *FakeRoundTripper) Client { + endpoint := "http://localhost:4243" + u, _ := parseEndpoint("http://localhost:4243", false) + client := Client{ + HTTPClient: &http.Client{Transport: rt}, + endpoint: endpoint, + endpointURL: u, + SkipServerVersionCheck: true, + } + return client +} + +type stdoutMock struct { + *bytes.Buffer +} + +func (m stdoutMock) Close() error { + return nil +} + +type stdinMock struct { + *bytes.Buffer +} + +func (m stdinMock) Close() error { + return nil +} + +func TestListImages(t *testing.T) { + body := `[ + { + "Repository":"base", + "Tag":"ubuntu-12.10", + "Id":"b750fe79269d", + "Created":1364102658 + }, + { + "Repository":"base", + "Tag":"ubuntu-quantal", + "Id":"b750fe79269d", + "Created":1364102658 + }, + { + "RepoTag": [ + "ubuntu:12.04", + "ubuntu:precise", + "ubuntu:latest" + ], + "Id": "8dbd9e392a964c", + "Created": 1365714795, + "Size": 131506275, + "VirtualSize": 131506275 + }, + { + "RepoTag": [ + "ubuntu:12.10", + "ubuntu:quantal" + ], + "ParentId": "27cf784147099545", + "Id": "b750fe79269d2e", + "Created": 1364102658, + "Size": 24653, + "VirtualSize": 180116135 + } +]` + var expected []APIImages + err := json.Unmarshal([]byte(body), &expected) + if err != nil { + t.Fatal(err) + } + client := newTestClient(&FakeRoundTripper{message: body, status: http.StatusOK}) + images, err := client.ListImages(ListImagesOptions{}) + if err != nil { + t.Error(err) + } + if !reflect.DeepEqual(images, expected) { + t.Errorf("ListImages: Wrong return value. Want %#v. Got %#v.", expected, images) + } +} + +func TestListImagesParameters(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "null", status: http.StatusOK} + client := newTestClient(fakeRT) + _, err := client.ListImages(ListImagesOptions{All: false}) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + if req.Method != "GET" { + t.Errorf("ListImages({All: false}: Wrong HTTP method. Want GET. Got %s.", req.Method) + } + if all := req.URL.Query().Get("all"); all != "0" && all != "" { + t.Errorf("ListImages({All: false}): Wrong parameter. Want all=0 or not present at all. Got all=%s", all) + } + fakeRT.Reset() + _, err = client.ListImages(ListImagesOptions{All: true}) + if err != nil { + t.Fatal(err) + } + req = fakeRT.requests[0] + if all := req.URL.Query().Get("all"); all != "1" { + t.Errorf("ListImages({All: true}): Wrong parameter. Want all=1. Got all=%s", all) + } + fakeRT.Reset() + _, err = client.ListImages(ListImagesOptions{Filters: map[string][]string{ + "dangling": {"true"}, + }}) + if err != nil { + t.Fatal(err) + } + req = fakeRT.requests[0] + body := req.URL.Query().Get("filters") + var filters map[string][]string + err = json.Unmarshal([]byte(body), &filters) + if err != nil { + t.Fatal(err) + } + if len(filters["dangling"]) != 1 || filters["dangling"][0] != "true" { + t.Errorf("ListImages(dangling=[true]): Wrong filter map. Want dangling=[true], got dangling=%v", filters["dangling"]) + } +} + +func TestImageHistory(t *testing.T) { + body := `[ + { + "Id": "25daec02219d2d852f7526137213a9b199926b4b24e732eab5b8bc6c49bd470e", + "Tags": [ + "debian:7.6", + "debian:latest", + "debian:7", + "debian:wheezy" + ], + "Created": 1409856216, + "CreatedBy": "/bin/sh -c #(nop) CMD [/bin/bash]" + }, + { + "Id": "41026a5347fb5be6ed16115bf22df8569697139f246186de9ae8d4f67c335dce", + "Created": 1409856213, + "CreatedBy": "/bin/sh -c #(nop) ADD file:1ee9e97209d00e3416a4543b23574cc7259684741a46bbcbc755909b8a053a38 in /", + "Size": 85178663 + }, + { + "Id": "511136ea3c5a64f264b78b5433614aec563103b4d4702f3ba7d4d2698e22c158", + "Tags": [ + "scratch:latest" + ], + "Created": 1371157430 + } +]` + var expected []ImageHistory + err := json.Unmarshal([]byte(body), &expected) + if err != nil { + t.Fatal(err) + } + client := newTestClient(&FakeRoundTripper{message: body, status: http.StatusOK}) + history, err := client.ImageHistory("debian:latest") + if err != nil { + t.Error(err) + } + if !reflect.DeepEqual(history, expected) { + t.Errorf("ImageHistory: Wrong return value. Want %#v. Got %#v.", expected, history) + } +} + +func TestRemoveImage(t *testing.T) { + name := "test" + fakeRT := &FakeRoundTripper{message: "", status: http.StatusNoContent} + client := newTestClient(fakeRT) + err := client.RemoveImage(name) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + expectedMethod := "DELETE" + if req.Method != expectedMethod { + t.Errorf("RemoveImage(%q): Wrong HTTP method. Want %s. Got %s.", name, expectedMethod, req.Method) + } + u, _ := url.Parse(client.getURL("/images/" + name)) + if req.URL.Path != u.Path { + t.Errorf("RemoveImage(%q): Wrong request path. Want %q. Got %q.", name, u.Path, req.URL.Path) + } +} + +func TestRemoveImageNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such image", status: http.StatusNotFound}) + err := client.RemoveImage("test:") + if err != ErrNoSuchImage { + t.Errorf("RemoveImage: wrong error. Want %#v. Got %#v.", ErrNoSuchImage, err) + } +} + +func TestRemoveImageExtended(t *testing.T) { + name := "test" + fakeRT := &FakeRoundTripper{message: "", status: http.StatusNoContent} + client := newTestClient(fakeRT) + err := client.RemoveImageExtended(name, RemoveImageOptions{Force: true, NoPrune: true}) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + expectedMethod := "DELETE" + if req.Method != expectedMethod { + t.Errorf("RemoveImage(%q): Wrong HTTP method. Want %s. Got %s.", name, expectedMethod, req.Method) + } + u, _ := url.Parse(client.getURL("/images/" + name)) + if req.URL.Path != u.Path { + t.Errorf("RemoveImage(%q): Wrong request path. Want %q. Got %q.", name, u.Path, req.URL.Path) + } + expectedQuery := "force=1&noprune=1" + if query := req.URL.Query().Encode(); query != expectedQuery { + t.Errorf("PushImage: Wrong query string. Want %q. Got %q.", expectedQuery, query) + } +} + +func TestInspectImage(t *testing.T) { + body := `{ + "Id":"b750fe79269d2ec9a3c593ef05b4332b1d1a02a62b4accb2c21d589ff2f5f2dc", + "Parent":"27cf784147099545", + "Created":"2013-03-23T22:24:18.818426Z", + "Container":"3d67245a8d72ecf13f33dffac9f79dcdf70f75acb84d308770391510e0c23ad0", + "ContainerConfig":{"Memory":1}, + "VirtualSize":12345 +}` + + created, err := time.Parse(time.RFC3339Nano, "2013-03-23T22:24:18.818426Z") + if err != nil { + t.Fatal(err) + } + + expected := Image{ + ID: "b750fe79269d2ec9a3c593ef05b4332b1d1a02a62b4accb2c21d589ff2f5f2dc", + Parent: "27cf784147099545", + Created: created, + Container: "3d67245a8d72ecf13f33dffac9f79dcdf70f75acb84d308770391510e0c23ad0", + ContainerConfig: Config{ + Memory: 1, + }, + VirtualSize: 12345, + } + fakeRT := &FakeRoundTripper{message: body, status: http.StatusOK} + client := newTestClient(fakeRT) + image, err := client.InspectImage(expected.ID) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(*image, expected) { + t.Errorf("InspectImage(%q): Wrong image returned. Want %#v. Got %#v.", expected.ID, expected, *image) + } + req := fakeRT.requests[0] + if req.Method != "GET" { + t.Errorf("InspectImage(%q): Wrong HTTP method. Want GET. Got %s.", expected.ID, req.Method) + } + u, _ := url.Parse(client.getURL("/images/" + expected.ID + "/json")) + if req.URL.Path != u.Path { + t.Errorf("InspectImage(%q): Wrong request URL. Want %q. Got %q.", expected.ID, u.Path, req.URL.Path) + } +} + +func TestInspectImageNotFound(t *testing.T) { + client := newTestClient(&FakeRoundTripper{message: "no such image", status: http.StatusNotFound}) + name := "test" + image, err := client.InspectImage(name) + if image != nil { + t.Errorf("InspectImage(%q): expected image, got %#v.", name, image) + } + if err != ErrNoSuchImage { + t.Errorf("InspectImage(%q): wrong error. Want %#v. Got %#v.", name, ErrNoSuchImage, err) + } +} + +func TestPushImage(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "Pushing 1/100", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + err := client.PushImage(PushImageOptions{Name: "test", OutputStream: &buf}, AuthConfiguration{}) + if err != nil { + t.Fatal(err) + } + expected := "Pushing 1/100" + if buf.String() != expected { + t.Errorf("PushImage: Wrong output. Want %q. Got %q.", expected, buf.String()) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("PushImage: Wrong HTTP method. Want POST. Got %s.", req.Method) + } + u, _ := url.Parse(client.getURL("/images/test/push")) + if req.URL.Path != u.Path { + t.Errorf("PushImage: Wrong request path. Want %q. Got %q.", u.Path, req.URL.Path) + } + if query := req.URL.Query().Encode(); query != "" { + t.Errorf("PushImage: Wrong query string. Want no parameters, got %q.", query) + } + + auth, err := base64.URLEncoding.DecodeString(req.Header.Get("X-Registry-Auth")) + if err != nil { + t.Errorf("PushImage: caught error decoding auth. %#v", err.Error()) + } + if strings.TrimSpace(string(auth)) != "{}" { + t.Errorf("PushImage: wrong body. Want %q. Got %q.", + base64.URLEncoding.EncodeToString([]byte("{}")), req.Header.Get("X-Registry-Auth")) + } +} + +func TestPushImageWithRawJSON(t *testing.T) { + body := ` + {"status":"Pushing..."} + {"status":"Pushing", "progress":"1/? (n/a)", "progressDetail":{"current":1}}} + {"status":"Image successfully pushed"} + ` + fakeRT := &FakeRoundTripper{ + message: body, + status: http.StatusOK, + header: map[string]string{ + "Content-Type": "application/json", + }, + } + client := newTestClient(fakeRT) + var buf bytes.Buffer + + err := client.PushImage(PushImageOptions{ + Name: "test", + OutputStream: &buf, + RawJSONStream: true, + }, AuthConfiguration{}) + if err != nil { + t.Fatal(err) + } + if buf.String() != body { + t.Errorf("PushImage: Wrong raw output. Want %q. Got %q.", body, buf.String()) + } +} + +func TestPushImageWithAuthentication(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "Pushing 1/100", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + inputAuth := AuthConfiguration{ + Username: "gopher", + Password: "gopher123", + Email: "gopher@tsuru.io", + } + err := client.PushImage(PushImageOptions{Name: "test", OutputStream: &buf}, inputAuth) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + var gotAuth AuthConfiguration + + auth, err := base64.URLEncoding.DecodeString(req.Header.Get("X-Registry-Auth")) + if err != nil { + t.Errorf("PushImage: caught error decoding auth. %#v", err.Error()) + } + + err = json.Unmarshal(auth, &gotAuth) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(gotAuth, inputAuth) { + t.Errorf("PushImage: wrong auth configuration. Want %#v. Got %#v.", inputAuth, gotAuth) + } +} + +func TestPushImageCustomRegistry(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "Pushing 1/100", status: http.StatusOK} + client := newTestClient(fakeRT) + var authConfig AuthConfiguration + var buf bytes.Buffer + opts := PushImageOptions{ + Name: "test", Registry: "docker.tsuru.io", + OutputStream: &buf, + } + err := client.PushImage(opts, authConfig) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + expectedQuery := "registry=docker.tsuru.io" + if query := req.URL.Query().Encode(); query != expectedQuery { + t.Errorf("PushImage: Wrong query string. Want %q. Got %q.", expectedQuery, query) + } +} + +func TestPushImageNoName(t *testing.T) { + client := Client{} + err := client.PushImage(PushImageOptions{}, AuthConfiguration{}) + if err != ErrNoSuchImage { + t.Errorf("PushImage: got wrong error. Want %#v. Got %#v.", ErrNoSuchImage, err) + } +} + +func TestPullImage(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "Pulling 1/100", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + err := client.PullImage(PullImageOptions{Repository: "base", OutputStream: &buf}, + AuthConfiguration{}) + if err != nil { + t.Fatal(err) + } + expected := "Pulling 1/100" + if buf.String() != expected { + t.Errorf("PullImage: Wrong output. Want %q. Got %q.", expected, buf.String()) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("PullImage: Wrong HTTP method. Want POST. Got %s.", req.Method) + } + u, _ := url.Parse(client.getURL("/images/create")) + if req.URL.Path != u.Path { + t.Errorf("PullImage: Wrong request path. Want %q. Got %q.", u.Path, req.URL.Path) + } + expectedQuery := "fromImage=base" + if query := req.URL.Query().Encode(); query != expectedQuery { + t.Errorf("PullImage: Wrong query strin. Want %q. Got %q.", expectedQuery, query) + } +} + +func TestPullImageWithRawJSON(t *testing.T) { + body := ` + {"status":"Pulling..."} + {"status":"Pulling", "progress":"1 B/ 100 B", "progressDetail":{"current":1, "total":100}} + ` + fakeRT := &FakeRoundTripper{ + message: body, + status: http.StatusOK, + header: map[string]string{ + "Content-Type": "application/json", + }, + } + client := newTestClient(fakeRT) + var buf bytes.Buffer + err := client.PullImage(PullImageOptions{ + Repository: "base", + OutputStream: &buf, + RawJSONStream: true, + }, AuthConfiguration{}) + if err != nil { + t.Fatal(err) + } + if buf.String() != body { + t.Errorf("PullImage: Wrong raw output. Want %q. Got %q", body, buf.String()) + } +} + +func TestPullImageWithoutOutputStream(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "Pulling 1/100", status: http.StatusOK} + client := newTestClient(fakeRT) + opts := PullImageOptions{ + Repository: "base", + Registry: "docker.tsuru.io", + } + err := client.PullImage(opts, AuthConfiguration{}) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + expected := map[string][]string{"fromImage": {"base"}, "registry": {"docker.tsuru.io"}} + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expected) { + t.Errorf("PullImage: wrong query string. Want %#v. Got %#v.", expected, got) + } +} + +func TestPullImageCustomRegistry(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "Pulling 1/100", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + opts := PullImageOptions{ + Repository: "base", + Registry: "docker.tsuru.io", + OutputStream: &buf, + } + err := client.PullImage(opts, AuthConfiguration{}) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + expected := map[string][]string{"fromImage": {"base"}, "registry": {"docker.tsuru.io"}} + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expected) { + t.Errorf("PullImage: wrong query string. Want %#v. Got %#v.", expected, got) + } +} + +func TestPullImageTag(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "Pulling 1/100", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + opts := PullImageOptions{ + Repository: "base", + Registry: "docker.tsuru.io", + Tag: "latest", + OutputStream: &buf, + } + err := client.PullImage(opts, AuthConfiguration{}) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + expected := map[string][]string{"fromImage": {"base"}, "registry": {"docker.tsuru.io"}, "tag": {"latest"}} + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expected) { + t.Errorf("PullImage: wrong query string. Want %#v. Got %#v.", expected, got) + } +} + +func TestPullImageNoRepository(t *testing.T) { + var opts PullImageOptions + client := Client{} + err := client.PullImage(opts, AuthConfiguration{}) + if err != ErrNoSuchImage { + t.Errorf("PullImage: got wrong error. Want %#v. Got %#v.", ErrNoSuchImage, err) + } +} + +func TestImportImageFromUrl(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + opts := ImportImageOptions{ + Source: "http://mycompany.com/file.tar", + Repository: "testimage", + Tag: "tag", + OutputStream: &buf, + } + err := client.ImportImage(opts) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + expected := map[string][]string{"fromSrc": {opts.Source}, "repo": {opts.Repository}, "tag": {opts.Tag}} + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expected) { + t.Errorf("ImportImage: wrong query string. Want %#v. Got %#v.", expected, got) + } +} + +func TestImportImageFromInput(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + in := bytes.NewBufferString("tar content") + var buf bytes.Buffer + opts := ImportImageOptions{ + Source: "-", Repository: "testimage", + InputStream: in, OutputStream: &buf, + Tag: "tag", + } + err := client.ImportImage(opts) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + expected := map[string][]string{"fromSrc": {opts.Source}, "repo": {opts.Repository}, "tag": {opts.Tag}} + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expected) { + t.Errorf("ImportImage: wrong query string. Want %#v. Got %#v.", expected, got) + } + body, err := ioutil.ReadAll(req.Body) + if err != nil { + t.Errorf("ImportImage: caugth error while reading body %#v", err.Error()) + } + e := "tar content" + if string(body) != e { + t.Errorf("ImportImage: wrong body. Want %#v. Got %#v.", e, string(body)) + } +} + +func TestImportImageDoesNotPassesInputIfSourceIsNotDash(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + in := bytes.NewBufferString("foo") + opts := ImportImageOptions{ + Source: "http://test.com/container.tar", Repository: "testimage", + InputStream: in, OutputStream: &buf, + } + err := client.ImportImage(opts) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + expected := map[string][]string{"fromSrc": {opts.Source}, "repo": {opts.Repository}} + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expected) { + t.Errorf("ImportImage: wrong query string. Want %#v. Got %#v.", expected, got) + } + body, err := ioutil.ReadAll(req.Body) + if err != nil { + t.Errorf("ImportImage: caugth error while reading body %#v", err.Error()) + } + if string(body) != "" { + t.Errorf("ImportImage: wrong body. Want nothing. Got %#v.", string(body)) + } +} + +func TestImportImageShouldPassTarContentToBodyWhenSourceIsFilePath(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + tarPath := "testing/data/container.tar" + opts := ImportImageOptions{ + Source: tarPath, Repository: "testimage", + OutputStream: &buf, + } + err := client.ImportImage(opts) + if err != nil { + t.Fatal(err) + } + tar, err := os.Open(tarPath) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + tarContent, err := ioutil.ReadAll(tar) + body, err := ioutil.ReadAll(req.Body) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(tarContent, body) { + t.Errorf("ImportImage: wrong body. Want %#v content. Got %#v.", tarPath, body) + } +} + +func TestImportImageShouldChangeSourceToDashWhenItsAFilePath(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + tarPath := "testing/data/container.tar" + opts := ImportImageOptions{ + Source: tarPath, Repository: "testimage", + OutputStream: &buf, + } + err := client.ImportImage(opts) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + expected := map[string][]string{"fromSrc": {"-"}, "repo": {opts.Repository}} + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expected) { + t.Errorf("ImportImage: wrong query string. Want %#v. Got %#v.", expected, got) + } +} + +func TestBuildImageParameters(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + opts := BuildImageOptions{ + Name: "testImage", + NoCache: true, + SuppressOutput: true, + Pull: true, + RmTmpContainer: true, + ForceRmTmpContainer: true, + Memory: 1024, + Memswap: 2048, + CPUShares: 10, + CPUSetCPUs: "0-3", + InputStream: &buf, + OutputStream: &buf, + } + err := client.BuildImage(opts) + if err != nil && strings.Index(err.Error(), "build image fail") == -1 { + t.Fatal(err) + } + req := fakeRT.requests[0] + expected := map[string][]string{ + "t": {opts.Name}, + "nocache": {"1"}, + "q": {"1"}, + "pull": {"1"}, + "rm": {"1"}, + "forcerm": {"1"}, + "memory": {"1024"}, + "memswap": {"2048"}, + "cpushares": {"10"}, + "cpusetcpus": {"0-3"}, + } + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expected) { + t.Errorf("BuildImage: wrong query string. Want %#v. Got %#v.", expected, got) + } +} + +func TestBuildImageParametersForRemoteBuild(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + opts := BuildImageOptions{ + Name: "testImage", + Remote: "testing/data/container.tar", + SuppressOutput: true, + OutputStream: &buf, + } + err := client.BuildImage(opts) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + expected := map[string][]string{"t": {opts.Name}, "remote": {opts.Remote}, "q": {"1"}} + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expected) { + t.Errorf("BuildImage: wrong query string. Want %#v. Got %#v.", expected, got) + } +} + +func TestBuildImageMissingRepoAndNilInput(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + opts := BuildImageOptions{ + Name: "testImage", + SuppressOutput: true, + OutputStream: &buf, + } + err := client.BuildImage(opts) + if err != ErrMissingRepo { + t.Errorf("BuildImage: wrong error returned. Want %#v. Got %#v.", ErrMissingRepo, err) + } +} + +func TestBuildImageMissingOutputStream(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + opts := BuildImageOptions{Name: "testImage"} + err := client.BuildImage(opts) + if err != ErrMissingOutputStream { + t.Errorf("BuildImage: wrong error returned. Want %#v. Got %#v.", ErrMissingOutputStream, err) + } +} + +func TestBuildImageWithRawJSON(t *testing.T) { + body := ` + {"stream":"Step 0 : FROM ubuntu:latest\n"} + {"stream":" ---\u003e 4300eb9d3c8d\n"} + {"stream":"Step 1 : MAINTAINER docker \n"} + {"stream":" ---\u003e Using cache\n"} + {"stream":" ---\u003e 3a3ed758c370\n"} + {"stream":"Step 2 : CMD /usr/bin/top\n"} + {"stream":" ---\u003e Running in 36b1479cc2e4\n"} + {"stream":" ---\u003e 4b6188aebe39\n"} + {"stream":"Removing intermediate container 36b1479cc2e4\n"} + {"stream":"Successfully built 4b6188aebe39\n"} + ` + fakeRT := &FakeRoundTripper{ + message: body, + status: http.StatusOK, + header: map[string]string{ + "Content-Type": "application/json", + }, + } + client := newTestClient(fakeRT) + var buf bytes.Buffer + opts := BuildImageOptions{ + Name: "testImage", + RmTmpContainer: true, + InputStream: &buf, + OutputStream: &buf, + RawJSONStream: true, + } + err := client.BuildImage(opts) + if err != nil { + t.Fatal(err) + } + if buf.String() != body { + t.Errorf("BuildImage: Wrong raw output. Want %q. Got %q.", body, buf.String()) + } +} + +func TestBuildImageRemoteWithoutName(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + var buf bytes.Buffer + opts := BuildImageOptions{ + Remote: "testing/data/container.tar", + SuppressOutput: true, + OutputStream: &buf, + } + err := client.BuildImage(opts) + if err != nil { + t.Fatal(err) + } + req := fakeRT.requests[0] + expected := map[string][]string{"t": {opts.Remote}, "remote": {opts.Remote}, "q": {"1"}} + got := map[string][]string(req.URL.Query()) + if !reflect.DeepEqual(got, expected) { + t.Errorf("BuildImage: wrong query string. Want %#v. Got %#v.", expected, got) + } +} + +func TestTagImageParameters(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + opts := TagImageOptions{Repo: "testImage"} + err := client.TagImage("base", opts) + if err != nil && strings.Index(err.Error(), "tag image fail") == -1 { + t.Fatal(err) + } + req := fakeRT.requests[0] + expected := "http://localhost:4243/images/base/tag?repo=testImage" + got := req.URL.String() + if !reflect.DeepEqual(got, expected) { + t.Errorf("TagImage: wrong query string. Want %#v. Got %#v.", expected, got) + } +} + +func TestTagImageMissingRepo(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + opts := TagImageOptions{Repo: "testImage"} + err := client.TagImage("", opts) + if err != ErrNoSuchImage { + t.Errorf("TestTag: wrong error returned. Want %#v. Got %#v.", + ErrNoSuchImage, err) + } +} + +func TestIsUrl(t *testing.T) { + url := "http://foo.bar/" + result := isURL(url) + if !result { + t.Errorf("isURL: wrong match. Expected %#v to be a url. Got %#v.", url, result) + } + url = "/foo/bar.tar" + result = isURL(url) + if result { + t.Errorf("isURL: wrong match. Expected %#v to not be a url. Got %#v", url, result) + } +} + +func TestLoadImage(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + tar, err := os.Open("testing/data/container.tar") + if err != nil { + t.Fatal(err) + } else { + defer tar.Close() + } + opts := LoadImageOptions{InputStream: tar} + err = client.LoadImage(opts) + if nil != err { + t.Error(err) + } + req := fakeRT.requests[0] + if req.Method != "POST" { + t.Errorf("LoadImage: wrong method. Expected %q. Got %q.", "POST", req.Method) + } + if req.URL.Path != "/images/load" { + t.Errorf("LoadImage: wrong URL. Expected %q. Got %q.", "/images/load", req.URL.Path) + } +} + +func TestExportImage(t *testing.T) { + var buf bytes.Buffer + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + opts := ExportImageOptions{Name: "testimage", OutputStream: &buf} + err := client.ExportImage(opts) + if nil != err { + t.Error(err) + } + req := fakeRT.requests[0] + if req.Method != "GET" { + t.Errorf("ExportImage: wrong method. Expected %q. Got %q.", "GET", req.Method) + } + expectedPath := "/images/testimage/get" + if req.URL.Path != expectedPath { + t.Errorf("ExportIMage: wrong path. Expected %q. Got %q.", expectedPath, req.URL.Path) + } +} + +func TestExportImages(t *testing.T) { + var buf bytes.Buffer + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + opts := ExportImagesOptions{Names: []string{"testimage1", "testimage2:latest"}, OutputStream: &buf} + err := client.ExportImages(opts) + if nil != err { + t.Error(err) + } + req := fakeRT.requests[0] + if req.Method != "GET" { + t.Errorf("ExportImage: wrong method. Expected %q. Got %q.", "GET", req.Method) + } + expected := "http://localhost:4243/images/get?names=testimage1&names=testimage2%3Alatest" + got := req.URL.String() + if !reflect.DeepEqual(got, expected) { + t.Errorf("ExportIMage: wrong path. Expected %q. Got %q.", expected, got) + } +} + +func TestExportImagesNoNames(t *testing.T) { + var buf bytes.Buffer + fakeRT := &FakeRoundTripper{message: "", status: http.StatusOK} + client := newTestClient(fakeRT) + opts := ExportImagesOptions{Names: []string{}, OutputStream: &buf} + err := client.ExportImages(opts) + if err == nil { + t.Error("Expected an error") + } + if err != ErrMustSpecifyNames { + t.Error(err) + } +} + +func TestSearchImages(t *testing.T) { + body := `[ + { + "description":"A container with Cassandra 2.0.3", + "is_official":true, + "is_automated":true, + "name":"poklet/cassandra", + "star_count":17 + }, + { + "description":"A container with Cassandra 2.0.3", + "is_official":true, + "is_automated":false, + "name":"poklet/cassandra", + "star_count":17 + } + , + { + "description":"A container with Cassandra 2.0.3", + "is_official":false, + "is_automated":true, + "name":"poklet/cassandra", + "star_count":17 + } +]` + var expected []APIImageSearch + err := json.Unmarshal([]byte(body), &expected) + if err != nil { + t.Fatal(err) + } + client := newTestClient(&FakeRoundTripper{message: body, status: http.StatusOK}) + result, err := client.SearchImages("cassandra") + if err != nil { + t.Error(err) + } + if !reflect.DeepEqual(result, expected) { + t.Errorf("SearchImages: Wrong return value. Want %#v. Got %#v.", expected, result) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/misc.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/misc.go new file mode 100644 index 0000000000000..42d1c7e48e31f --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/misc.go @@ -0,0 +1,59 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "bytes" + "strings" +) + +// Version returns version information about the docker server. +// +// See http://goo.gl/BOZrF5 for more details. +func (c *Client) Version() (*Env, error) { + body, _, err := c.do("GET", "/version", doOptions{}) + if err != nil { + return nil, err + } + var env Env + if err := env.Decode(bytes.NewReader(body)); err != nil { + return nil, err + } + return &env, nil +} + +// Info returns system-wide information about the Docker server. +// +// See http://goo.gl/wmqZsW for more details. +func (c *Client) Info() (*Env, error) { + body, _, err := c.do("GET", "/info", doOptions{}) + if err != nil { + return nil, err + } + var info Env + err = info.Decode(bytes.NewReader(body)) + if err != nil { + return nil, err + } + return &info, nil +} + +// ParseRepositoryTag gets the name of the repository and returns it splitted +// in two parts: the repository and the tag. +// +// Some examples: +// +// localhost.localdomain:5000/samalba/hipache:latest -> localhost.localdomain:5000/samalba/hipache, latest +// localhost.localdomain:5000/samalba/hipache -> localhost.localdomain:5000/samalba/hipache, "" +func ParseRepositoryTag(repoTag string) (repository string, tag string) { + n := strings.LastIndex(repoTag, ":") + if n < 0 { + return repoTag, "" + } + if tag := repoTag[n+1:]; !strings.Contains(tag, "/") { + return repoTag[:n], tag + } + return repoTag, "" +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/misc_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/misc_test.go new file mode 100644 index 0000000000000..ceaf076eda682 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/misc_test.go @@ -0,0 +1,159 @@ +// Copyright 2014 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "net/http" + "net/url" + "reflect" + "sort" + "testing" +) + +type DockerVersion struct { + Version string + GitCommit string + GoVersion string +} + +func TestVersion(t *testing.T) { + body := `{ + "Version":"0.2.2", + "GitCommit":"5a2a5cc+CHANGES", + "GoVersion":"go1.0.3" +}` + fakeRT := FakeRoundTripper{message: body, status: http.StatusOK} + client := newTestClient(&fakeRT) + expected := DockerVersion{ + Version: "0.2.2", + GitCommit: "5a2a5cc+CHANGES", + GoVersion: "go1.0.3", + } + version, err := client.Version() + if err != nil { + t.Fatal(err) + } + + if result := version.Get("Version"); result != expected.Version { + t.Errorf("Version(): Wrong result. Want %#v. Got %#v.", expected.Version, version.Get("Version")) + } + if result := version.Get("GitCommit"); result != expected.GitCommit { + t.Errorf("GitCommit(): Wrong result. Want %#v. Got %#v.", expected.GitCommit, version.Get("GitCommit")) + } + if result := version.Get("GoVersion"); result != expected.GoVersion { + t.Errorf("GoVersion(): Wrong result. Want %#v. Got %#v.", expected.GoVersion, version.Get("GoVersion")) + } + req := fakeRT.requests[0] + if req.Method != "GET" { + t.Errorf("Version(): wrong request method. Want GET. Got %s.", req.Method) + } + u, _ := url.Parse(client.getURL("/version")) + if req.URL.Path != u.Path { + t.Errorf("Version(): wrong request path. Want %q. Got %q.", u.Path, req.URL.Path) + } +} + +func TestVersionError(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "internal error", status: http.StatusInternalServerError} + client := newTestClient(fakeRT) + version, err := client.Version() + if version != nil { + t.Errorf("Version(): expected value, got %#v.", version) + } + if err == nil { + t.Error("Version(): unexpected error") + } +} + +func TestInfo(t *testing.T) { + body := `{ + "Containers":11, + "Images":16, + "Debug":0, + "NFd":11, + "NGoroutines":21, + "MemoryLimit":1, + "SwapLimit":0 +}` + fakeRT := FakeRoundTripper{message: body, status: http.StatusOK} + client := newTestClient(&fakeRT) + expected := Env{} + expected.SetInt("Containers", 11) + expected.SetInt("Images", 16) + expected.SetBool("Debug", false) + expected.SetInt("NFd", 11) + expected.SetInt("NGoroutines", 21) + expected.SetBool("MemoryLimit", true) + expected.SetBool("SwapLimit", false) + info, err := client.Info() + if err != nil { + t.Fatal(err) + } + infoSlice := []string(*info) + expectedSlice := []string(expected) + sort.Strings(infoSlice) + sort.Strings(expectedSlice) + if !reflect.DeepEqual(expectedSlice, infoSlice) { + t.Errorf("Info(): Wrong result.\nWant %#v.\nGot %#v.", expected, *info) + } + req := fakeRT.requests[0] + if req.Method != "GET" { + t.Errorf("Info(): Wrong HTTP method. Want GET. Got %s.", req.Method) + } + u, _ := url.Parse(client.getURL("/info")) + if req.URL.Path != u.Path { + t.Errorf("Info(): Wrong request path. Want %q. Got %q.", u.Path, req.URL.Path) + } +} + +func TestInfoError(t *testing.T) { + fakeRT := &FakeRoundTripper{message: "internal error", status: http.StatusInternalServerError} + client := newTestClient(fakeRT) + version, err := client.Info() + if version != nil { + t.Errorf("Info(): expected value, got %#v.", version) + } + if err == nil { + t.Error("Info(): unexpected error") + } +} + +func TestParseRepositoryTag(t *testing.T) { + var tests = []struct { + input string + expectedRepo string + expectedTag string + }{ + { + "localhost.localdomain:5000/samalba/hipache:latest", + "localhost.localdomain:5000/samalba/hipache", + "latest", + }, + { + "localhost.localdomain:5000/samalba/hipache", + "localhost.localdomain:5000/samalba/hipache", + "", + }, + { + "tsuru/python", + "tsuru/python", + "", + }, + { + "tsuru/python:2.7", + "tsuru/python", + "2.7", + }, + } + for _, tt := range tests { + repo, tag := ParseRepositoryTag(tt.input) + if repo != tt.expectedRepo { + t.Errorf("ParseRepositoryTag(%q): wrong repository. Want %q. Got %q", tt.input, tt.expectedRepo, repo) + } + if tag != tt.expectedTag { + t.Errorf("ParseRepositoryTag(%q): wrong tag. Want %q. Got %q", tt.input, tt.expectedTag, tag) + } + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/network.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/network.go new file mode 100644 index 0000000000000..0d3e2d43f2433 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/network.go @@ -0,0 +1,127 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "encoding/json" + "errors" + "fmt" + "net/http" +) + +// ErrNetworkAlreadyExists is the error returned by CreateNetwork when the +// network already exists. +var ErrNetworkAlreadyExists = errors.New("network already exists") + +// Network represents a network. +// +// See https://goo.gl/FDkCdQ for more details. +type Network struct { + Name string `json:"name"` + ID string `json:"id"` + Type string `json:"type"` + Endpoints []*Endpoint `json:"endpoints"` +} + +// Endpoint represents an endpoint. +// +// See https://goo.gl/FDkCdQ for more details. +type Endpoint struct { + Name string `json:"name"` + ID string `json:"id"` + Network string `json:"network"` +} + +// ListNetworks returns all networks. +// +// See https://goo.gl/4hCNtZ for more details. +func (c *Client) ListNetworks() ([]Network, error) { + body, _, err := c.do("GET", "/networks", doOptions{}) + if err != nil { + return nil, err + } + var networks []Network + if err := json.Unmarshal(body, &networks); err != nil { + return nil, err + } + return networks, nil +} + +// NetworkInfo returns information about a network by its ID. +// +// See https://goo.gl/4hCNtZ for more details. +func (c *Client) NetworkInfo(id string) (*Network, error) { + path := "/networks/" + id + body, status, err := c.do("GET", path, doOptions{}) + if status == http.StatusNotFound { + return nil, &NoSuchNetwork{ID: id} + } + if err != nil { + return nil, err + } + var network Network + if err := json.Unmarshal(body, &network); err != nil { + return nil, err + } + return &network, nil +} + +// CreateNetworkOptions specify parameters to the CreateNetwork function and +// (for now) is the expected body of the "create network" http request message +// +// See https://goo.gl/FDkCdQ for more details. +type CreateNetworkOptions struct { + Name string `json:"name"` + NetworkType string `json:"network_type"` + Options map[string]interface{} `json:"options"` +} + +// CreateNetwork creates a new network, returning the network instance, +// or an error in case of failure. +// +// See http://goo.gl/mErxNp for more details. +func (c *Client) CreateNetwork(opts CreateNetworkOptions) (*Network, error) { + body, status, err := c.do( + "POST", + "/networks", + doOptions{ + data: opts, + }, + ) + + if status == http.StatusConflict { + return nil, ErrNetworkAlreadyExists + } + if err != nil { + return nil, err + } + + type createNetworkResponse struct { + ID string + } + var ( + network Network + resp createNetworkResponse + ) + err = json.Unmarshal(body, &resp) + if err != nil { + return nil, err + } + + network.Name = opts.Name + network.ID = resp.ID + network.Type = opts.NetworkType + + return &network, nil +} + +// NoSuchNetwork is the error returned when a given network does not exist. +type NoSuchNetwork struct { + ID string +} + +func (err *NoSuchNetwork) Error() string { + return fmt.Sprintf("No such network: %s", err.ID) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/network_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/network_test.go new file mode 100644 index 0000000000000..970988cf4778d --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/network_test.go @@ -0,0 +1,96 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "encoding/json" + "net/http" + "net/url" + "reflect" + "testing" +) + +func TestListNetworks(t *testing.T) { + jsonNetworks := `[ + { + "ID": "8dfafdbc3a40", + "Name": "blah", + "Type": "bridge", + "Endpoints":[{"ID": "918c11c8288a", "Name": "dsafdsaf", "Network": "8dfafdbc3a40"}] + }, + { + "ID": "9fb1e39c", + "Name": "foo", + "Type": "bridge", + "Endpoints":[{"ID": "c080be979dda", "Name": "lllll2222", "Network": "9fb1e39c"}] + } +]` + var expected []Network + err := json.Unmarshal([]byte(jsonNetworks), &expected) + if err != nil { + t.Fatal(err) + } + client := newTestClient(&FakeRoundTripper{message: jsonNetworks, status: http.StatusOK}) + containers, err := client.ListNetworks() + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(containers, expected) { + t.Errorf("ListNetworks: Expected %#v. Got %#v.", expected, containers) + } +} + +func TestNetworkInfo(t *testing.T) { + jsonNetwork := `{ + "ID": "8dfafdbc3a40", + "Name": "blah", + "Type": "bridge", + "Endpoints":[{"ID": "918c11c8288a", "Name": "dsafdsaf", "Network": "8dfafdbc3a40"}] + }` + var expected Network + err := json.Unmarshal([]byte(jsonNetwork), &expected) + if err != nil { + t.Fatal(err) + } + fakeRT := &FakeRoundTripper{message: jsonNetwork, status: http.StatusOK} + client := newTestClient(fakeRT) + id := "8dfafdbc3a40" + network, err := client.NetworkInfo(id) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(*network, expected) { + t.Errorf("NetworkInfo(%q): Expected %#v. Got %#v.", id, expected, network) + } + expectedURL, _ := url.Parse(client.getURL("/networks/8dfafdbc3a40")) + if gotPath := fakeRT.requests[0].URL.Path; gotPath != expectedURL.Path { + t.Errorf("NetworkInfo(%q): Wrong path in request. Want %q. Got %q.", id, expectedURL.Path, gotPath) + } +} + +func TestNetworkCreate(t *testing.T) { + jsonID := `{"ID": "8dfafdbc3a40"}` + jsonNetwork := `{ + "ID": "8dfafdbc3a40", + "Name": "foobar", + "Type": "bridge" + }` + var expected Network + err := json.Unmarshal([]byte(jsonNetwork), &expected) + if err != nil { + t.Fatal(err) + } + + client := newTestClient(&FakeRoundTripper{message: jsonID, status: http.StatusOK}) + opts := CreateNetworkOptions{"foobar", "bridge", nil} + network, err := client.CreateNetwork(opts) + if err != nil { + t.Fatal(err) + } + + if !reflect.DeepEqual(*network, expected) { + t.Errorf("CreateNetwork: Expected %#v. Got %#v.", expected, network) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/signal.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/signal.go new file mode 100644 index 0000000000000..16aa00388fdd5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/signal.go @@ -0,0 +1,49 @@ +// Copyright 2014 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +// Signal represents a signal that can be send to the container on +// KillContainer call. +type Signal int + +// These values represent all signals available on Linux, where containers will +// be running. +const ( + SIGABRT = Signal(0x6) + SIGALRM = Signal(0xe) + SIGBUS = Signal(0x7) + SIGCHLD = Signal(0x11) + SIGCLD = Signal(0x11) + SIGCONT = Signal(0x12) + SIGFPE = Signal(0x8) + SIGHUP = Signal(0x1) + SIGILL = Signal(0x4) + SIGINT = Signal(0x2) + SIGIO = Signal(0x1d) + SIGIOT = Signal(0x6) + SIGKILL = Signal(0x9) + SIGPIPE = Signal(0xd) + SIGPOLL = Signal(0x1d) + SIGPROF = Signal(0x1b) + SIGPWR = Signal(0x1e) + SIGQUIT = Signal(0x3) + SIGSEGV = Signal(0xb) + SIGSTKFLT = Signal(0x10) + SIGSTOP = Signal(0x13) + SIGSYS = Signal(0x1f) + SIGTERM = Signal(0xf) + SIGTRAP = Signal(0x5) + SIGTSTP = Signal(0x14) + SIGTTIN = Signal(0x15) + SIGTTOU = Signal(0x16) + SIGUNUSED = Signal(0x1f) + SIGURG = Signal(0x17) + SIGUSR1 = Signal(0xa) + SIGUSR2 = Signal(0xc) + SIGVTALRM = Signal(0x1a) + SIGWINCH = Signal(0x1c) + SIGXCPU = Signal(0x18) + SIGXFSZ = Signal(0x19) +) diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tar.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tar.go new file mode 100644 index 0000000000000..48042cbda05b7 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tar.go @@ -0,0 +1,117 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package docker + +import ( + "fmt" + "io" + "io/ioutil" + "os" + "path" + "path/filepath" + "strings" + + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/archive" + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/fileutils" +) + +func createTarStream(srcPath, dockerfilePath string) (io.ReadCloser, error) { + excludes, err := parseDockerignore(srcPath) + if err != nil { + return nil, err + } + + includes := []string{"."} + + // If .dockerignore mentions .dockerignore or the Dockerfile + // then make sure we send both files over to the daemon + // because Dockerfile is, obviously, needed no matter what, and + // .dockerignore is needed to know if either one needs to be + // removed. The deamon will remove them for us, if needed, after it + // parses the Dockerfile. + // + // https://github.com/docker/docker/issues/8330 + // + forceIncludeFiles := []string{".dockerignore", dockerfilePath} + + for _, includeFile := range forceIncludeFiles { + if includeFile == "" { + continue + } + keepThem, err := fileutils.Matches(includeFile, excludes) + if err != nil { + return nil, fmt.Errorf("cannot match .dockerfile: '%s', error: %s", includeFile, err) + } + if keepThem { + includes = append(includes, includeFile) + } + } + + if err := validateContextDirectory(srcPath, excludes); err != nil { + return nil, err + } + tarOpts := &archive.TarOptions{ + ExcludePatterns: excludes, + IncludeFiles: includes, + Compression: archive.Uncompressed, + NoLchown: true, + } + return archive.TarWithOptions(srcPath, tarOpts) +} + +// validateContextDirectory checks if all the contents of the directory +// can be read and returns an error if some files can't be read. +// Symlinks which point to non-existing files don't trigger an error +func validateContextDirectory(srcPath string, excludes []string) error { + return filepath.Walk(filepath.Join(srcPath, "."), func(filePath string, f os.FileInfo, err error) error { + // skip this directory/file if it's not in the path, it won't get added to the context + if relFilePath, err := filepath.Rel(srcPath, filePath); err != nil { + return err + } else if skip, err := fileutils.Matches(relFilePath, excludes); err != nil { + return err + } else if skip { + if f.IsDir() { + return filepath.SkipDir + } + return nil + } + + if err != nil { + if os.IsPermission(err) { + return fmt.Errorf("can't stat '%s'", filePath) + } + if os.IsNotExist(err) { + return nil + } + return err + } + + // skip checking if symlinks point to non-existing files, such symlinks can be useful + // also skip named pipes, because they hanging on open + if f.Mode()&(os.ModeSymlink|os.ModeNamedPipe) != 0 { + return nil + } + + if !f.IsDir() { + currentFile, err := os.Open(filePath) + if err != nil && os.IsPermission(err) { + return fmt.Errorf("no permission to read from '%s'", filePath) + } + currentFile.Close() + } + return nil + }) +} + +func parseDockerignore(root string) ([]string, error) { + var excludes []string + ignore, err := ioutil.ReadFile(path.Join(root, ".dockerignore")) + if err != nil && !os.IsNotExist(err) { + return excludes, fmt.Errorf("error reading .dockerignore: '%s'", err) + } + excludes = strings.Split(string(ignore), "\n") + + return excludes, nil +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/.dockerignore b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/.dockerignore new file mode 100644 index 0000000000000..027e8c20e6166 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/.dockerignore @@ -0,0 +1,3 @@ +container.tar +dockerfile.tar +foofile diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/Dockerfile b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/Dockerfile new file mode 100644 index 0000000000000..0948dcfa8cc5e --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/Dockerfile @@ -0,0 +1,15 @@ +# this file describes how to build tsuru python image +# to run it: +# 1- install docker +# 2- run: $ docker build -t tsuru/python https://raw.github.com/tsuru/basebuilder/master/python/Dockerfile + +from base:ubuntu-quantal +run apt-get install wget -y --force-yes +run wget http://github.com/tsuru/basebuilder/tarball/master -O basebuilder.tar.gz --no-check-certificate +run mkdir /var/lib/tsuru +run tar -xvf basebuilder.tar.gz -C /var/lib/tsuru --strip 1 +run cp /var/lib/tsuru/python/deploy /var/lib/tsuru +run cp /var/lib/tsuru/base/restart /var/lib/tsuru +run cp /var/lib/tsuru/base/start /var/lib/tsuru +run /var/lib/tsuru/base/install +run /var/lib/tsuru/base/setup diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/barfile b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/barfile new file mode 100644 index 0000000000000..e69de29bb2d1d diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/ca.pem b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/ca.pem new file mode 100644 index 0000000000000..8e38bba13c694 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/ca.pem @@ -0,0 +1,18 @@ +-----BEGIN CERTIFICATE----- +MIIC1TCCAb+gAwIBAgIQJ9MsNxrUxumNbAytGi3GEDALBgkqhkiG9w0BAQswFjEU +MBIGA1UEChMLQm9vdDJEb2NrZXIwHhcNMTQxMDE2MjAyMTM4WhcNMTcwOTMwMjAy +MTM4WjAWMRQwEgYDVQQKEwtCb290MkRvY2tlcjCCASIwDQYJKoZIhvcNAQEBBQAD +ggEPADCCAQoCggEBALpFCSARjG+5yXoqr7UMzuE0df7RRZfeRZI06lJ02ZqV4Iii +rgL7ML9yPxX50NbLnjiilSDTUhnyocYFItokzUzz8qpX/nlYhuN2Iqwh4d0aWS8z +f5y248F+H1z+HY2W8NPl/6DVlVwYaNW1/k+RPMlHS0INLR6j+3Ievew7RNE0NnM2 +znELW6NetekDt3GUcz0Z95vDUDfdPnIk1eIFMmYvLxZh23xOca4Q37a3S8F3d+dN ++OOpwjdgY9Qme0NQUaXpgp58jWuQfB8q7mZrdnLlLqRa8gx1HeDSotX7UmWtWPkb +vd9EdlKLYw5PVpxMV1rkwf2t4TdgD5NfkpXlXkkCAwEAAaMjMCEwDgYDVR0PAQH/ +BAQDAgCkMA8GA1UdEwEB/wQFMAMBAf8wCwYJKoZIhvcNAQELA4IBAQBxYjHVSKqE +MJw7CW0GddesULtXXVWGJuZdWJLQlPvPMfIfjIvlcZyS4cdVNiQ3sREFIZz8TpII +CT0/Pg3sgv/FcOQe1CN0xZYZcyiAZHK1z0fJQq2qVpdv7+tJcjI2vvU6NI24iQCo +W1wz25trJz9QbdB2MRLMjyz7TSWuafztIvcfEzaIdQ0Whqund/cSuPGQx5IwF83F +rvlkOyJSH2+VIEBTCIuykJeL0DLTt8cePBQR5L1ISXb4RUMK9ZtqRscBRv8sn7o2 +ixG3wtL0gYF4xLtsQWVxI3iFVrU3WzOH/3c5shVRkWBd+AQRSwCJI4mKH7penJCF +i3/zzlkvOnjV +-----END CERTIFICATE----- diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/cert.pem b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/cert.pem new file mode 100644 index 0000000000000..5e7244b24fffe --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/cert.pem @@ -0,0 +1,18 @@ +-----BEGIN CERTIFICATE----- +MIIC6DCCAdKgAwIBAgIRANO6ymxQAjp66KmEka1G6b0wCwYJKoZIhvcNAQELMBYx +FDASBgNVBAoTC0Jvb3QyRG9ja2VyMB4XDTE0MTAxNjIwMjE1MloXDTE3MDkzMDIw +MjE1MlowFjEUMBIGA1UEChMLQm9vdDJEb2NrZXIwggEiMA0GCSqGSIb3DQEBAQUA +A4IBDwAwggEKAoIBAQDGA1mAhSOpZspD1dpZ7qVEQrIJw4Xo8252jHaORnEdDiFm +b6brEmr6jw8t4P3IGxbqBc/TqRV+SSXxwYEVvfpeQKH+SmqStoMNtD3Ura161az4 +V0BcxMtSlsUGpoz+//QCAq8qiaxMwgiyc5253mkQm88anj2cNt7xbewiu/KFWuf7 +BVpNK1+ltpJmlukfcj/G+I1bw7j1KxBjDrFqe5cyDuuZcDL2tmUXP/ZWDyXwSv+H +AOckqn44z6aXlBkVvOXDBZJqY76d/vWVDNCuZeXRnqlhP3t1kH4V0RQXo+JD2tgt +JgdU0unzyoFOSWNUBPm73tqmjUGGAmGHBmeegJr/AgMBAAGjNTAzMA4GA1UdDwEB +/wQEAwIAgDATBgNVHSUEDDAKBggrBgEFBQcDAjAMBgNVHRMBAf8EAjAAMAsGCSqG +SIb3DQEBCwOCAQEABVTWl5SmBP+j5He5bQsgnIXjviSKqe40/10V4LJAOmilycRF +zLrzM+YMwfjg6PLIs8CldAMWHw9y9ktZY4MxkgCktaiaN/QmMTMwFWEcN4wy5IpM +U5l93eAg7xsnY430h3QBBADujX4wdF3fs8rSL8zAAQFL0ihurwU124K3yXKsrwpb +CiVUGfIN4sPwjy8Ws9oxHFDC9/P8lgjHZ1nBIf8KSHnMzlxDGj7isQfhtH+7mcCL +cM1qO2NirS2v7uaEPPY+MJstAz+W7EJCW9dfMSmHna2SDC37Xkin7uEY9z+qaKFL +8d/XxOB/L8Ucy8VZhdsv0dsBq5KfJntITM0ksQ== +-----END CERTIFICATE----- diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/container.tar b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/container.tar new file mode 100644 index 0000000000000..e4b066e3b6df8 Binary files /dev/null and b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/container.tar differ diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/dockerfile.tar b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/dockerfile.tar new file mode 100644 index 0000000000000..32c9ce6470483 Binary files /dev/null and b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/dockerfile.tar differ diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/foofile b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/foofile new file mode 100644 index 0000000000000..e69de29bb2d1d diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/key.pem b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/key.pem new file mode 100644 index 0000000000000..a9346bcf45ad5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/key.pem @@ -0,0 +1,27 @@ +-----BEGIN RSA PRIVATE KEY----- +MIIEowIBAAKCAQEAxgNZgIUjqWbKQ9XaWe6lREKyCcOF6PNudox2jkZxHQ4hZm+m +6xJq+o8PLeD9yBsW6gXP06kVfkkl8cGBFb36XkCh/kpqkraDDbQ91K2tetWs+FdA +XMTLUpbFBqaM/v/0AgKvKomsTMIIsnOdud5pEJvPGp49nDbe8W3sIrvyhVrn+wVa +TStfpbaSZpbpH3I/xviNW8O49SsQYw6xanuXMg7rmXAy9rZlFz/2Vg8l8Er/hwDn +JKp+OM+ml5QZFbzlwwWSamO+nf71lQzQrmXl0Z6pYT97dZB+FdEUF6PiQ9rYLSYH +VNLp88qBTkljVAT5u97apo1BhgJhhwZnnoCa/wIDAQABAoIBAQCaGy9EC9pmU95l +DwGh7k5nIrUnTilg1FwLHWSDdCVCZKXv8ENrPelOWZqJrUo1u4eI2L8XTsewgkNq +tJu/DRzWz9yDaO0qg6rZNobMh+K076lvmZA44twOydJLS8H+D7ua+PXU2FLlZjmY +kMyXRJZmW6zCXZc7haTbJx6ZJccoquk/DkS4FcFurJP177u1YrWS9TTw9kensUtU +jQ63uf56UTN1i+0+Rxl7OW1TZlqwlri5I4njg5249+FxwwHzIq8+l7zD7K9pl8c/ +nG1HuulvU2bVlDlRdyslMPAH34vw9Sku1BD8furrJLr1na5lRSLKJODEaIPEsLwv +CdEUwP9JAoGBAO76ZW80RyNB2fA+wbTq70Sr8CwrXxYemXrez5LKDC7SsohKFCPE +IedpO/n+nmymiiJvMm874EExoG6BVrbkWkeb+2vinEfOQNlDMsDx7WLjPekP3t6i +rXHO3CjFooVFq2z3mZa/Nc5NZqu8fNWNCKJxZDJphdoj6sORNJIUvZVjAoGBANQd +++J+ITcu3/+A6JrGcgLunBFQYPqkiItk0J4QKYKuX5ik9rWcQDN8TTtfW2mDuiQ4 +NrCwuVPq1V1kB16JzH017SsYLo9g8I20YjnBZge9pKTeUaLVTb3C50LW8FBylop0 +Bnm597dNbtSjphjoTMg0XyC19o3Esf2YeWG0QNS1AoGAWWDfFRNJU99qIldmXULM +0DM6NVrXSk+ReYnhunXEzrJQwXZrR+EwCPurydk36Uz0NuK9yypquhdUeF/5TZfk +SAoHo5byekyipl9imRUigqyY2BTudvgCxKDoaHtaSFwBPFTyZZYICquaLbrmOXxw +8UhVgCFFRYvPXuts7QHC0h8CgYBWEvy9gfU0kV7wLX02IUTuj6jhFb7ktpN6DSTi +nyhZES1VoctDEu6ydcRZTW6ouH12aSE4Pd5WgTqntQmQgVZrkNB25k8ue2Xh+srJ +KQOgLIJ9LIHwE6KCWG7DnrjRzE3uTPq7to0g4tkQjH/AJ7PQof/gJDayfJjFkXPg +A+cy6QKBgEPbKpiqscm03gT2QanBut5pg4dqPOxp0SlErA3kSFNTRK3oYBQPC+LH +qA5nD5brdkeNBB58Rll8Zpzxiff50bcvLP/7/Sb3NjaXFTEY0gVbdRof3n6N0YP3 +Hu5XDNJ9RNkNzE5RIG1g86KE+aKlcrKMaigqAiuIy2PSnjkQeGk8 +-----END RSA PRIVATE KEY----- diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/server.pem b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/server.pem new file mode 100644 index 0000000000000..89cc445e1ba1e --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/server.pem @@ -0,0 +1,18 @@ +-----BEGIN CERTIFICATE----- +MIIC/DCCAeagAwIBAgIQMUILcXtvmSOK63zEBo0VXzALBgkqhkiG9w0BAQswFjEU +MBIGA1UEChMLQm9vdDJEb2NrZXIwHhcNMTQxMDE2MjAyMTQ2WhcNMTcwOTMwMjAy +MTQ2WjAWMRQwEgYDVQQKEwtCb290MkRvY2tlcjCCASIwDQYJKoZIhvcNAQEBBQAD +ggEPADCCAQoCggEBANxUOUhNnqFnrTlLsBYzfFRZWQo268l+4K4lOJCVbfDonP3g +Mz0vGi9fcyFqEWSA8Y+ShXna625HTnReCwFdsu0861qCIq7v95hFFCyOe0iIxpd0 +AKLnl90d+1vonE7andgFgoobbTiMly4UK4H6z8D148fFNIihoteOG3PIF89TFxP7 +CJ/3wXnx/IKpdlO8PAnub3tBPJHvGDj7KORLy4IBxRX5VBAdfGNybE66fcrehEva +rLA4m9pgiaR/Nnr9FdKhPyqYdjflLNvzydxNvMIV4M0hFlhXmYvpMjA5/XsTnsyV +t9JHJa5Upwqsbne08t7rsm7liZNxZlko8xPOTQcCAwEAAaNKMEgwDgYDVR0PAQH/ +BAQDAgCgMAwGA1UdEwEB/wQCMAAwKAYDVR0RBCEwH4ILYm9vdDJkb2NrZXKHBH8A +AAGHBAoAAg+HBMCoO2cwCwYJKoZIhvcNAQELA4IBAQAYoYcDkDWkl73FZ0WnPmAj +LiF7HU95Qg3KyEpFsAJeShSLPPbQntmwhdekEzY4tQ3eKQB/+zHFjzsCr/lmDUmH +Ea/ryQ17C+jyH+Ykg0IWW6L6veZhvRDg6Z9focVtPVBRxPTqC/Qhb54blWRASV+W +UreMuXQ5+1dQptAM7ixOeLVHjBi/bd9TL3jvwBVCr9QedteMjjK4TCF9Tbcou+MF +2w3OJJZMDhcD+YwoK9uJDqlKmcTm/vVMbSsp/pTMcnQ7jxCeR8/XyX+VwTZwaHAa +o92Q/eg3THAiWhvyT/SzyH9dHHBAyXynUwGCggKawHktfvW4QXRPuLxLrJ7iB5cy +-----END CERTIFICATE----- diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/serverkey.pem b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/serverkey.pem new file mode 100644 index 0000000000000..c897e5da5501e --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/serverkey.pem @@ -0,0 +1,27 @@ +-----BEGIN RSA PRIVATE KEY----- +MIIEoAIBAAKCAQEA3FQ5SE2eoWetOUuwFjN8VFlZCjbryX7griU4kJVt8Oic/eAz +PS8aL19zIWoRZIDxj5KFedrrbkdOdF4LAV2y7TzrWoIiru/3mEUULI57SIjGl3QA +oueX3R37W+icTtqd2AWCihttOIyXLhQrgfrPwPXjx8U0iKGi144bc8gXz1MXE/sI +n/fBefH8gql2U7w8Ce5ve0E8ke8YOPso5EvLggHFFflUEB18Y3JsTrp9yt6ES9qs +sDib2mCJpH82ev0V0qE/Kph2N+Us2/PJ3E28whXgzSEWWFeZi+kyMDn9exOezJW3 +0kclrlSnCqxud7Ty3uuybuWJk3FmWSjzE85NBwIDAQABAoIBAG0ak+cW8LeShHf7 +3+2Of0GxoOLrAWWdG5uAuPr31CJYve0FybnBimDtDjD8ujIfm/7xmoEWBEFutA3x +x9dcU88gvJbsHEqub9gKVQwfXjMz78tt2SbSMiR/xUnk7QorPcCMMfE71aEMFYzu +1gCed6Rg3vO81t/V0rKVH0j9S7UQz5v/oX15eVDV5LOqyCHwAi6K0eXXbqnbI0TH +SOQ/nexM2msVXWbO9t6ra6f5V7FXziDK5Xi+rPxRbX9mkrDzxDAevfuRqYBx5vtL +W2Q2hKjUAHFgXFniNSZBS7dCdAtz0el/3ct+cNmpuTMhhs7M6wC1CuYiZ/DxLiFh +Si73VckCgYEA+/ceh3+VjtQ0rgEw8sD9bqYEA8IaBiObjneIoFnKBYRG7yZd8JMm +HD4M/aQ1qhcRLPN7GR03YQULgQJURbKSjJHnhfTXHyeHC3NN4gMVHQXewu2MHCh6 +7FCQ9CfK0KcYLgegVVvL3PrF3hyWGnmTu+G0UkDQRYVnaNrB7snrW6UCgYEA39tq ++MCQdu0moJ5szSZf02undg9EeW6isk9qzi7TId3/MLci2eH7PEnipipPUK3+DERq +aba0y0TKgBR2EXvXLFJA/+kfdo2loIEHOfox85HVfxgUaFRti63ZI0uF8D0QT2Yy +oJal+RFghVoSnv4LjhRKEPbIkScTXGjdK+7wFjsCfz79iKRXQQx0ALd/lL0bgkAn +QNmvrNHcFQeI2p8700WNzC39aX67SsvEt3qxkrjzC1gxhpTAuReIK1gVPPwvqHN8 +BmV20FD5kMlMCix2mNCopwgUWvKvLAvoGFTxncKMA39+aJbuXAjiqJTekKgNvOE7 +i9kEWw0GTNPp3JHV6QECgYAPwb0M11kT1euDIMOdyRazpf86kyaJuZzgGjD1ZFxe +JOcigbGFTp/FhZnbglzk2+pm6KXo3QBq0mPCki4hWusxZnTGzpz1VlETNCHTFeZQ +M7KoaIR/N3oie9Et59H8r/+m5xWnMhNqratyl316DX24uXrhKM3DUdHODl+LCR2D +IwKBgE1MbHuwolUPEw3HeO4R7NMFVTFei7E/fpUsimPfArGg8UydwvloNT1myJos +N2JzfGGjN2KPVcBk9fOs71mJ6VcK3C3g5JIccplk6h9VNaw55+zdQvKPTzoBoTvy +A+Fwx2AlF61KeRF87DL2YTRJ6B9MHmWgf7+GVZOxomLgEAcZ +-----END RSA PRIVATE KEY----- diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/symlink b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/symlink new file mode 120000 index 0000000000000..3ddf86a359474 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/data/symlink @@ -0,0 +1 @@ +doesnotexist \ No newline at end of file diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go new file mode 100644 index 0000000000000..41d87277159ba --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server.go @@ -0,0 +1,1062 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// Package testing provides a fake implementation of the Docker API, useful for +// testing purpose. +package testing + +import ( + "archive/tar" + "crypto/rand" + "encoding/json" + "errors" + "fmt" + mathrand "math/rand" + "net" + "net/http" + "regexp" + "strconv" + "strings" + "sync" + "time" + + "github.com/fsouza/go-dockerclient" + "github.com/fsouza/go-dockerclient/external/github.com/docker/docker/pkg/stdcopy" + "github.com/fsouza/go-dockerclient/external/github.com/gorilla/mux" +) + +var nameRegexp = regexp.MustCompile(`^[a-zA-Z0-9][a-zA-Z0-9_.-]+$`) + +// DockerServer represents a programmable, concurrent (not much), HTTP server +// implementing a fake version of the Docker remote API. +// +// It can used in standalone mode, listening for connections or as an arbitrary +// HTTP handler. +// +// For more details on the remote API, check http://goo.gl/G3plxW. +type DockerServer struct { + containers []*docker.Container + execs []*docker.ExecInspect + execMut sync.RWMutex + cMut sync.RWMutex + images []docker.Image + iMut sync.RWMutex + imgIDs map[string]string + networks []*docker.Network + netMut sync.RWMutex + listener net.Listener + mux *mux.Router + hook func(*http.Request) + failures map[string]string + multiFailures []map[string]string + execCallbacks map[string]func() + statsCallbacks map[string]func(string) docker.Stats + customHandlers map[string]http.Handler + handlerMutex sync.RWMutex + cChan chan<- *docker.Container +} + +// NewServer returns a new instance of the fake server, in standalone mode. Use +// the method URL to get the URL of the server. +// +// It receives the bind address (use 127.0.0.1:0 for getting an available port +// on the host), a channel of containers and a hook function, that will be +// called on every request. +// +// The fake server will send containers in the channel whenever the container +// changes its state, via the HTTP API (i.e.: create, start and stop). This +// channel may be nil, which means that the server won't notify on state +// changes. +func NewServer(bind string, containerChan chan<- *docker.Container, hook func(*http.Request)) (*DockerServer, error) { + listener, err := net.Listen("tcp", bind) + if err != nil { + return nil, err + } + server := DockerServer{ + listener: listener, + imgIDs: make(map[string]string), + hook: hook, + failures: make(map[string]string), + execCallbacks: make(map[string]func()), + statsCallbacks: make(map[string]func(string) docker.Stats), + customHandlers: make(map[string]http.Handler), + cChan: containerChan, + } + server.buildMuxer() + go http.Serve(listener, &server) + return &server, nil +} + +func (s *DockerServer) notify(container *docker.Container) { + if s.cChan != nil { + s.cChan <- container + } +} + +func (s *DockerServer) buildMuxer() { + s.mux = mux.NewRouter() + s.mux.Path("/commit").Methods("POST").HandlerFunc(s.handlerWrapper(s.commitContainer)) + s.mux.Path("/containers/json").Methods("GET").HandlerFunc(s.handlerWrapper(s.listContainers)) + s.mux.Path("/containers/create").Methods("POST").HandlerFunc(s.handlerWrapper(s.createContainer)) + s.mux.Path("/containers/{id:.*}/json").Methods("GET").HandlerFunc(s.handlerWrapper(s.inspectContainer)) + s.mux.Path("/containers/{id:.*}/rename").Methods("POST").HandlerFunc(s.handlerWrapper(s.renameContainer)) + s.mux.Path("/containers/{id:.*}/top").Methods("GET").HandlerFunc(s.handlerWrapper(s.topContainer)) + s.mux.Path("/containers/{id:.*}/start").Methods("POST").HandlerFunc(s.handlerWrapper(s.startContainer)) + s.mux.Path("/containers/{id:.*}/kill").Methods("POST").HandlerFunc(s.handlerWrapper(s.stopContainer)) + s.mux.Path("/containers/{id:.*}/stop").Methods("POST").HandlerFunc(s.handlerWrapper(s.stopContainer)) + s.mux.Path("/containers/{id:.*}/pause").Methods("POST").HandlerFunc(s.handlerWrapper(s.pauseContainer)) + s.mux.Path("/containers/{id:.*}/unpause").Methods("POST").HandlerFunc(s.handlerWrapper(s.unpauseContainer)) + s.mux.Path("/containers/{id:.*}/wait").Methods("POST").HandlerFunc(s.handlerWrapper(s.waitContainer)) + s.mux.Path("/containers/{id:.*}/attach").Methods("POST").HandlerFunc(s.handlerWrapper(s.attachContainer)) + s.mux.Path("/containers/{id:.*}").Methods("DELETE").HandlerFunc(s.handlerWrapper(s.removeContainer)) + s.mux.Path("/containers/{id:.*}/exec").Methods("POST").HandlerFunc(s.handlerWrapper(s.createExecContainer)) + s.mux.Path("/containers/{id:.*}/stats").Methods("GET").HandlerFunc(s.handlerWrapper(s.statsContainer)) + s.mux.Path("/exec/{id:.*}/resize").Methods("POST").HandlerFunc(s.handlerWrapper(s.resizeExecContainer)) + s.mux.Path("/exec/{id:.*}/start").Methods("POST").HandlerFunc(s.handlerWrapper(s.startExecContainer)) + s.mux.Path("/exec/{id:.*}/json").Methods("GET").HandlerFunc(s.handlerWrapper(s.inspectExecContainer)) + s.mux.Path("/images/create").Methods("POST").HandlerFunc(s.handlerWrapper(s.pullImage)) + s.mux.Path("/build").Methods("POST").HandlerFunc(s.handlerWrapper(s.buildImage)) + s.mux.Path("/images/json").Methods("GET").HandlerFunc(s.handlerWrapper(s.listImages)) + s.mux.Path("/images/{id:.*}").Methods("DELETE").HandlerFunc(s.handlerWrapper(s.removeImage)) + s.mux.Path("/images/{name:.*}/json").Methods("GET").HandlerFunc(s.handlerWrapper(s.inspectImage)) + s.mux.Path("/images/{name:.*}/push").Methods("POST").HandlerFunc(s.handlerWrapper(s.pushImage)) + s.mux.Path("/images/{name:.*}/tag").Methods("POST").HandlerFunc(s.handlerWrapper(s.tagImage)) + s.mux.Path("/events").Methods("GET").HandlerFunc(s.listEvents) + s.mux.Path("/_ping").Methods("GET").HandlerFunc(s.handlerWrapper(s.pingDocker)) + s.mux.Path("/images/load").Methods("POST").HandlerFunc(s.handlerWrapper(s.loadImage)) + s.mux.Path("/images/{id:.*}/get").Methods("GET").HandlerFunc(s.handlerWrapper(s.getImage)) + s.mux.Path("/networks").Methods("GET").HandlerFunc(s.handlerWrapper(s.listNetworks)) + s.mux.Path("/networks/{id:.*}").Methods("GET").HandlerFunc(s.handlerWrapper(s.networkInfo)) + s.mux.Path("/networks").Methods("POST").HandlerFunc(s.handlerWrapper(s.createNetwork)) +} + +// SetHook changes the hook function used by the server. +// +// The hook function is a function called on every request. +func (s *DockerServer) SetHook(hook func(*http.Request)) { + s.hook = hook +} + +// PrepareExec adds a callback to a container exec in the fake server. +// +// This function will be called whenever the given exec id is started, and the +// given exec id will remain in the "Running" start while the function is +// running, so it's useful for emulating an exec that runs for two seconds, for +// example: +// +// opts := docker.CreateExecOptions{ +// AttachStdin: true, +// AttachStdout: true, +// AttachStderr: true, +// Tty: true, +// Cmd: []string{"/bin/bash", "-l"}, +// } +// // Client points to a fake server. +// exec, err := client.CreateExec(opts) +// // handle error +// server.PrepareExec(exec.ID, func() {time.Sleep(2 * time.Second)}) +// err = client.StartExec(exec.ID, docker.StartExecOptions{Tty: true}) // will block for 2 seconds +// // handle error +func (s *DockerServer) PrepareExec(id string, callback func()) { + s.execCallbacks[id] = callback +} + +// PrepareStats adds a callback that will be called for each container stats +// call. +// +// This callback function will be called multiple times if stream is set to +// true when stats is called. +func (s *DockerServer) PrepareStats(id string, callback func(string) docker.Stats) { + s.statsCallbacks[id] = callback +} + +// PrepareFailure adds a new expected failure based on a URL regexp it receives +// an id for the failure. +func (s *DockerServer) PrepareFailure(id string, urlRegexp string) { + s.failures[id] = urlRegexp +} + +// PrepareMultiFailures enqueues a new expected failure based on a URL regexp +// it receives an id for the failure. +func (s *DockerServer) PrepareMultiFailures(id string, urlRegexp string) { + s.multiFailures = append(s.multiFailures, map[string]string{"error": id, "url": urlRegexp}) +} + +// ResetFailure removes an expected failure identified by the given id. +func (s *DockerServer) ResetFailure(id string) { + delete(s.failures, id) +} + +// ResetMultiFailures removes all enqueued failures. +func (s *DockerServer) ResetMultiFailures() { + s.multiFailures = []map[string]string{} +} + +// CustomHandler registers a custom handler for a specific path. +// +// For example: +// +// server.CustomHandler("/containers/json", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { +// http.Error(w, "Something wrong is not right", http.StatusInternalServerError) +// })) +func (s *DockerServer) CustomHandler(path string, handler http.Handler) { + s.handlerMutex.Lock() + s.customHandlers[path] = handler + s.handlerMutex.Unlock() +} + +// MutateContainer changes the state of a container, returning an error if the +// given id does not match to any container "running" in the server. +func (s *DockerServer) MutateContainer(id string, state docker.State) error { + for _, container := range s.containers { + if container.ID == id { + container.State = state + return nil + } + } + return errors.New("container not found") +} + +// Stop stops the server. +func (s *DockerServer) Stop() { + if s.listener != nil { + s.listener.Close() + } +} + +// URL returns the HTTP URL of the server. +func (s *DockerServer) URL() string { + if s.listener == nil { + return "" + } + return "http://" + s.listener.Addr().String() + "/" +} + +// ServeHTTP handles HTTP requests sent to the server. +func (s *DockerServer) ServeHTTP(w http.ResponseWriter, r *http.Request) { + s.handlerMutex.RLock() + defer s.handlerMutex.RUnlock() + for re, handler := range s.customHandlers { + if m, _ := regexp.MatchString(re, r.URL.Path); m { + handler.ServeHTTP(w, r) + return + } + } + s.mux.ServeHTTP(w, r) + if s.hook != nil { + s.hook(r) + } +} + +// DefaultHandler returns default http.Handler mux, it allows customHandlers to +// call the default behavior if wanted. +func (s *DockerServer) DefaultHandler() http.Handler { + return s.mux +} + +func (s *DockerServer) handlerWrapper(f func(http.ResponseWriter, *http.Request)) func(http.ResponseWriter, *http.Request) { + return func(w http.ResponseWriter, r *http.Request) { + for errorID, urlRegexp := range s.failures { + matched, err := regexp.MatchString(urlRegexp, r.URL.Path) + if err != nil { + http.Error(w, err.Error(), http.StatusBadRequest) + return + } + if !matched { + continue + } + http.Error(w, errorID, http.StatusBadRequest) + return + } + for i, failure := range s.multiFailures { + matched, err := regexp.MatchString(failure["url"], r.URL.Path) + if err != nil { + http.Error(w, err.Error(), http.StatusBadRequest) + return + } + if !matched { + continue + } + http.Error(w, failure["error"], http.StatusBadRequest) + s.multiFailures = append(s.multiFailures[:i], s.multiFailures[i+1:]...) + return + } + f(w, r) + } +} + +func (s *DockerServer) listContainers(w http.ResponseWriter, r *http.Request) { + all := r.URL.Query().Get("all") + s.cMut.RLock() + result := make([]docker.APIContainers, 0, len(s.containers)) + for _, container := range s.containers { + if all == "1" || container.State.Running { + result = append(result, docker.APIContainers{ + ID: container.ID, + Image: container.Image, + Command: fmt.Sprintf("%s %s", container.Path, strings.Join(container.Args, " ")), + Created: container.Created.Unix(), + Status: container.State.String(), + Ports: container.NetworkSettings.PortMappingAPI(), + Names: []string{fmt.Sprintf("/%s", container.Name)}, + }) + } + } + s.cMut.RUnlock() + w.Header().Set("Content-Type", "application/json") + w.WriteHeader(http.StatusOK) + json.NewEncoder(w).Encode(result) +} + +func (s *DockerServer) listImages(w http.ResponseWriter, r *http.Request) { + s.cMut.RLock() + result := make([]docker.APIImages, len(s.images)) + for i, image := range s.images { + result[i] = docker.APIImages{ + ID: image.ID, + Created: image.Created.Unix(), + } + for tag, id := range s.imgIDs { + if id == image.ID { + result[i].RepoTags = append(result[i].RepoTags, tag) + } + } + } + s.cMut.RUnlock() + w.Header().Set("Content-Type", "application/json") + w.WriteHeader(http.StatusOK) + json.NewEncoder(w).Encode(result) +} + +func (s *DockerServer) findImage(id string) (string, error) { + s.iMut.RLock() + defer s.iMut.RUnlock() + image, ok := s.imgIDs[id] + if ok { + return image, nil + } + image, _, err := s.findImageByID(id) + return image, err +} + +func (s *DockerServer) findImageByID(id string) (string, int, error) { + s.iMut.RLock() + defer s.iMut.RUnlock() + for i, image := range s.images { + if image.ID == id { + return image.ID, i, nil + } + } + return "", -1, errors.New("No such image") +} + +func (s *DockerServer) createContainer(w http.ResponseWriter, r *http.Request) { + var config struct { + *docker.Config + HostConfig *docker.HostConfig + } + defer r.Body.Close() + err := json.NewDecoder(r.Body).Decode(&config) + if err != nil { + http.Error(w, err.Error(), http.StatusBadRequest) + return + } + name := r.URL.Query().Get("name") + if name != "" && !nameRegexp.MatchString(name) { + http.Error(w, "Invalid container name", http.StatusInternalServerError) + return + } + if _, err := s.findImage(config.Image); err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + ports := map[docker.Port][]docker.PortBinding{} + for port := range config.ExposedPorts { + ports[port] = []docker.PortBinding{{ + HostIP: "0.0.0.0", + HostPort: strconv.Itoa(mathrand.Int() % 65536), + }} + } + + //the container may not have cmd when using a Dockerfile + var path string + var args []string + if len(config.Cmd) == 1 { + path = config.Cmd[0] + } else if len(config.Cmd) > 1 { + path = config.Cmd[0] + args = config.Cmd[1:] + } + + generatedID := s.generateID() + config.Config.Hostname = generatedID[:12] + container := docker.Container{ + Name: name, + ID: generatedID, + Created: time.Now(), + Path: path, + Args: args, + Config: config.Config, + HostConfig: config.HostConfig, + State: docker.State{ + Running: false, + Pid: mathrand.Int() % 50000, + ExitCode: 0, + StartedAt: time.Now(), + }, + Image: config.Image, + NetworkSettings: &docker.NetworkSettings{ + IPAddress: fmt.Sprintf("172.16.42.%d", mathrand.Int()%250+2), + IPPrefixLen: 24, + Gateway: "172.16.42.1", + Bridge: "docker0", + Ports: ports, + }, + } + s.cMut.Lock() + if container.Name != "" { + for _, c := range s.containers { + if c.Name == container.Name { + defer s.cMut.Unlock() + http.Error(w, "there's already a container with this name", http.StatusConflict) + return + } + } + } + s.containers = append(s.containers, &container) + s.cMut.Unlock() + w.WriteHeader(http.StatusCreated) + s.notify(&container) + var c = struct{ ID string }{ID: container.ID} + json.NewEncoder(w).Encode(c) +} + +func (s *DockerServer) generateID() string { + var buf [16]byte + rand.Read(buf[:]) + return fmt.Sprintf("%x", buf) +} + +func (s *DockerServer) renameContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + container, index, err := s.findContainer(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + copy := *container + copy.Name = r.URL.Query().Get("name") + s.cMut.Lock() + defer s.cMut.Unlock() + if s.containers[index].ID == copy.ID { + s.containers[index] = © + } + w.WriteHeader(http.StatusNoContent) +} + +func (s *DockerServer) inspectContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + container, _, err := s.findContainer(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + w.Header().Set("Content-Type", "application/json") + w.WriteHeader(http.StatusOK) + json.NewEncoder(w).Encode(container) +} + +func (s *DockerServer) statsContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + _, _, err := s.findContainer(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + stream, _ := strconv.ParseBool(r.URL.Query().Get("stream")) + callback := s.statsCallbacks[id] + w.Header().Set("Content-Type", "application/json") + w.WriteHeader(http.StatusOK) + encoder := json.NewEncoder(w) + for { + var stats docker.Stats + if callback != nil { + stats = callback(id) + } + encoder.Encode(stats) + if !stream { + break + } + } +} + +func (s *DockerServer) topContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + container, _, err := s.findContainer(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + if !container.State.Running { + w.WriteHeader(http.StatusInternalServerError) + fmt.Fprintf(w, "Container %s is not running", id) + return + } + w.Header().Set("Content-Type", "application/json") + w.WriteHeader(http.StatusOK) + result := docker.TopResult{ + Titles: []string{"UID", "PID", "PPID", "C", "STIME", "TTY", "TIME", "CMD"}, + Processes: [][]string{ + {"root", "7535", "7516", "0", "03:20", "?", "00:00:00", container.Path + " " + strings.Join(container.Args, " ")}, + }, + } + json.NewEncoder(w).Encode(result) +} + +func (s *DockerServer) startContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + container, _, err := s.findContainer(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + s.cMut.Lock() + defer s.cMut.Unlock() + defer r.Body.Close() + var hostConfig docker.HostConfig + err = json.NewDecoder(r.Body).Decode(&hostConfig) + if err != nil { + http.Error(w, err.Error(), http.StatusInternalServerError) + return + } + container.HostConfig = &hostConfig + if container.State.Running { + http.Error(w, "Container already running", http.StatusBadRequest) + return + } + container.State.Running = true + s.notify(container) +} + +func (s *DockerServer) stopContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + container, _, err := s.findContainer(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + s.cMut.Lock() + defer s.cMut.Unlock() + if !container.State.Running { + http.Error(w, "Container not running", http.StatusBadRequest) + return + } + w.WriteHeader(http.StatusNoContent) + container.State.Running = false + s.notify(container) +} + +func (s *DockerServer) pauseContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + container, _, err := s.findContainer(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + s.cMut.Lock() + defer s.cMut.Unlock() + if container.State.Paused { + http.Error(w, "Container already paused", http.StatusBadRequest) + return + } + w.WriteHeader(http.StatusNoContent) + container.State.Paused = true +} + +func (s *DockerServer) unpauseContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + container, _, err := s.findContainer(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + s.cMut.Lock() + defer s.cMut.Unlock() + if !container.State.Paused { + http.Error(w, "Container not paused", http.StatusBadRequest) + return + } + w.WriteHeader(http.StatusNoContent) + container.State.Paused = false +} + +func (s *DockerServer) attachContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + container, _, err := s.findContainer(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + hijacker, ok := w.(http.Hijacker) + if !ok { + http.Error(w, "cannot hijack connection", http.StatusInternalServerError) + return + } + w.Header().Set("Content-Type", "application/vnd.docker.raw-stream") + w.WriteHeader(http.StatusOK) + conn, _, err := hijacker.Hijack() + if err != nil { + http.Error(w, err.Error(), http.StatusInternalServerError) + return + } + outStream := stdcopy.NewStdWriter(conn, stdcopy.Stdout) + if container.State.Running { + fmt.Fprintf(outStream, "Container %q is running\n", container.ID) + } else { + fmt.Fprintf(outStream, "Container %q is not running\n", container.ID) + } + fmt.Fprintln(outStream, "What happened?") + fmt.Fprintln(outStream, "Something happened") + conn.Close() +} + +func (s *DockerServer) waitContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + container, _, err := s.findContainer(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + for { + time.Sleep(1e6) + s.cMut.RLock() + if !container.State.Running { + s.cMut.RUnlock() + break + } + s.cMut.RUnlock() + } + result := map[string]int{"StatusCode": container.State.ExitCode} + json.NewEncoder(w).Encode(result) +} + +func (s *DockerServer) removeContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + force := r.URL.Query().Get("force") + _, index, err := s.findContainer(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + if s.containers[index].State.Running && force != "1" { + msg := "Error: API error (406): Impossible to remove a running container, please stop it first" + http.Error(w, msg, http.StatusInternalServerError) + return + } + w.WriteHeader(http.StatusNoContent) + s.cMut.Lock() + defer s.cMut.Unlock() + s.containers[index] = s.containers[len(s.containers)-1] + s.containers = s.containers[:len(s.containers)-1] +} + +func (s *DockerServer) commitContainer(w http.ResponseWriter, r *http.Request) { + id := r.URL.Query().Get("container") + container, _, err := s.findContainer(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + var config *docker.Config + runConfig := r.URL.Query().Get("run") + if runConfig != "" { + config = new(docker.Config) + err = json.Unmarshal([]byte(runConfig), config) + if err != nil { + http.Error(w, err.Error(), http.StatusBadRequest) + return + } + } + w.WriteHeader(http.StatusOK) + image := docker.Image{ + ID: "img-" + container.ID, + Parent: container.Image, + Container: container.ID, + Comment: r.URL.Query().Get("m"), + Author: r.URL.Query().Get("author"), + Config: config, + } + repository := r.URL.Query().Get("repo") + tag := r.URL.Query().Get("tag") + s.iMut.Lock() + s.images = append(s.images, image) + if repository != "" { + if tag != "" { + repository += ":" + tag + } + s.imgIDs[repository] = image.ID + } + s.iMut.Unlock() + fmt.Fprintf(w, `{"ID":%q}`, image.ID) +} + +func (s *DockerServer) findContainer(idOrName string) (*docker.Container, int, error) { + s.cMut.RLock() + defer s.cMut.RUnlock() + for i, container := range s.containers { + if container.ID == idOrName || container.Name == idOrName { + return container, i, nil + } + } + return nil, -1, errors.New("No such container") +} + +func (s *DockerServer) buildImage(w http.ResponseWriter, r *http.Request) { + if ct := r.Header.Get("Content-Type"); ct == "application/tar" { + gotDockerFile := false + tr := tar.NewReader(r.Body) + for { + header, err := tr.Next() + if err != nil { + break + } + if header.Name == "Dockerfile" { + gotDockerFile = true + } + } + if !gotDockerFile { + w.WriteHeader(http.StatusBadRequest) + w.Write([]byte("miss Dockerfile")) + return + } + } + //we did not use that Dockerfile to build image cause we are a fake Docker daemon + image := docker.Image{ + ID: s.generateID(), + Created: time.Now(), + } + + query := r.URL.Query() + repository := image.ID + if t := query.Get("t"); t != "" { + repository = t + } + s.iMut.Lock() + s.images = append(s.images, image) + s.imgIDs[repository] = image.ID + s.iMut.Unlock() + w.Write([]byte(fmt.Sprintf("Successfully built %s", image.ID))) +} + +func (s *DockerServer) pullImage(w http.ResponseWriter, r *http.Request) { + fromImageName := r.URL.Query().Get("fromImage") + tag := r.URL.Query().Get("tag") + image := docker.Image{ + ID: s.generateID(), + } + s.iMut.Lock() + s.images = append(s.images, image) + if fromImageName != "" { + if tag != "" { + fromImageName = fmt.Sprintf("%s:%s", fromImageName, tag) + } + s.imgIDs[fromImageName] = image.ID + } + s.iMut.Unlock() +} + +func (s *DockerServer) pushImage(w http.ResponseWriter, r *http.Request) { + name := mux.Vars(r)["name"] + tag := r.URL.Query().Get("tag") + if tag != "" { + name += ":" + tag + } + s.iMut.RLock() + if _, ok := s.imgIDs[name]; !ok { + s.iMut.RUnlock() + http.Error(w, "No such image", http.StatusNotFound) + return + } + s.iMut.RUnlock() + fmt.Fprintln(w, "Pushing...") + fmt.Fprintln(w, "Pushed") +} + +func (s *DockerServer) tagImage(w http.ResponseWriter, r *http.Request) { + name := mux.Vars(r)["name"] + s.iMut.RLock() + if _, ok := s.imgIDs[name]; !ok { + s.iMut.RUnlock() + http.Error(w, "No such image", http.StatusNotFound) + return + } + s.iMut.RUnlock() + s.iMut.Lock() + defer s.iMut.Unlock() + newRepo := r.URL.Query().Get("repo") + newTag := r.URL.Query().Get("tag") + if newTag != "" { + newRepo += ":" + newTag + } + s.imgIDs[newRepo] = s.imgIDs[name] + w.WriteHeader(http.StatusCreated) +} + +func (s *DockerServer) removeImage(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + s.iMut.RLock() + var tag string + if img, ok := s.imgIDs[id]; ok { + id, tag = img, id + } + var tags []string + for tag, taggedID := range s.imgIDs { + if taggedID == id { + tags = append(tags, tag) + } + } + s.iMut.RUnlock() + _, index, err := s.findImageByID(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + w.WriteHeader(http.StatusNoContent) + s.iMut.Lock() + defer s.iMut.Unlock() + if len(tags) < 2 { + s.images[index] = s.images[len(s.images)-1] + s.images = s.images[:len(s.images)-1] + } + if tag != "" { + delete(s.imgIDs, tag) + } +} + +func (s *DockerServer) inspectImage(w http.ResponseWriter, r *http.Request) { + name := mux.Vars(r)["name"] + s.iMut.RLock() + defer s.iMut.RUnlock() + if id, ok := s.imgIDs[name]; ok { + for _, img := range s.images { + if img.ID == id { + w.Header().Set("Content-Type", "application/json") + w.WriteHeader(http.StatusOK) + json.NewEncoder(w).Encode(img) + return + } + } + } + http.Error(w, "not found", http.StatusNotFound) +} + +func (s *DockerServer) listEvents(w http.ResponseWriter, r *http.Request) { + w.Header().Set("Content-Type", "application/json") + var events [][]byte + count := mathrand.Intn(20) + for i := 0; i < count; i++ { + data, err := json.Marshal(s.generateEvent()) + if err != nil { + w.WriteHeader(http.StatusInternalServerError) + return + } + events = append(events, data) + } + w.WriteHeader(http.StatusOK) + for _, d := range events { + fmt.Fprintln(w, d) + time.Sleep(time.Duration(mathrand.Intn(200)) * time.Millisecond) + } +} + +func (s *DockerServer) pingDocker(w http.ResponseWriter, r *http.Request) { + w.WriteHeader(http.StatusOK) +} + +func (s *DockerServer) generateEvent() *docker.APIEvents { + var eventType string + switch mathrand.Intn(4) { + case 0: + eventType = "create" + case 1: + eventType = "start" + case 2: + eventType = "stop" + case 3: + eventType = "destroy" + } + return &docker.APIEvents{ + ID: s.generateID(), + Status: eventType, + From: "mybase:latest", + Time: time.Now().Unix(), + } +} + +func (s *DockerServer) loadImage(w http.ResponseWriter, r *http.Request) { + w.WriteHeader(http.StatusOK) +} + +func (s *DockerServer) getImage(w http.ResponseWriter, r *http.Request) { + w.WriteHeader(http.StatusOK) + w.Header().Set("Content-Type", "application/tar") +} + +func (s *DockerServer) createExecContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + container, _, err := s.findContainer(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + exec := docker.ExecInspect{ + ID: s.generateID(), + Container: *container, + } + var params docker.CreateExecOptions + err = json.NewDecoder(r.Body).Decode(¶ms) + if err != nil { + http.Error(w, err.Error(), http.StatusInternalServerError) + return + } + if len(params.Cmd) > 0 { + exec.ProcessConfig.EntryPoint = params.Cmd[0] + if len(params.Cmd) > 1 { + exec.ProcessConfig.Arguments = params.Cmd[1:] + } + } + s.execMut.Lock() + s.execs = append(s.execs, &exec) + s.execMut.Unlock() + w.WriteHeader(http.StatusOK) + w.Header().Set("Content-Type", "application/json") + json.NewEncoder(w).Encode(map[string]string{"Id": exec.ID}) +} + +func (s *DockerServer) startExecContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + if exec, err := s.getExec(id); err == nil { + s.execMut.Lock() + exec.Running = true + s.execMut.Unlock() + if callback, ok := s.execCallbacks[id]; ok { + callback() + delete(s.execCallbacks, id) + } else if callback, ok := s.execCallbacks["*"]; ok { + callback() + delete(s.execCallbacks, "*") + } + s.execMut.Lock() + exec.Running = false + s.execMut.Unlock() + w.WriteHeader(http.StatusOK) + return + } + w.WriteHeader(http.StatusNotFound) +} + +func (s *DockerServer) resizeExecContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + if _, err := s.getExec(id); err == nil { + w.WriteHeader(http.StatusOK) + return + } + w.WriteHeader(http.StatusNotFound) +} + +func (s *DockerServer) inspectExecContainer(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + if exec, err := s.getExec(id); err == nil { + w.WriteHeader(http.StatusOK) + w.Header().Set("Content-Type", "application/json") + json.NewEncoder(w).Encode(exec) + return + } + w.WriteHeader(http.StatusNotFound) +} + +func (s *DockerServer) getExec(id string) (*docker.ExecInspect, error) { + s.execMut.RLock() + defer s.execMut.RUnlock() + for _, exec := range s.execs { + if exec.ID == id { + return exec, nil + } + } + return nil, errors.New("exec not found") +} + +func (s *DockerServer) findNetwork(idOrName string) (*docker.Network, int, error) { + s.netMut.RLock() + defer s.netMut.RUnlock() + for i, network := range s.networks { + if network.ID == idOrName || network.Name == idOrName { + return network, i, nil + } + } + return nil, -1, errors.New("No such network") +} + +func (s *DockerServer) listNetworks(w http.ResponseWriter, r *http.Request) { + s.netMut.RLock() + result := make([]docker.Network, 0, len(s.networks)) + for _, network := range s.networks { + result = append(result, *network) + } + s.netMut.RUnlock() + w.Header().Set("Content-Type", "application/json") + w.WriteHeader(http.StatusOK) + json.NewEncoder(w).Encode(result) +} + +func (s *DockerServer) networkInfo(w http.ResponseWriter, r *http.Request) { + id := mux.Vars(r)["id"] + network, _, err := s.findNetwork(id) + if err != nil { + http.Error(w, err.Error(), http.StatusNotFound) + return + } + w.Header().Set("Content-Type", "application/json") + w.WriteHeader(http.StatusOK) + json.NewEncoder(w).Encode(network) +} + +// isValidName validates configuration objects supported by libnetwork +func isValidName(name string) bool { + if name == "" || strings.Contains(name, ".") { + return false + } + return true +} + +func (s *DockerServer) createNetwork(w http.ResponseWriter, r *http.Request) { + var config *docker.CreateNetworkOptions + defer r.Body.Close() + err := json.NewDecoder(r.Body).Decode(&config) + if err != nil { + http.Error(w, err.Error(), http.StatusBadRequest) + return + } + if !isValidName(config.Name) { + http.Error(w, "Invalid network name", http.StatusBadRequest) + return + } + if n, _, _ := s.findNetwork(config.Name); n != nil { + http.Error(w, "network already exists", http.StatusForbidden) + return + } + + generatedID := s.generateID() + network := docker.Network{ + Name: config.Name, + ID: generatedID, + Type: config.NetworkType, + } + s.netMut.Lock() + s.networks = append(s.networks, &network) + s.netMut.Unlock() + w.WriteHeader(http.StatusCreated) + var c = struct{ ID string }{ID: network.ID} + json.NewEncoder(w).Encode(c) +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server_test.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server_test.go new file mode 100644 index 0000000000000..36789abb36665 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/testing/server_test.go @@ -0,0 +1,1784 @@ +// Copyright 2015 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package testing + +import ( + "bytes" + "encoding/json" + "fmt" + "math/rand" + "net" + "net/http" + "net/http/httptest" + "os" + "reflect" + "strings" + "sync" + "testing" + "time" + + "github.com/fsouza/go-dockerclient" +) + +func TestNewServer(t *testing.T) { + server, err := NewServer("127.0.0.1:0", nil, nil) + if err != nil { + t.Fatal(err) + } + defer server.listener.Close() + conn, err := net.Dial("tcp", server.listener.Addr().String()) + if err != nil { + t.Fatal(err) + } + conn.Close() +} + +func TestServerStop(t *testing.T) { + server, err := NewServer("127.0.0.1:0", nil, nil) + if err != nil { + t.Fatal(err) + } + server.Stop() + _, err = net.Dial("tcp", server.listener.Addr().String()) + if err == nil { + t.Error("Unexpected error when dialing to stopped server") + } +} + +func TestServerStopNoListener(t *testing.T) { + server := DockerServer{} + server.Stop() +} + +func TestServerURL(t *testing.T) { + server, err := NewServer("127.0.0.1:0", nil, nil) + if err != nil { + t.Fatal(err) + } + defer server.Stop() + url := server.URL() + if expected := "http://" + server.listener.Addr().String() + "/"; url != expected { + t.Errorf("DockerServer.URL(): Want %q. Got %q.", expected, url) + } +} + +func TestServerURLNoListener(t *testing.T) { + server := DockerServer{} + url := server.URL() + if url != "" { + t.Errorf("DockerServer.URL(): Expected empty URL on handler mode, got %q.", url) + } +} + +func TestHandleWithHook(t *testing.T) { + var called bool + server, _ := NewServer("127.0.0.1:0", nil, func(*http.Request) { called = true }) + defer server.Stop() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("GET", "/containers/json?all=1", nil) + server.ServeHTTP(recorder, request) + if !called { + t.Error("ServeHTTP did not call the hook function.") + } +} + +func TestSetHook(t *testing.T) { + var called bool + server, _ := NewServer("127.0.0.1:0", nil, nil) + defer server.Stop() + server.SetHook(func(*http.Request) { called = true }) + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("GET", "/containers/json?all=1", nil) + server.ServeHTTP(recorder, request) + if !called { + t.Error("ServeHTTP did not call the hook function.") + } +} + +func TestCustomHandler(t *testing.T) { + var called bool + server, _ := NewServer("127.0.0.1:0", nil, nil) + addContainers(server, 2) + server.CustomHandler("/containers/json", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + called = true + fmt.Fprint(w, "Hello world") + })) + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("GET", "/containers/json?all=1", nil) + server.ServeHTTP(recorder, request) + if !called { + t.Error("Did not call the custom handler") + } + if got := recorder.Body.String(); got != "Hello world" { + t.Errorf("Wrong output for custom handler: want %q. Got %q.", "Hello world", got) + } +} + +func TestCustomHandlerRegexp(t *testing.T) { + var called bool + server, _ := NewServer("127.0.0.1:0", nil, nil) + addContainers(server, 2) + server.CustomHandler("/containers/.*/json", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + called = true + fmt.Fprint(w, "Hello world") + })) + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("GET", "/containers/.*/json?all=1", nil) + server.ServeHTTP(recorder, request) + if !called { + t.Error("Did not call the custom handler") + } + if got := recorder.Body.String(); got != "Hello world" { + t.Errorf("Wrong output for custom handler: want %q. Got %q.", "Hello world", got) + } +} + +func TestListContainers(t *testing.T) { + server := DockerServer{} + addContainers(&server, 2) + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("GET", "/containers/json?all=1", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("ListContainers: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + expected := make([]docker.APIContainers, 2) + for i, container := range server.containers { + expected[i] = docker.APIContainers{ + ID: container.ID, + Image: container.Image, + Command: strings.Join(container.Config.Cmd, " "), + Created: container.Created.Unix(), + Status: container.State.String(), + Ports: container.NetworkSettings.PortMappingAPI(), + Names: []string{"/" + container.Name}, + } + } + var got []docker.APIContainers + err := json.NewDecoder(recorder.Body).Decode(&got) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(got, expected) { + t.Errorf("ListContainers. Want %#v. Got %#v.", expected, got) + } +} + +func TestListRunningContainers(t *testing.T) { + server := DockerServer{} + addContainers(&server, 2) + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("GET", "/containers/json?all=0", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("ListRunningContainers: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + var got []docker.APIContainers + err := json.NewDecoder(recorder.Body).Decode(&got) + if err != nil { + t.Fatal(err) + } + if len(got) != 0 { + t.Errorf("ListRunningContainers: Want 0. Got %d.", len(got)) + } +} + +func TestCreateContainer(t *testing.T) { + server := DockerServer{} + server.imgIDs = map[string]string{"base": "a1234"} + server.buildMuxer() + recorder := httptest.NewRecorder() + body := `{"Hostname":"", "User":"ubuntu", "Memory":0, "MemorySwap":0, "AttachStdin":false, "AttachStdout":true, "AttachStderr":true, +"PortSpecs":null, "Tty":false, "OpenStdin":false, "StdinOnce":false, "Env":null, "Cmd":["date"], "Image":"base", "Volumes":{}, "VolumesFrom":"","HostConfig":{"Binds":["/var/run/docker.sock:/var/run/docker.sock:rw"]}}` + request, _ := http.NewRequest("POST", "/containers/create", strings.NewReader(body)) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusCreated { + t.Errorf("CreateContainer: wrong status. Want %d. Got %d.", http.StatusCreated, recorder.Code) + } + var returned docker.Container + err := json.NewDecoder(recorder.Body).Decode(&returned) + if err != nil { + t.Fatal(err) + } + stored := server.containers[0] + if returned.ID != stored.ID { + t.Errorf("CreateContainer: ID mismatch. Stored: %q. Returned: %q.", stored.ID, returned.ID) + } + if stored.State.Running { + t.Errorf("CreateContainer should not set container to running state.") + } + if stored.Config.User != "ubuntu" { + t.Errorf("CreateContainer: wrong config. Expected: %q. Returned: %q.", "ubuntu", stored.Config.User) + } + if stored.Config.Hostname != returned.ID[:12] { + t.Errorf("CreateContainer: wrong hostname. Expected: %q. Returned: %q.", returned.ID[:12], stored.Config.Hostname) + } + expectedBind := []string{"/var/run/docker.sock:/var/run/docker.sock:rw"} + if !reflect.DeepEqual(stored.HostConfig.Binds, expectedBind) { + t.Errorf("CreateContainer: wrong host config. Expected: %v. Returned %v.", expectedBind, stored.HostConfig.Binds) + } +} + +func TestCreateContainerWithNotifyChannel(t *testing.T) { + ch := make(chan *docker.Container, 1) + server := DockerServer{} + server.imgIDs = map[string]string{"base": "a1234"} + server.cChan = ch + server.buildMuxer() + recorder := httptest.NewRecorder() + body := `{"Hostname":"", "User":"", "Memory":0, "MemorySwap":0, "AttachStdin":false, "AttachStdout":true, "AttachStderr":true, +"PortSpecs":null, "Tty":false, "OpenStdin":false, "StdinOnce":false, "Env":null, "Cmd":["date"], "Image":"base", "Volumes":{}, "VolumesFrom":""}` + request, _ := http.NewRequest("POST", "/containers/create", strings.NewReader(body)) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusCreated { + t.Errorf("CreateContainer: wrong status. Want %d. Got %d.", http.StatusCreated, recorder.Code) + } + if notified := <-ch; notified != server.containers[0] { + t.Errorf("CreateContainer: did not notify the proper container. Want %q. Got %q.", server.containers[0].ID, notified.ID) + } +} + +func TestCreateContainerInvalidBody(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/containers/create", strings.NewReader("whaaaaaat---")) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusBadRequest { + t.Errorf("CreateContainer: wrong status. Want %d. Got %d.", http.StatusBadRequest, recorder.Code) + } +} + +func TestCreateContainerDuplicateName(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + server.imgIDs = map[string]string{"base": "a1234"} + addContainers(&server, 1) + server.containers[0].Name = "mycontainer" + recorder := httptest.NewRecorder() + body := `{"Hostname":"", "User":"ubuntu", "Memory":0, "MemorySwap":0, "AttachStdin":false, "AttachStdout":true, "AttachStderr":true, +"PortSpecs":null, "Tty":false, "OpenStdin":false, "StdinOnce":false, "Env":null, "Cmd":["date"], "Image":"base", "Volumes":{}, "VolumesFrom":"","HostConfig":{"Binds":["/var/run/docker.sock:/var/run/docker.sock:rw"]}}` + request, _ := http.NewRequest("POST", "/containers/create?name=mycontainer", strings.NewReader(body)) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusConflict { + t.Errorf("CreateContainer: wrong status. Want %d. Got %d.", http.StatusConflict, recorder.Code) + } +} + +func TestCreateMultipleContainersEmptyName(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + server.imgIDs = map[string]string{"base": "a1234"} + addContainers(&server, 1) + server.containers[0].Name = "" + recorder := httptest.NewRecorder() + body := `{"Hostname":"", "User":"ubuntu", "Memory":0, "MemorySwap":0, "AttachStdin":false, "AttachStdout":true, "AttachStderr":true, +"PortSpecs":null, "Tty":false, "OpenStdin":false, "StdinOnce":false, "Env":null, "Cmd":["date"], "Image":"base", "Volumes":{}, "VolumesFrom":"","HostConfig":{"Binds":["/var/run/docker.sock:/var/run/docker.sock:rw"]}}` + request, _ := http.NewRequest("POST", "/containers/create", strings.NewReader(body)) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusCreated { + t.Errorf("CreateContainer: wrong status. Want %d. Got %d.", http.StatusCreated, recorder.Code) + } + var returned docker.Container + err := json.NewDecoder(recorder.Body).Decode(&returned) + if err != nil { + t.Fatal(err) + } + stored := server.containers[1] + if returned.ID != stored.ID { + t.Errorf("CreateContainer: ID mismatch. Stored: %q. Returned: %q.", stored.ID, returned.ID) + } + if stored.State.Running { + t.Errorf("CreateContainer should not set container to running state.") + } + if stored.Config.User != "ubuntu" { + t.Errorf("CreateContainer: wrong config. Expected: %q. Returned: %q.", "ubuntu", stored.Config.User) + } + expectedBind := []string{"/var/run/docker.sock:/var/run/docker.sock:rw"} + if !reflect.DeepEqual(stored.HostConfig.Binds, expectedBind) { + t.Errorf("CreateContainer: wrong host config. Expected: %v. Returned %v.", expectedBind, stored.HostConfig.Binds) + } +} + +func TestCreateContainerInvalidName(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + body := `{"Hostname":"", "User":"", "Memory":0, "MemorySwap":0, "AttachStdin":false, "AttachStdout":true, "AttachStderr":true, +"PortSpecs":null, "Tty":false, "OpenStdin":false, "StdinOnce":false, "Env":null, "Cmd":["date"], +"Image":"base", "Volumes":{}, "VolumesFrom":""}` + request, _ := http.NewRequest("POST", "/containers/create?name=myapp/container1", strings.NewReader(body)) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusInternalServerError { + t.Errorf("CreateContainer: wrong status. Want %d. Got %d.", http.StatusInternalServerError, recorder.Code) + } + expectedBody := "Invalid container name\n" + if got := recorder.Body.String(); got != expectedBody { + t.Errorf("CreateContainer: wrong body. Want %q. Got %q.", expectedBody, got) + } +} + +func TestCreateContainerImageNotFound(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + body := `{"Hostname":"", "User":"", "Memory":0, "MemorySwap":0, "AttachStdin":false, "AttachStdout":true, "AttachStderr":true, +"PortSpecs":null, "Tty":false, "OpenStdin":false, "StdinOnce":false, "Env":null, "Cmd":["date"], +"Image":"base", "Volumes":{}, "VolumesFrom":""}` + request, _ := http.NewRequest("POST", "/containers/create", strings.NewReader(body)) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNotFound { + t.Errorf("CreateContainer: wrong status. Want %d. Got %d.", http.StatusNotFound, recorder.Code) + } +} + +func TestRenameContainer(t *testing.T) { + server := DockerServer{} + addContainers(&server, 2) + server.buildMuxer() + recorder := httptest.NewRecorder() + newName := server.containers[0].Name + "abc" + path := fmt.Sprintf("/containers/%s/rename?name=%s", server.containers[0].ID, newName) + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNoContent { + t.Errorf("RenameContainer: wrong status. Want %d. Got %d.", http.StatusNoContent, recorder.Code) + } + container := server.containers[0] + if container.Name != newName { + t.Errorf("RenameContainer: did not rename the container. Want %q. Got %q.", newName, container.Name) + } +} + +func TestRenameContainerNotFound(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/containers/blabla/rename?name=something", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNotFound { + t.Errorf("RenameContainer: wrong status. Want %d. Got %d.", http.StatusNotFound, recorder.Code) + } +} + +func TestCommitContainer(t *testing.T) { + server := DockerServer{} + addContainers(&server, 2) + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/commit?container="+server.containers[0].ID, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("CommitContainer: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + expected := fmt.Sprintf(`{"ID":"%s"}`, server.images[0].ID) + if got := recorder.Body.String(); got != expected { + t.Errorf("CommitContainer: wrong response body. Want %q. Got %q.", expected, got) + } +} + +func TestCommitContainerComplete(t *testing.T) { + server := DockerServer{} + server.imgIDs = make(map[string]string) + addContainers(&server, 2) + server.buildMuxer() + recorder := httptest.NewRecorder() + queryString := "container=" + server.containers[0].ID + "&repo=tsuru/python&m=saving&author=developers" + queryString += `&run={"Cmd": ["cat", "/world"],"PortSpecs":["22"]}` + request, _ := http.NewRequest("POST", "/commit?"+queryString, nil) + server.ServeHTTP(recorder, request) + image := server.images[0] + if image.Parent != server.containers[0].Image { + t.Errorf("CommitContainer: wrong parent image. Want %q. Got %q.", server.containers[0].Image, image.Parent) + } + if image.Container != server.containers[0].ID { + t.Errorf("CommitContainer: wrong container. Want %q. Got %q.", server.containers[0].ID, image.Container) + } + message := "saving" + if image.Comment != message { + t.Errorf("CommitContainer: wrong comment (commit message). Want %q. Got %q.", message, image.Comment) + } + author := "developers" + if image.Author != author { + t.Errorf("CommitContainer: wrong author. Want %q. Got %q.", author, image.Author) + } + if id := server.imgIDs["tsuru/python"]; id != image.ID { + t.Errorf("CommitContainer: wrong ID saved for repository. Want %q. Got %q.", image.ID, id) + } + portSpecs := []string{"22"} + if !reflect.DeepEqual(image.Config.PortSpecs, portSpecs) { + t.Errorf("CommitContainer: wrong port spec in config. Want %#v. Got %#v.", portSpecs, image.Config.PortSpecs) + } + cmd := []string{"cat", "/world"} + if !reflect.DeepEqual(image.Config.Cmd, cmd) { + t.Errorf("CommitContainer: wrong cmd in config. Want %#v. Got %#v.", cmd, image.Config.Cmd) + } +} + +func TestCommitContainerWithTag(t *testing.T) { + server := DockerServer{} + server.imgIDs = make(map[string]string) + addContainers(&server, 2) + server.buildMuxer() + recorder := httptest.NewRecorder() + queryString := "container=" + server.containers[0].ID + "&repo=tsuru/python&tag=v1" + request, _ := http.NewRequest("POST", "/commit?"+queryString, nil) + server.ServeHTTP(recorder, request) + image := server.images[0] + if image.Parent != server.containers[0].Image { + t.Errorf("CommitContainer: wrong parent image. Want %q. Got %q.", server.containers[0].Image, image.Parent) + } + if image.Container != server.containers[0].ID { + t.Errorf("CommitContainer: wrong container. Want %q. Got %q.", server.containers[0].ID, image.Container) + } + if id := server.imgIDs["tsuru/python:v1"]; id != image.ID { + t.Errorf("CommitContainer: wrong ID saved for repository. Want %q. Got %q.", image.ID, id) + } +} + +func TestCommitContainerInvalidRun(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/commit?container="+server.containers[0].ID+"&run=abc---", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusBadRequest { + t.Errorf("CommitContainer. Wrong status. Want %d. Got %d.", http.StatusBadRequest, recorder.Code) + } +} + +func TestCommitContainerNotFound(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/commit?container=abc123", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNotFound { + t.Errorf("CommitContainer. Wrong status. Want %d. Got %d.", http.StatusNotFound, recorder.Code) + } +} + +func TestInspectContainer(t *testing.T) { + server := DockerServer{} + addContainers(&server, 2) + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/json", server.containers[0].ID) + request, _ := http.NewRequest("GET", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("InspectContainer: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + expected := server.containers[0] + var got docker.Container + err := json.NewDecoder(recorder.Body).Decode(&got) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(got.Config, expected.Config) { + t.Errorf("InspectContainer: wrong value. Want %#v. Got %#v.", *expected, got) + } + if !reflect.DeepEqual(got.NetworkSettings, expected.NetworkSettings) { + t.Errorf("InspectContainer: wrong value. Want %#v. Got %#v.", *expected, got) + } + got.State.StartedAt = expected.State.StartedAt + got.State.FinishedAt = expected.State.FinishedAt + got.Config = expected.Config + got.Created = expected.Created + got.NetworkSettings = expected.NetworkSettings + if !reflect.DeepEqual(got, *expected) { + t.Errorf("InspectContainer: wrong value. Want %#v. Got %#v.", *expected, got) + } +} + +func TestInspectContainerNotFound(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("GET", "/containers/abc123/json", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNotFound { + t.Errorf("InspectContainer: wrong status code. Want %d. Got %d.", http.StatusNotFound, recorder.Code) + } +} + +func TestTopContainer(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.containers[0].State.Running = true + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/top", server.containers[0].ID) + request, _ := http.NewRequest("GET", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("TopContainer: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + var got docker.TopResult + err := json.NewDecoder(recorder.Body).Decode(&got) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(got.Titles, []string{"UID", "PID", "PPID", "C", "STIME", "TTY", "TIME", "CMD"}) { + t.Fatalf("TopContainer: Unexpected titles, got: %#v", got.Titles) + } + if len(got.Processes) != 1 { + t.Fatalf("TopContainer: Unexpected process len, got: %d", len(got.Processes)) + } + if got.Processes[0][len(got.Processes[0])-1] != "ls -la .." { + t.Fatalf("TopContainer: Unexpected command name, got: %s", got.Processes[0][len(got.Processes[0])-1]) + } +} + +func TestTopContainerNotFound(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("GET", "/containers/xyz/top", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNotFound { + t.Errorf("TopContainer: wrong status. Want %d. Got %d.", http.StatusNotFound, recorder.Code) + } +} + +func TestTopContainerStopped(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/top", server.containers[0].ID) + request, _ := http.NewRequest("GET", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusInternalServerError { + t.Errorf("TopContainer: wrong status. Want %d. Got %d.", http.StatusInternalServerError, recorder.Code) + } +} + +func TestStartContainer(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.buildMuxer() + memory := int64(536870912) + hostConfig := docker.HostConfig{Memory: memory} + configBytes, err := json.Marshal(hostConfig) + if err != nil { + t.Fatal(err) + } + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/start", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, bytes.NewBuffer(configBytes)) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("StartContainer: wrong status code. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + if !server.containers[0].State.Running { + t.Error("StartContainer: did not set the container to running state") + } + if gotMemory := server.containers[0].HostConfig.Memory; gotMemory != memory { + t.Errorf("StartContainer: wrong HostConfig. Wants %d of memory. Got %d", memory, gotMemory) + } +} + +func TestStartContainerWithNotifyChannel(t *testing.T) { + ch := make(chan *docker.Container, 1) + server := DockerServer{} + server.cChan = ch + addContainers(&server, 1) + addContainers(&server, 1) + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/start", server.containers[1].ID) + request, _ := http.NewRequest("POST", path, bytes.NewBuffer([]byte("{}"))) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("StartContainer: wrong status code. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + if notified := <-ch; notified != server.containers[1] { + t.Errorf("StartContainer: did not notify the proper container. Want %q. Got %q.", server.containers[1].ID, notified.ID) + } +} + +func TestStartContainerNotFound(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + path := "/containers/abc123/start" + request, _ := http.NewRequest("POST", path, bytes.NewBuffer([]byte("null"))) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNotFound { + t.Errorf("StartContainer: wrong status code. Want %d. Got %d.", http.StatusNotFound, recorder.Code) + } +} + +func TestStartContainerAlreadyRunning(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.containers[0].State.Running = true + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/start", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, bytes.NewBuffer([]byte("null"))) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusBadRequest { + t.Errorf("StartContainer: wrong status code. Want %d. Got %d.", http.StatusBadRequest, recorder.Code) + } +} + +func TestStopContainer(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.containers[0].State.Running = true + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/stop", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNoContent { + t.Errorf("StopContainer: wrong status code. Want %d. Got %d.", http.StatusNoContent, recorder.Code) + } + if server.containers[0].State.Running { + t.Error("StopContainer: did not stop the container") + } +} + +func TestKillContainer(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.containers[0].State.Running = true + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/kill", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNoContent { + t.Errorf("KillContainer: wrong status code. Want %d. Got %d.", http.StatusNoContent, recorder.Code) + } + if server.containers[0].State.Running { + t.Error("KillContainer: did not stop the container") + } +} + +func TestStopContainerWithNotifyChannel(t *testing.T) { + ch := make(chan *docker.Container, 1) + server := DockerServer{} + server.cChan = ch + addContainers(&server, 1) + addContainers(&server, 1) + server.containers[1].State.Running = true + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/stop", server.containers[1].ID) + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNoContent { + t.Errorf("StopContainer: wrong status code. Want %d. Got %d.", http.StatusNoContent, recorder.Code) + } + if notified := <-ch; notified != server.containers[1] { + t.Errorf("StopContainer: did not notify the proper container. Want %q. Got %q.", server.containers[1].ID, notified.ID) + } +} + +func TestStopContainerNotFound(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + path := "/containers/abc123/stop" + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNotFound { + t.Errorf("StopContainer: wrong status code. Want %d. Got %d.", http.StatusNotFound, recorder.Code) + } +} + +func TestStopContainerNotRunning(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/stop", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusBadRequest { + t.Errorf("StopContainer: wrong status code. Want %d. Got %d.", http.StatusBadRequest, recorder.Code) + } +} + +func TestPauseContainer(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/pause", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNoContent { + t.Errorf("PauseContainer: wrong status code. Want %d. Got %d.", http.StatusNoContent, recorder.Code) + } + if !server.containers[0].State.Paused { + t.Error("PauseContainer: did not pause the container") + } +} + +func TestPauseContainerAlreadyPaused(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.containers[0].State.Paused = true + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/pause", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusBadRequest { + t.Errorf("PauseContainer: wrong status code. Want %d. Got %d.", http.StatusBadRequest, recorder.Code) + } +} + +func TestPauseContainerNotFound(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + path := "/containers/abc123/pause" + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNotFound { + t.Errorf("PauseContainer: wrong status code. Want %d. Got %d.", http.StatusNotFound, recorder.Code) + } +} + +func TestUnpauseContainer(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.containers[0].State.Paused = true + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/unpause", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNoContent { + t.Errorf("UnpauseContainer: wrong status code. Want %d. Got %d.", http.StatusNoContent, recorder.Code) + } + if server.containers[0].State.Paused { + t.Error("UnpauseContainer: did not unpause the container") + } +} + +func TestUnpauseContainerNotPaused(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/unpause", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusBadRequest { + t.Errorf("UnpauseContainer: wrong status code. Want %d. Got %d.", http.StatusBadRequest, recorder.Code) + } +} + +func TestUnpauseContainerNotFound(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + path := "/containers/abc123/unpause" + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNotFound { + t.Errorf("UnpauseContainer: wrong status code. Want %d. Got %d.", http.StatusNotFound, recorder.Code) + } +} + +func TestWaitContainer(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.containers[0].State.Running = true + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/wait", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, nil) + go func() { + server.cMut.Lock() + server.containers[0].State.Running = false + server.cMut.Unlock() + }() + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("WaitContainer: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + expected := `{"StatusCode":0}` + "\n" + if body := recorder.Body.String(); body != expected { + t.Errorf("WaitContainer: wrong body. Want %q. Got %q.", expected, body) + } +} + +func TestWaitContainerStatus(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.buildMuxer() + server.containers[0].State.ExitCode = 63 + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/wait", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("WaitContainer: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + expected := `{"StatusCode":63}` + "\n" + if body := recorder.Body.String(); body != expected { + t.Errorf("WaitContainer: wrong body. Want %q. Got %q.", expected, body) + } +} + +func TestWaitContainerNotFound(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + path := "/containers/abc123/wait" + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNotFound { + t.Errorf("WaitContainer: wrong status code. Want %d. Got %d.", http.StatusNotFound, recorder.Code) + } +} + +func TestAttachContainer(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.containers[0].State.Running = true + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/attach?logs=1", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + lines := []string{ + fmt.Sprintf("\x01\x00\x00\x00\x03\x00\x00\x00Container %q is running", server.containers[0].ID), + "What happened?", + "Something happened", + } + expected := strings.Join(lines, "\n") + "\n" + if body := recorder.Body.String(); body == expected { + t.Errorf("AttachContainer: wrong body. Want %q. Got %q.", expected, body) + } +} + +func TestAttachContainerNotFound(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + path := "/containers/abc123/attach?logs=1" + request, _ := http.NewRequest("POST", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNotFound { + t.Errorf("AttachContainer: wrong status. Want %d. Got %d.", http.StatusNotFound, recorder.Code) + } +} + +func TestRemoveContainer(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s", server.containers[0].ID) + request, _ := http.NewRequest("DELETE", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNoContent { + t.Errorf("RemoveContainer: wrong status. Want %d. Got %d.", http.StatusNoContent, recorder.Code) + } + if len(server.containers) > 0 { + t.Error("RemoveContainer: did not remove the container.") + } +} + +func TestRemoveContainerByName(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s", server.containers[0].Name) + request, _ := http.NewRequest("DELETE", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNoContent { + t.Errorf("RemoveContainer: wrong status. Want %d. Got %d.", http.StatusNoContent, recorder.Code) + } + if len(server.containers) > 0 { + t.Error("RemoveContainer: did not remove the container.") + } +} + +func TestRemoveContainerNotFound(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/abc123") + request, _ := http.NewRequest("DELETE", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNotFound { + t.Errorf("RemoveContainer: wrong status. Want %d. Got %d.", http.StatusNotFound, recorder.Code) + } +} + +func TestRemoveContainerRunning(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.containers[0].State.Running = true + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s", server.containers[0].ID) + request, _ := http.NewRequest("DELETE", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusInternalServerError { + t.Errorf("RemoveContainer: wrong status. Want %d. Got %d.", http.StatusInternalServerError, recorder.Code) + } + if len(server.containers) < 1 { + t.Error("RemoveContainer: should not remove the container.") + } +} + +func TestRemoveContainerRunningForce(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.containers[0].State.Running = true + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s?%s", server.containers[0].ID, "force=1") + request, _ := http.NewRequest("DELETE", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNoContent { + t.Errorf("RemoveContainer: wrong status. Want %d. Got %d.", http.StatusNoContent, recorder.Code) + } + if len(server.containers) > 0 { + t.Error("RemoveContainer: did not remove the container.") + } +} + +func TestPullImage(t *testing.T) { + server := DockerServer{imgIDs: make(map[string]string)} + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/images/create?fromImage=base", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("PullImage: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + if len(server.images) != 1 { + t.Errorf("PullImage: Want 1 image. Got %d.", len(server.images)) + } + if _, ok := server.imgIDs["base"]; !ok { + t.Error("PullImage: Repository should not be empty.") + } +} + +func TestPullImageWithTag(t *testing.T) { + server := DockerServer{imgIDs: make(map[string]string)} + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/images/create?fromImage=base&tag=tag", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("PullImage: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + if len(server.images) != 1 { + t.Errorf("PullImage: Want 1 image. Got %d.", len(server.images)) + } + if _, ok := server.imgIDs["base:tag"]; !ok { + t.Error("PullImage: Repository should not be empty.") + } +} + +func TestPushImage(t *testing.T) { + server := DockerServer{imgIDs: map[string]string{"tsuru/python": "a123"}} + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/images/tsuru/python/push", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("PushImage: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } +} + +func TestPushImageWithTag(t *testing.T) { + server := DockerServer{imgIDs: map[string]string{"tsuru/python:v1": "a123"}} + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/images/tsuru/python/push?tag=v1", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("PushImage: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } +} + +func TestPushImageNotFound(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/images/tsuru/python/push", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNotFound { + t.Errorf("PushImage: wrong status. Want %d. Got %d.", http.StatusNotFound, recorder.Code) + } +} + +func TestTagImage(t *testing.T) { + server := DockerServer{imgIDs: map[string]string{"tsuru/python": "a123"}} + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/images/tsuru/python/tag?repo=tsuru/new-python", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusCreated { + t.Errorf("TagImage: wrong status. Want %d. Got %d.", http.StatusCreated, recorder.Code) + } + if server.imgIDs["tsuru/python"] != server.imgIDs["tsuru/new-python"] { + t.Errorf("TagImage: did not tag the image") + } +} + +func TestTagImageWithRepoAndTag(t *testing.T) { + server := DockerServer{imgIDs: map[string]string{"tsuru/python": "a123"}} + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/images/tsuru/python/tag?repo=tsuru/new-python&tag=v1", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusCreated { + t.Errorf("TagImage: wrong status. Want %d. Got %d.", http.StatusCreated, recorder.Code) + } + if server.imgIDs["tsuru/python"] != server.imgIDs["tsuru/new-python:v1"] { + t.Errorf("TagImage: did not tag the image") + } +} + +func TestTagImageNotFound(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/images/tsuru/python/tag", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNotFound { + t.Errorf("TagImage: wrong status. Want %d. Got %d.", http.StatusNotFound, recorder.Code) + } +} + +func addContainers(server *DockerServer, n int) { + server.cMut.Lock() + defer server.cMut.Unlock() + for i := 0; i < n; i++ { + date := time.Now().Add(time.Duration((rand.Int() % (i + 1))) * time.Hour) + container := docker.Container{ + Name: fmt.Sprintf("%x", rand.Int()%10000), + ID: fmt.Sprintf("%x", rand.Int()%10000), + Created: date, + Path: "ls", + Args: []string{"-la", ".."}, + Config: &docker.Config{ + Hostname: fmt.Sprintf("docker-%d", i), + AttachStdout: true, + AttachStderr: true, + Env: []string{"ME=you", fmt.Sprintf("NUMBER=%d", i)}, + Cmd: []string{"ls", "-la", ".."}, + Image: "base", + }, + State: docker.State{ + Running: false, + Pid: 400 + i, + ExitCode: 0, + StartedAt: date, + }, + Image: "b750fe79269d2ec9a3c593ef05b4332b1d1a02a62b4accb2c21d589ff2f5f2dc", + NetworkSettings: &docker.NetworkSettings{ + IPAddress: fmt.Sprintf("10.10.10.%d", i+2), + IPPrefixLen: 24, + Gateway: "10.10.10.1", + Bridge: "docker0", + PortMapping: map[string]docker.PortMapping{ + "Tcp": {"8888": fmt.Sprintf("%d", 49600+i)}, + }, + }, + ResolvConfPath: "/etc/resolv.conf", + } + server.containers = append(server.containers, &container) + } +} + +func addImages(server *DockerServer, n int, repo bool) { + server.iMut.Lock() + defer server.iMut.Unlock() + if server.imgIDs == nil { + server.imgIDs = make(map[string]string) + } + for i := 0; i < n; i++ { + date := time.Now().Add(time.Duration((rand.Int() % (i + 1))) * time.Hour) + image := docker.Image{ + ID: fmt.Sprintf("%x", rand.Int()%10000), + Created: date, + } + server.images = append(server.images, image) + if repo { + repo := "docker/python-" + image.ID + server.imgIDs[repo] = image.ID + } + } +} + +func TestListImages(t *testing.T) { + server := DockerServer{} + addImages(&server, 2, true) + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("GET", "/images/json?all=1", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("ListImages: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + expected := make([]docker.APIImages, 2) + for i, image := range server.images { + expected[i] = docker.APIImages{ + ID: image.ID, + Created: image.Created.Unix(), + RepoTags: []string{"docker/python-" + image.ID}, + } + } + var got []docker.APIImages + err := json.NewDecoder(recorder.Body).Decode(&got) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(got, expected) { + t.Errorf("ListImages. Want %#v. Got %#v.", expected, got) + } +} + +func TestRemoveImage(t *testing.T) { + server := DockerServer{} + addImages(&server, 1, false) + server.buildMuxer() + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/images/%s", server.images[0].ID) + request, _ := http.NewRequest("DELETE", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNoContent { + t.Errorf("RemoveImage: wrong status. Want %d. Got %d.", http.StatusNoContent, recorder.Code) + } + if len(server.images) > 0 { + t.Error("RemoveImage: did not remove the image.") + } +} + +func TestRemoveImageByName(t *testing.T) { + server := DockerServer{} + addImages(&server, 1, true) + server.buildMuxer() + recorder := httptest.NewRecorder() + imgName := "docker/python-" + server.images[0].ID + path := "/images/" + imgName + request, _ := http.NewRequest("DELETE", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusNoContent { + t.Errorf("RemoveImage: wrong status. Want %d. Got %d.", http.StatusNoContent, recorder.Code) + } + if len(server.images) > 0 { + t.Error("RemoveImage: did not remove the image.") + } + _, ok := server.imgIDs[imgName] + if ok { + t.Error("RemoveImage: did not remove image tag name.") + } +} + +func TestRemoveImageWithMultipleTags(t *testing.T) { + server := DockerServer{} + addImages(&server, 1, true) + server.buildMuxer() + imgID := server.images[0].ID + imgName := "docker/python-" + imgID + server.imgIDs["docker/python-wat"] = imgID + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/images/%s", imgName) + request, _ := http.NewRequest("DELETE", path, nil) + server.ServeHTTP(recorder, request) + _, ok := server.imgIDs[imgName] + if ok { + t.Error("RemoveImage: did not remove image tag name.") + } + id, ok := server.imgIDs["docker/python-wat"] + if !ok { + t.Error("RemoveImage: removed the wrong tag name.") + } + if id != imgID { + t.Error("RemoveImage: disassociated the wrong ID from the tag") + } + if len(server.images) < 1 { + t.Fatal("RemoveImage: removed the image, but should keep it") + } + if server.images[0].ID != imgID { + t.Error("RemoveImage: changed the ID of the image!") + } +} + +func TestPrepareFailure(t *testing.T) { + server := DockerServer{failures: make(map[string]string)} + server.buildMuxer() + errorID := "my_error" + server.PrepareFailure(errorID, "containers/json") + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("GET", "/containers/json?all=1", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusBadRequest { + t.Errorf("PrepareFailure: wrong status. Want %d. Got %d.", http.StatusBadRequest, recorder.Code) + } + if recorder.Body.String() != errorID+"\n" { + t.Errorf("PrepareFailure: wrong message. Want %s. Got %s.", errorID, recorder.Body.String()) + } +} + +func TestPrepareMultiFailures(t *testing.T) { + server := DockerServer{multiFailures: []map[string]string{}} + server.buildMuxer() + errorID := "multi error" + server.PrepareMultiFailures(errorID, "containers/json") + server.PrepareMultiFailures(errorID, "containers/json") + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("GET", "/containers/json?all=1", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusBadRequest { + t.Errorf("PrepareFailure: wrong status. Want %d. Got %d.", http.StatusBadRequest, recorder.Code) + } + if recorder.Body.String() != errorID+"\n" { + t.Errorf("PrepareFailure: wrong message. Want %s. Got %s.", errorID, recorder.Body.String()) + } + recorder = httptest.NewRecorder() + request, _ = http.NewRequest("GET", "/containers/json?all=1", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusBadRequest { + t.Errorf("PrepareFailure: wrong status. Want %d. Got %d.", http.StatusBadRequest, recorder.Code) + } + if recorder.Body.String() != errorID+"\n" { + t.Errorf("PrepareFailure: wrong message. Want %s. Got %s.", errorID, recorder.Body.String()) + } + recorder = httptest.NewRecorder() + request, _ = http.NewRequest("GET", "/containers/json?all=1", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("PrepareFailure: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + if recorder.Body.String() == errorID+"\n" { + t.Errorf("PrepareFailure: wrong message. Want %s. Got %s.", errorID, recorder.Body.String()) + } +} + +func TestRemoveFailure(t *testing.T) { + server := DockerServer{failures: make(map[string]string)} + server.buildMuxer() + errorID := "my_error" + server.PrepareFailure(errorID, "containers/json") + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("GET", "/containers/json?all=1", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusBadRequest { + t.Errorf("PrepareFailure: wrong status. Want %d. Got %d.", http.StatusBadRequest, recorder.Code) + } + server.ResetFailure(errorID) + recorder = httptest.NewRecorder() + request, _ = http.NewRequest("GET", "/containers/json?all=1", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("RemoveFailure: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } +} + +func TestResetMultiFailures(t *testing.T) { + server := DockerServer{multiFailures: []map[string]string{}} + server.buildMuxer() + errorID := "multi error" + server.PrepareMultiFailures(errorID, "containers/json") + server.PrepareMultiFailures(errorID, "containers/json") + if len(server.multiFailures) != 2 { + t.Errorf("PrepareMultiFailures: error adding multi failures.") + } + server.ResetMultiFailures() + if len(server.multiFailures) != 0 { + t.Errorf("ResetMultiFailures: error reseting multi failures.") + } +} + +func TestMutateContainer(t *testing.T) { + server := DockerServer{failures: make(map[string]string)} + server.buildMuxer() + server.containers = append(server.containers, &docker.Container{ID: "id123"}) + state := docker.State{Running: false, ExitCode: 1} + err := server.MutateContainer("id123", state) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(server.containers[0].State, state) { + t.Errorf("Wrong state after mutation.\nWant %#v.\nGot %#v.", + state, server.containers[0].State) + } +} + +func TestMutateContainerNotFound(t *testing.T) { + server := DockerServer{failures: make(map[string]string)} + server.buildMuxer() + state := docker.State{Running: false, ExitCode: 1} + err := server.MutateContainer("id123", state) + if err == nil { + t.Error("Unexpected error") + } + if err.Error() != "container not found" { + t.Errorf("wrong error message. Want %q. Got %q.", "container not found", err) + } +} + +func TestBuildImageWithContentTypeTar(t *testing.T) { + server := DockerServer{imgIDs: make(map[string]string)} + imageName := "teste" + recorder := httptest.NewRecorder() + tarFile, err := os.Open("data/dockerfile.tar") + if err != nil { + t.Fatal(err) + } + defer tarFile.Close() + request, _ := http.NewRequest("POST", "/build?t=teste", tarFile) + request.Header.Add("Content-Type", "application/tar") + server.buildImage(recorder, request) + if recorder.Body.String() == "miss Dockerfile" { + t.Errorf("BuildImage: miss Dockerfile") + return + } + if _, ok := server.imgIDs[imageName]; ok == false { + t.Errorf("BuildImage: image %s not builded", imageName) + } +} + +func TestBuildImageWithRemoteDockerfile(t *testing.T) { + server := DockerServer{imgIDs: make(map[string]string)} + imageName := "teste" + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/build?t=teste&remote=http://localhost/Dockerfile", nil) + server.buildImage(recorder, request) + if _, ok := server.imgIDs[imageName]; ok == false { + t.Errorf("BuildImage: image %s not builded", imageName) + } +} + +func TestPing(t *testing.T) { + server := DockerServer{} + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("GET", "/_ping", nil) + server.pingDocker(recorder, request) + if recorder.Body.String() != "" { + t.Errorf("Ping: Unexpected body: %s", recorder.Body.String()) + } + if recorder.Code != http.StatusOK { + t.Errorf("Ping: Expected code %d, got: %d", http.StatusOK, recorder.Code) + } +} + +func TestDefaultHandler(t *testing.T) { + server, err := NewServer("127.0.0.1:0", nil, nil) + if err != nil { + t.Fatal(err) + } + defer server.listener.Close() + if server.mux != server.DefaultHandler() { + t.Fatalf("DefaultHandler: Expected to return server.mux, got: %#v", server.DefaultHandler()) + } +} + +func TestCreateExecContainer(t *testing.T) { + server := DockerServer{} + addContainers(&server, 2) + server.buildMuxer() + recorder := httptest.NewRecorder() + body := `{"Cmd": ["bash", "-c", "ls"]}` + path := fmt.Sprintf("/containers/%s/exec", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, strings.NewReader(body)) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Fatalf("CreateExec: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + serverExec := server.execs[0] + var got docker.Exec + err := json.NewDecoder(recorder.Body).Decode(&got) + if err != nil { + t.Fatal(err) + } + if got.ID != serverExec.ID { + t.Errorf("CreateExec: wrong value. Want %#v. Got %#v.", serverExec.ID, got.ID) + } + expected := docker.ExecInspect{ + ID: got.ID, + ProcessConfig: docker.ExecProcessConfig{ + EntryPoint: "bash", + Arguments: []string{"-c", "ls"}, + }, + Container: *server.containers[0], + } + if !reflect.DeepEqual(*serverExec, expected) { + t.Errorf("InspectContainer: wrong value. Want:\n%#v\nGot:\n%#v\n", expected, *serverExec) + } +} + +func TestInspectExecContainer(t *testing.T) { + server := DockerServer{} + addContainers(&server, 1) + server.buildMuxer() + recorder := httptest.NewRecorder() + body := `{"Cmd": ["bash", "-c", "ls"]}` + path := fmt.Sprintf("/containers/%s/exec", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, strings.NewReader(body)) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Fatalf("CreateExec: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + var got docker.Exec + err := json.NewDecoder(recorder.Body).Decode(&got) + if err != nil { + t.Fatal(err) + } + path = fmt.Sprintf("/exec/%s/json", got.ID) + request, _ = http.NewRequest("GET", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Fatalf("CreateExec: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + var got2 docker.ExecInspect + err = json.NewDecoder(recorder.Body).Decode(&got2) + if err != nil { + t.Fatal(err) + } + expected := docker.ExecInspect{ + ID: got.ID, + ProcessConfig: docker.ExecProcessConfig{ + EntryPoint: "bash", + Arguments: []string{"-c", "ls"}, + }, + Container: *server.containers[0], + } + got2.Container.State.StartedAt = expected.Container.State.StartedAt + got2.Container.State.FinishedAt = expected.Container.State.FinishedAt + got2.Container.Config = expected.Container.Config + got2.Container.Created = expected.Container.Created + got2.Container.NetworkSettings = expected.Container.NetworkSettings + if !reflect.DeepEqual(got2, expected) { + t.Errorf("InspectContainer: wrong value. Want:\n%#v\nGot:\n%#v\n", expected, got2) + } +} + +func TestStartExecContainer(t *testing.T) { + server, _ := NewServer("127.0.0.1:0", nil, nil) + addContainers(server, 1) + server.buildMuxer() + recorder := httptest.NewRecorder() + body := `{"Cmd": ["bash", "-c", "ls"]}` + path := fmt.Sprintf("/containers/%s/exec", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, strings.NewReader(body)) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Fatalf("CreateExec: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + var exec docker.Exec + err := json.NewDecoder(recorder.Body).Decode(&exec) + if err != nil { + t.Fatal(err) + } + unleash := make(chan bool) + server.PrepareExec(exec.ID, func() { + <-unleash + }) + codes := make(chan int, 1) + sent := make(chan bool) + go func() { + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/exec/%s/start", exec.ID) + body := `{"Tty":true}` + request, _ := http.NewRequest("POST", path, strings.NewReader(body)) + close(sent) + server.ServeHTTP(recorder, request) + codes <- recorder.Code + }() + <-sent + execInfo, err := waitExec(server.URL(), exec.ID, true, 5) + if err != nil { + t.Fatal(err) + } + if !execInfo.Running { + t.Error("StartExec: expected exec to be running, but it's not running") + } + close(unleash) + if code := <-codes; code != http.StatusOK { + t.Errorf("StartExec: wrong status. Want %d. Got %d.", http.StatusOK, code) + } + execInfo, err = waitExec(server.URL(), exec.ID, false, 5) + if err != nil { + t.Fatal(err) + } + if execInfo.Running { + t.Error("StartExec: expected exec to be not running after start returns, but it's running") + } +} + +func TestStartExecContainerWildcardCallback(t *testing.T) { + server, _ := NewServer("127.0.0.1:0", nil, nil) + addContainers(server, 1) + server.buildMuxer() + recorder := httptest.NewRecorder() + body := `{"Cmd": ["bash", "-c", "ls"]}` + path := fmt.Sprintf("/containers/%s/exec", server.containers[0].ID) + request, _ := http.NewRequest("POST", path, strings.NewReader(body)) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Fatalf("CreateExec: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + unleash := make(chan bool) + server.PrepareExec("*", func() { + <-unleash + }) + var exec docker.Exec + err := json.NewDecoder(recorder.Body).Decode(&exec) + if err != nil { + t.Fatal(err) + } + codes := make(chan int, 1) + sent := make(chan bool) + go func() { + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/exec/%s/start", exec.ID) + body := `{"Tty":true}` + request, _ := http.NewRequest("POST", path, strings.NewReader(body)) + close(sent) + server.ServeHTTP(recorder, request) + codes <- recorder.Code + }() + <-sent + execInfo, err := waitExec(server.URL(), exec.ID, true, 5) + if err != nil { + t.Fatal(err) + } + if !execInfo.Running { + t.Error("StartExec: expected exec to be running, but it's not running") + } + close(unleash) + if code := <-codes; code != http.StatusOK { + t.Errorf("StartExec: wrong status. Want %d. Got %d.", http.StatusOK, code) + } + execInfo, err = waitExec(server.URL(), exec.ID, false, 5) + if err != nil { + t.Fatal(err) + } + if execInfo.Running { + t.Error("StartExec: expected exec to be not running after start returns, but it's running") + } +} + +func TestStartExecContainerNotFound(t *testing.T) { + server, _ := NewServer("127.0.0.1:0", nil, nil) + addContainers(server, 1) + server.buildMuxer() + recorder := httptest.NewRecorder() + body := `{"Tty":true}` + request, _ := http.NewRequest("POST", "/exec/something-wat/start", strings.NewReader(body)) + server.ServeHTTP(recorder, request) +} + +func waitExec(url, execID string, running bool, maxTry int) (*docker.ExecInspect, error) { + client, err := docker.NewClient(url) + if err != nil { + return nil, err + } + exec, err := client.InspectExec(execID) + for i := 0; i < maxTry && exec.Running != running && err == nil; i++ { + time.Sleep(100e6) + exec, err = client.InspectExec(exec.ID) + } + return exec, err +} + +func TestStatsContainer(t *testing.T) { + server, err := NewServer("127.0.0.1:0", nil, nil) + if err != nil { + t.Fatal(err) + } + defer server.Stop() + addContainers(server, 2) + server.buildMuxer() + expected := docker.Stats{} + expected.CPUStats.CPUUsage.TotalUsage = 20 + server.PrepareStats(server.containers[0].ID, func(id string) docker.Stats { + return expected + }) + recorder := httptest.NewRecorder() + path := fmt.Sprintf("/containers/%s/stats?stream=false", server.containers[0].ID) + request, _ := http.NewRequest("GET", path, nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("StatsContainer: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + body := recorder.Body.Bytes() + var got docker.Stats + err = json.Unmarshal(body, &got) + if err != nil { + t.Fatal(err) + } + got.Read = time.Time{} + if !reflect.DeepEqual(got, expected) { + t.Errorf("StatsContainer: wrong value. Want %#v. Got %#v.", expected, got) + } +} + +type safeWriter struct { + sync.Mutex + *httptest.ResponseRecorder +} + +func (w *safeWriter) Write(buf []byte) (int, error) { + w.Lock() + defer w.Unlock() + return w.ResponseRecorder.Write(buf) +} + +func TestStatsContainerStream(t *testing.T) { + server, err := NewServer("127.0.0.1:0", nil, nil) + if err != nil { + t.Fatal(err) + } + defer server.Stop() + addContainers(server, 2) + server.buildMuxer() + expected := docker.Stats{} + expected.CPUStats.CPUUsage.TotalUsage = 20 + server.PrepareStats(server.containers[0].ID, func(id string) docker.Stats { + time.Sleep(50 * time.Millisecond) + return expected + }) + recorder := &safeWriter{ + ResponseRecorder: httptest.NewRecorder(), + } + path := fmt.Sprintf("/containers/%s/stats?stream=true", server.containers[0].ID) + request, _ := http.NewRequest("GET", path, nil) + go func() { + server.ServeHTTP(recorder, request) + }() + time.Sleep(200 * time.Millisecond) + recorder.Lock() + defer recorder.Unlock() + body := recorder.Body.Bytes() + parts := bytes.Split(body, []byte("\n")) + if len(parts) < 2 { + t.Errorf("StatsContainer: wrong number of parts. Want at least 2. Got %#v.", len(parts)) + } + var got docker.Stats + err = json.Unmarshal(parts[0], &got) + if err != nil { + t.Fatal(err) + } + got.Read = time.Time{} + if !reflect.DeepEqual(got, expected) { + t.Errorf("StatsContainer: wrong value. Want %#v. Got %#v.", expected, got) + } +} + +func addNetworks(server *DockerServer, n int) { + server.netMut.Lock() + defer server.netMut.Unlock() + for i := 0; i < n; i++ { + netid := fmt.Sprintf("%x", rand.Int()%10000) + network := docker.Network{ + Name: netid, + ID: fmt.Sprintf("%x", rand.Int()%10000), + Type: "bridge", + Endpoints: []*docker.Endpoint{ + &docker.Endpoint{ + Name: "blah", + ID: fmt.Sprintf("%x", rand.Int()%10000), + Network: netid, + }, + }, + } + server.networks = append(server.networks, &network) + } +} + +func TestListNetworks(t *testing.T) { + server := DockerServer{} + addNetworks(&server, 2) + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("GET", "/networks", nil) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusOK { + t.Errorf("ListNetworks: wrong status. Want %d. Got %d.", http.StatusOK, recorder.Code) + } + expected := make([]docker.Network, 2) + for i, network := range server.networks { + expected[i] = docker.Network{ + ID: network.ID, + Name: network.Name, + Type: network.Type, + Endpoints: network.Endpoints, + } + } + var got []docker.Network + err := json.NewDecoder(recorder.Body).Decode(&got) + if err != nil { + t.Fatal(err) + } + if !reflect.DeepEqual(got, expected) { + t.Errorf("ListNetworks. Want %#v. Got %#v.", expected, got) + } +} + +type createNetworkResponse struct { + ID string `json:"ID"` +} + +func TestCreateNetwork(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + netid := fmt.Sprintf("%x", rand.Int()%10000) + netname := fmt.Sprintf("%x", rand.Int()%10000) + body := fmt.Sprintf(`{"ID": "%s", "Name": "%s", "Type": "bridge" }`, netid, netname) + request, _ := http.NewRequest("POST", "/networks", strings.NewReader(body)) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusCreated { + t.Errorf("CreateNetwork: wrong status. Want %d. Got %d.", http.StatusCreated, recorder.Code) + } + + var returned createNetworkResponse + err := json.NewDecoder(recorder.Body).Decode(&returned) + if err != nil { + t.Fatal(err) + } + stored := server.networks[0] + if returned.ID != stored.ID { + t.Errorf("CreateNetwork: ID mismatch. Stored: %q. Returned: %q.", stored.ID, returned) + } +} + +func TestCreateNetworkInvalidBody(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + recorder := httptest.NewRecorder() + request, _ := http.NewRequest("POST", "/networks", strings.NewReader("whaaaaaat---")) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusBadRequest { + t.Errorf("CreateNetwork: wrong status. Want %d. Got %d.", http.StatusBadRequest, recorder.Code) + } +} + +func TestCreateNetworkDuplicateName(t *testing.T) { + server := DockerServer{} + server.buildMuxer() + addNetworks(&server, 1) + server.networks[0].Name = "mynetwork" + recorder := httptest.NewRecorder() + body := fmt.Sprintf(`{"ID": "%s", "Name": "mynetwork", "Type": "bridge" }`, fmt.Sprintf("%x", rand.Int()%10000)) + request, _ := http.NewRequest("POST", "/networks", strings.NewReader(body)) + server.ServeHTTP(recorder, request) + if recorder.Code != http.StatusForbidden { + t.Errorf("CreateNetwork: wrong status. Want %d. Got %d.", http.StatusForbidden, recorder.Code) + } +} diff --git a/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tls.go b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tls.go new file mode 100644 index 0000000000000..11d571761a8c1 --- /dev/null +++ b/Godeps/_workspace/src/github.com/fsouza/go-dockerclient/tls.go @@ -0,0 +1,100 @@ +// Copyright 2014 go-dockerclient authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. +// +// The content is borrowed from Docker's own source code to provide a simple +// tls based dialer + +package docker + +import ( + "crypto/tls" + "errors" + "net" + "strings" + "time" +) + +type tlsClientCon struct { + *tls.Conn + rawConn net.Conn +} + +func (c *tlsClientCon) CloseWrite() error { + // Go standard tls.Conn doesn't provide the CloseWrite() method so we do it + // on its underlying connection. + if cwc, ok := c.rawConn.(interface { + CloseWrite() error + }); ok { + return cwc.CloseWrite() + } + return nil +} + +func tlsDialWithDialer(dialer *net.Dialer, network, addr string, config *tls.Config) (net.Conn, error) { + // We want the Timeout and Deadline values from dialer to cover the + // whole process: TCP connection and TLS handshake. This means that we + // also need to start our own timers now. + timeout := dialer.Timeout + + if !dialer.Deadline.IsZero() { + deadlineTimeout := dialer.Deadline.Sub(time.Now()) + if timeout == 0 || deadlineTimeout < timeout { + timeout = deadlineTimeout + } + } + + var errChannel chan error + + if timeout != 0 { + errChannel = make(chan error, 2) + time.AfterFunc(timeout, func() { + errChannel <- errors.New("") + }) + } + + rawConn, err := dialer.Dial(network, addr) + if err != nil { + return nil, err + } + + colonPos := strings.LastIndex(addr, ":") + if colonPos == -1 { + colonPos = len(addr) + } + hostname := addr[:colonPos] + + // If no ServerName is set, infer the ServerName + // from the hostname we're connecting to. + if config.ServerName == "" { + // Make a copy to avoid polluting argument or default. + c := *config + c.ServerName = hostname + config = &c + } + + conn := tls.Client(rawConn, config) + + if timeout == 0 { + err = conn.Handshake() + } else { + go func() { + errChannel <- conn.Handshake() + }() + + err = <-errChannel + } + + if err != nil { + rawConn.Close() + return nil, err + } + + // This is Docker difference with standard's crypto/tls package: returned a + // wrapper which holds both the TLS and raw connections. + return &tlsClientCon{conn, rawConn}, nil +} + +func tlsDial(network, addr string, config *tls.Config) (net.Conn, error) { + return tlsDialWithDialer(new(net.Dialer), network, addr, config) +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/.gitignore b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/.gitignore new file mode 100644 index 0000000000000..ba8e0cb3aa340 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/.gitignore @@ -0,0 +1,8 @@ +.DS_Store +.DS_Store? +._* +.Spotlight-V100 +.Trashes +Icon? +ehthumbs.db +Thumbs.db diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/.travis.yml b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/.travis.yml new file mode 100644 index 0000000000000..2f4e3c2f0680c --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/.travis.yml @@ -0,0 +1,10 @@ +sudo: false +language: go +go: + - 1.2 + - 1.3 + - 1.4 + - tip + +before_script: + - mysql -e 'create database gotest;' diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/AUTHORS b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/AUTHORS new file mode 100644 index 0000000000000..6fc4c6f7b254a --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/AUTHORS @@ -0,0 +1,44 @@ +# This is the official list of Go-MySQL-Driver authors for copyright purposes. + +# If you are submitting a patch, please add your name or the name of the +# organization which holds the copyright to this list in alphabetical order. + +# Names should be added to this file as +# Name +# The email address is not required for organizations. +# Please keep the list sorted. + + +# Individual Persons + +Aaron Hopkins +Arne Hormann +Carlos Nieto +Chris Moos +DisposaBoy +Frederick Mayle +Gustavo Kristic +Hanno Braun +Henri Yandell +INADA Naoki +James Harr +Jian Zhen +Joshua Prunier +Julien Schmidt +Kamil Dziedzic +Leonardo YongUk Kim +Lucas Liu +Luke Scott +Michael Woolnough +Nicola Peduzzi +Runrioter Wung +Soroush Pour +Stan Putrya +Xiaobing Jiang +Xiuming Chen + +# Organizations + +Barracuda Networks, Inc. +Google Inc. +Stripe Inc. diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/CHANGELOG.md b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/CHANGELOG.md new file mode 100644 index 0000000000000..161ad0fccbe50 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/CHANGELOG.md @@ -0,0 +1,92 @@ +## HEAD + +Changes: + + - Go 1.1 is no longer supported + - Use decimals field from MySQL to format time types (#249) + - Buffer optimizations (#269) + - TLS ServerName defaults to the host (#283) + +Bugfixes: + + - Enable microsecond resolution on TIME, DATETIME and TIMESTAMP (#249) + - Fixed handling of queries without columns and rows (#255) + - Fixed a panic when SetKeepAlive() failed (#298) + +New Features: + - Support for returning table alias on Columns() (#289) + - Placeholder interpolation, can be actived with the DSN parameter `interpolateParams=true` (#309, #318) + + +## Version 1.2 (2014-06-03) + +Changes: + + - We switched back to a "rolling release". `go get` installs the current master branch again + - Version v1 of the driver will not be maintained anymore. Go 1.0 is no longer supported by this driver + - Exported errors to allow easy checking from application code + - Enabled TCP Keepalives on TCP connections + - Optimized INFILE handling (better buffer size calculation, lazy init, ...) + - The DSN parser also checks for a missing separating slash + - Faster binary date / datetime to string formatting + - Also exported the MySQLWarning type + - mysqlConn.Close returns the first error encountered instead of ignoring all errors + - writePacket() automatically writes the packet size to the header + - readPacket() uses an iterative approach instead of the recursive approach to merge splitted packets + +New Features: + + - `RegisterDial` allows the usage of a custom dial function to establish the network connection + - Setting the connection collation is possible with the `collation` DSN parameter. This parameter should be preferred over the `charset` parameter + - Logging of critical errors is configurable with `SetLogger` + - Google CloudSQL support + +Bugfixes: + + - Allow more than 32 parameters in prepared statements + - Various old_password fixes + - Fixed TestConcurrent test to pass Go's race detection + - Fixed appendLengthEncodedInteger for large numbers + - Renamed readLengthEnodedString to readLengthEncodedString and skipLengthEnodedString to skipLengthEncodedString (fixed typo) + + +## Version 1.1 (2013-11-02) + +Changes: + + - Go-MySQL-Driver now requires Go 1.1 + - Connections now use the collation `utf8_general_ci` by default. Adding `&charset=UTF8` to the DSN should not be necessary anymore + - Made closing rows and connections error tolerant. This allows for example deferring rows.Close() without checking for errors + - `[]byte(nil)` is now treated as a NULL value. Before, it was treated like an empty string / `[]byte("")` + - DSN parameter values must now be url.QueryEscape'ed. This allows text values to contain special characters, such as '&'. + - Use the IO buffer also for writing. This results in zero allocations (by the driver) for most queries + - Optimized the buffer for reading + - stmt.Query now caches column metadata + - New Logo + - Changed the copyright header to include all contributors + - Improved the LOAD INFILE documentation + - The driver struct is now exported to make the driver directly accessible + - Refactored the driver tests + - Added more benchmarks and moved all to a separate file + - Other small refactoring + +New Features: + + - Added *old_passwords* support: Required in some cases, but must be enabled by adding `allowOldPasswords=true` to the DSN since it is insecure + - Added a `clientFoundRows` parameter: Return the number of matching rows instead of the number of rows changed on UPDATEs + - Added TLS/SSL support: Use a TLS/SSL encrypted connection to the server. Custom TLS configs can be registered and used + +Bugfixes: + + - Fixed MySQL 4.1 support: MySQL 4.1 sends packets with lengths which differ from the specification + - Convert to DB timezone when inserting `time.Time` + - Splitted packets (more than 16MB) are now merged correctly + - Fixed false positive `io.EOF` errors when the data was fully read + - Avoid panics on reuse of closed connections + - Fixed empty string producing false nil values + - Fixed sign byte for positive TIME fields + + +## Version 1.0 (2013-05-14) + +Initial Release diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/CONTRIBUTING.md b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/CONTRIBUTING.md new file mode 100644 index 0000000000000..f87c19824cc18 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/CONTRIBUTING.md @@ -0,0 +1,40 @@ +# Contributing Guidelines + +## Reporting Issues + +Before creating a new Issue, please check first if a similar Issue [already exists](https://github.com/go-sql-driver/mysql/issues?state=open) or was [recently closed](https://github.com/go-sql-driver/mysql/issues?direction=desc&page=1&sort=updated&state=closed). + +Please provide the following minimum information: +* Your Go-MySQL-Driver version (or git SHA) +* Your Go version (run `go version` in your console) +* A detailed issue description +* Error Log if present +* If possible, a short example + + +## Contributing Code + +By contributing to this project, you share your code under the Mozilla Public License 2, as specified in the LICENSE file. +Don't forget to add yourself to the AUTHORS file. + +### Pull Requests Checklist + +Please check the following points before submitting your pull request: +- [x] Code compiles correctly +- [x] Created tests, if possible +- [x] All tests pass +- [x] Extended the README / documentation, if necessary +- [x] Added yourself to the AUTHORS file + +### Code Review + +Everyone is invited to review and comment on pull requests. +If it looks fine to you, comment with "LGTM" (Looks good to me). + +If changes are required, notice the reviewers with "PTAL" (Please take another look) after committing the fixes. + +Before merging the Pull Request, at least one [team member](https://github.com/go-sql-driver?tab=members) must have commented with "LGTM". + +## Development Ideas + +If you are looking for ideas for code contributions, please check our [Development Ideas](https://github.com/go-sql-driver/mysql/wiki/Development-Ideas) Wiki page. diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/LICENSE b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/LICENSE new file mode 100644 index 0000000000000..14e2f777f6c39 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/LICENSE @@ -0,0 +1,373 @@ +Mozilla Public License Version 2.0 +================================== + +1. Definitions +-------------- + +1.1. "Contributor" + means each individual or legal entity that creates, contributes to + the creation of, or owns Covered Software. + +1.2. "Contributor Version" + means the combination of the Contributions of others (if any) used + by a Contributor and that particular Contributor's Contribution. + +1.3. "Contribution" + means Covered Software of a particular Contributor. + +1.4. "Covered Software" + means Source Code Form to which the initial Contributor has attached + the notice in Exhibit A, the Executable Form of such Source Code + Form, and Modifications of such Source Code Form, in each case + including portions thereof. + +1.5. "Incompatible With Secondary Licenses" + means + + (a) that the initial Contributor has attached the notice described + in Exhibit B to the Covered Software; or + + (b) that the Covered Software was made available under the terms of + version 1.1 or earlier of the License, but not also under the + terms of a Secondary License. + +1.6. "Executable Form" + means any form of the work other than Source Code Form. + +1.7. "Larger Work" + means a work that combines Covered Software with other material, in + a separate file or files, that is not Covered Software. + +1.8. "License" + means this document. + +1.9. "Licensable" + means having the right to grant, to the maximum extent possible, + whether at the time of the initial grant or subsequently, any and + all of the rights conveyed by this License. + +1.10. "Modifications" + means any of the following: + + (a) any file in Source Code Form that results from an addition to, + deletion from, or modification of the contents of Covered + Software; or + + (b) any new file in Source Code Form that contains any Covered + Software. + +1.11. "Patent Claims" of a Contributor + means any patent claim(s), including without limitation, method, + process, and apparatus claims, in any patent Licensable by such + Contributor that would be infringed, but for the grant of the + License, by the making, using, selling, offering for sale, having + made, import, or transfer of either its Contributions or its + Contributor Version. + +1.12. "Secondary License" + means either the GNU General Public License, Version 2.0, the GNU + Lesser General Public License, Version 2.1, the GNU Affero General + Public License, Version 3.0, or any later versions of those + licenses. + +1.13. "Source Code Form" + means the form of the work preferred for making modifications. + +1.14. "You" (or "Your") + means an individual or a legal entity exercising rights under this + License. For legal entities, "You" includes any entity that + controls, is controlled by, or is under common control with You. For + purposes of this definition, "control" means (a) the power, direct + or indirect, to cause the direction or management of such entity, + whether by contract or otherwise, or (b) ownership of more than + fifty percent (50%) of the outstanding shares or beneficial + ownership of such entity. + +2. License Grants and Conditions +-------------------------------- + +2.1. Grants + +Each Contributor hereby grants You a world-wide, royalty-free, +non-exclusive license: + +(a) under intellectual property rights (other than patent or trademark) + Licensable by such Contributor to use, reproduce, make available, + modify, display, perform, distribute, and otherwise exploit its + Contributions, either on an unmodified basis, with Modifications, or + as part of a Larger Work; and + +(b) under Patent Claims of such Contributor to make, use, sell, offer + for sale, have made, import, and otherwise transfer either its + Contributions or its Contributor Version. + +2.2. Effective Date + +The licenses granted in Section 2.1 with respect to any Contribution +become effective for each Contribution on the date the Contributor first +distributes such Contribution. + +2.3. Limitations on Grant Scope + +The licenses granted in this Section 2 are the only rights granted under +this License. No additional rights or licenses will be implied from the +distribution or licensing of Covered Software under this License. +Notwithstanding Section 2.1(b) above, no patent license is granted by a +Contributor: + +(a) for any code that a Contributor has removed from Covered Software; + or + +(b) for infringements caused by: (i) Your and any other third party's + modifications of Covered Software, or (ii) the combination of its + Contributions with other software (except as part of its Contributor + Version); or + +(c) under Patent Claims infringed by Covered Software in the absence of + its Contributions. + +This License does not grant any rights in the trademarks, service marks, +or logos of any Contributor (except as may be necessary to comply with +the notice requirements in Section 3.4). + +2.4. Subsequent Licenses + +No Contributor makes additional grants as a result of Your choice to +distribute the Covered Software under a subsequent version of this +License (see Section 10.2) or under the terms of a Secondary License (if +permitted under the terms of Section 3.3). + +2.5. Representation + +Each Contributor represents that the Contributor believes its +Contributions are its original creation(s) or it has sufficient rights +to grant the rights to its Contributions conveyed by this License. + +2.6. Fair Use + +This License is not intended to limit any rights You have under +applicable copyright doctrines of fair use, fair dealing, or other +equivalents. + +2.7. Conditions + +Sections 3.1, 3.2, 3.3, and 3.4 are conditions of the licenses granted +in Section 2.1. + +3. Responsibilities +------------------- + +3.1. Distribution of Source Form + +All distribution of Covered Software in Source Code Form, including any +Modifications that You create or to which You contribute, must be under +the terms of this License. You must inform recipients that the Source +Code Form of the Covered Software is governed by the terms of this +License, and how they can obtain a copy of this License. You may not +attempt to alter or restrict the recipients' rights in the Source Code +Form. + +3.2. Distribution of Executable Form + +If You distribute Covered Software in Executable Form then: + +(a) such Covered Software must also be made available in Source Code + Form, as described in Section 3.1, and You must inform recipients of + the Executable Form how they can obtain a copy of such Source Code + Form by reasonable means in a timely manner, at a charge no more + than the cost of distribution to the recipient; and + +(b) You may distribute such Executable Form under the terms of this + License, or sublicense it under different terms, provided that the + license for the Executable Form does not attempt to limit or alter + the recipients' rights in the Source Code Form under this License. + +3.3. Distribution of a Larger Work + +You may create and distribute a Larger Work under terms of Your choice, +provided that You also comply with the requirements of this License for +the Covered Software. If the Larger Work is a combination of Covered +Software with a work governed by one or more Secondary Licenses, and the +Covered Software is not Incompatible With Secondary Licenses, this +License permits You to additionally distribute such Covered Software +under the terms of such Secondary License(s), so that the recipient of +the Larger Work may, at their option, further distribute the Covered +Software under the terms of either this License or such Secondary +License(s). + +3.4. Notices + +You may not remove or alter the substance of any license notices +(including copyright notices, patent notices, disclaimers of warranty, +or limitations of liability) contained within the Source Code Form of +the Covered Software, except that You may alter any license notices to +the extent required to remedy known factual inaccuracies. + +3.5. Application of Additional Terms + +You may choose to offer, and to charge a fee for, warranty, support, +indemnity or liability obligations to one or more recipients of Covered +Software. However, You may do so only on Your own behalf, and not on +behalf of any Contributor. You must make it absolutely clear that any +such warranty, support, indemnity, or liability obligation is offered by +You alone, and You hereby agree to indemnify every Contributor for any +liability incurred by such Contributor as a result of warranty, support, +indemnity or liability terms You offer. You may include additional +disclaimers of warranty and limitations of liability specific to any +jurisdiction. + +4. Inability to Comply Due to Statute or Regulation +--------------------------------------------------- + +If it is impossible for You to comply with any of the terms of this +License with respect to some or all of the Covered Software due to +statute, judicial order, or regulation then You must: (a) comply with +the terms of this License to the maximum extent possible; and (b) +describe the limitations and the code they affect. Such description must +be placed in a text file included with all distributions of the Covered +Software under this License. Except to the extent prohibited by statute +or regulation, such description must be sufficiently detailed for a +recipient of ordinary skill to be able to understand it. + +5. Termination +-------------- + +5.1. The rights granted under this License will terminate automatically +if You fail to comply with any of its terms. However, if You become +compliant, then the rights granted under this License from a particular +Contributor are reinstated (a) provisionally, unless and until such +Contributor explicitly and finally terminates Your grants, and (b) on an +ongoing basis, if such Contributor fails to notify You of the +non-compliance by some reasonable means prior to 60 days after You have +come back into compliance. Moreover, Your grants from a particular +Contributor are reinstated on an ongoing basis if such Contributor +notifies You of the non-compliance by some reasonable means, this is the +first time You have received notice of non-compliance with this License +from such Contributor, and You become compliant prior to 30 days after +Your receipt of the notice. + +5.2. If You initiate litigation against any entity by asserting a patent +infringement claim (excluding declaratory judgment actions, +counter-claims, and cross-claims) alleging that a Contributor Version +directly or indirectly infringes any patent, then the rights granted to +You by any and all Contributors for the Covered Software under Section +2.1 of this License shall terminate. + +5.3. In the event of termination under Sections 5.1 or 5.2 above, all +end user license agreements (excluding distributors and resellers) which +have been validly granted by You or Your distributors under this License +prior to termination shall survive termination. + +************************************************************************ +* * +* 6. Disclaimer of Warranty * +* ------------------------- * +* * +* Covered Software is provided under this License on an "as is" * +* basis, without warranty of any kind, either expressed, implied, or * +* statutory, including, without limitation, warranties that the * +* Covered Software is free of defects, merchantable, fit for a * +* particular purpose or non-infringing. The entire risk as to the * +* quality and performance of the Covered Software is with You. * +* Should any Covered Software prove defective in any respect, You * +* (not any Contributor) assume the cost of any necessary servicing, * +* repair, or correction. This disclaimer of warranty constitutes an * +* essential part of this License. No use of any Covered Software is * +* authorized under this License except under this disclaimer. * +* * +************************************************************************ + +************************************************************************ +* * +* 7. Limitation of Liability * +* -------------------------- * +* * +* Under no circumstances and under no legal theory, whether tort * +* (including negligence), contract, or otherwise, shall any * +* Contributor, or anyone who distributes Covered Software as * +* permitted above, be liable to You for any direct, indirect, * +* special, incidental, or consequential damages of any character * +* including, without limitation, damages for lost profits, loss of * +* goodwill, work stoppage, computer failure or malfunction, or any * +* and all other commercial damages or losses, even if such party * +* shall have been informed of the possibility of such damages. This * +* limitation of liability shall not apply to liability for death or * +* personal injury resulting from such party's negligence to the * +* extent applicable law prohibits such limitation. Some * +* jurisdictions do not allow the exclusion or limitation of * +* incidental or consequential damages, so this exclusion and * +* limitation may not apply to You. * +* * +************************************************************************ + +8. Litigation +------------- + +Any litigation relating to this License may be brought only in the +courts of a jurisdiction where the defendant maintains its principal +place of business and such litigation shall be governed by laws of that +jurisdiction, without reference to its conflict-of-law provisions. +Nothing in this Section shall prevent a party's ability to bring +cross-claims or counter-claims. + +9. Miscellaneous +---------------- + +This License represents the complete agreement concerning the subject +matter hereof. If any provision of this License is held to be +unenforceable, such provision shall be reformed only to the extent +necessary to make it enforceable. Any law or regulation which provides +that the language of a contract shall be construed against the drafter +shall not be used to construe this License against a Contributor. + +10. Versions of the License +--------------------------- + +10.1. New Versions + +Mozilla Foundation is the license steward. Except as provided in Section +10.3, no one other than the license steward has the right to modify or +publish new versions of this License. Each version will be given a +distinguishing version number. + +10.2. Effect of New Versions + +You may distribute the Covered Software under the terms of the version +of the License under which You originally received the Covered Software, +or under the terms of any subsequent version published by the license +steward. + +10.3. Modified Versions + +If you create software not governed by this License, and you want to +create a new license for such software, you may create and use a +modified version of this License if you rename the license and remove +any references to the name of the license steward (except to note that +such modified license differs from this License). + +10.4. Distributing Source Code Form that is Incompatible With Secondary +Licenses + +If You choose to distribute Source Code Form that is Incompatible With +Secondary Licenses under the terms of this version of the License, the +notice described in Exhibit B of this License must be attached. + +Exhibit A - Source Code Form License Notice +------------------------------------------- + + This Source Code Form is subject to the terms of the Mozilla Public + License, v. 2.0. If a copy of the MPL was not distributed with this + file, You can obtain one at http://mozilla.org/MPL/2.0/. + +If it is not possible or desirable to put the notice in a particular +file, then You may include the notice in a location (such as a LICENSE +file in a relevant directory) where a recipient would be likely to look +for such a notice. + +You may add additional accurate notices of copyright ownership. + +Exhibit B - "Incompatible With Secondary Licenses" Notice +--------------------------------------------------------- + + This Source Code Form is "Incompatible With Secondary Licenses", as + defined by the Mozilla Public License, v. 2.0. diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/README.md b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/README.md new file mode 100644 index 0000000000000..6a2bb2ca37939 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/README.md @@ -0,0 +1,386 @@ +# Go-MySQL-Driver + +A MySQL-Driver for Go's [database/sql](http://golang.org/pkg/database/sql) package + +![Go-MySQL-Driver logo](https://raw.github.com/wiki/go-sql-driver/mysql/gomysql_m.png "Golang Gopher holding the MySQL Dolphin") + +**Latest stable Release:** [Version 1.2 (June 03, 2014)](https://github.com/go-sql-driver/mysql/releases) + +[![Build Status](https://travis-ci.org/go-sql-driver/mysql.png?branch=master)](https://travis-ci.org/go-sql-driver/mysql) + +--------------------------------------- + * [Features](#features) + * [Requirements](#requirements) + * [Installation](#installation) + * [Usage](#usage) + * [DSN (Data Source Name)](#dsn-data-source-name) + * [Password](#password) + * [Protocol](#protocol) + * [Address](#address) + * [Parameters](#parameters) + * [Examples](#examples) + * [LOAD DATA LOCAL INFILE support](#load-data-local-infile-support) + * [time.Time support](#timetime-support) + * [Unicode support](#unicode-support) + * [Testing / Development](#testing--development) + * [License](#license) + +--------------------------------------- + +## Features + * Lightweight and [fast](https://github.com/go-sql-driver/sql-benchmark "golang MySQL-Driver performance") + * Native Go implementation. No C-bindings, just pure Go + * Connections over TCP/IPv4, TCP/IPv6 or Unix domain sockets + * Automatic handling of broken connections + * Automatic Connection Pooling *(by database/sql package)* + * Supports queries larger than 16MB + * Full [`sql.RawBytes`](http://golang.org/pkg/database/sql/#RawBytes) support. + * Intelligent `LONG DATA` handling in prepared statements + * Secure `LOAD DATA LOCAL INFILE` support with file Whitelisting and `io.Reader` support + * Optional `time.Time` parsing + * Optional placeholder interpolation + +## Requirements + * Go 1.2 or higher + * MySQL (4.1+), MariaDB, Percona Server, Google CloudSQL or Sphinx (2.2.3+) + +--------------------------------------- + +## Installation +Simple install the package to your [$GOPATH](http://code.google.com/p/go-wiki/wiki/GOPATH "GOPATH") with the [go tool](http://golang.org/cmd/go/ "go command") from shell: +```bash +$ go get github.com/go-sql-driver/mysql +``` +Make sure [Git is installed](http://git-scm.com/downloads) on your machine and in your system's `PATH`. + +## Usage +_Go MySQL Driver_ is an implementation of Go's `database/sql/driver` interface. You only need to import the driver and can use the full [`database/sql`](http://golang.org/pkg/database/sql) API then. + +Use `mysql` as `driverName` and a valid [DSN](#dsn-data-source-name) as `dataSourceName`: +```go +import "database/sql" +import _ "github.com/go-sql-driver/mysql" + +db, err := sql.Open("mysql", "user:password@/dbname") +``` + +[Examples are available in our Wiki](https://github.com/go-sql-driver/mysql/wiki/Examples "Go-MySQL-Driver Examples"). + + +### DSN (Data Source Name) + +The Data Source Name has a common format, like e.g. [PEAR DB](http://pear.php.net/manual/en/package.database.db.intro-dsn.php) uses it, but without type-prefix (optional parts marked by squared brackets): +``` +[username[:password]@][protocol[(address)]]/dbname[?param1=value1&...¶mN=valueN] +``` + +A DSN in its fullest form: +``` +username:password@protocol(address)/dbname?param=value +``` + +Except for the databasename, all values are optional. So the minimal DSN is: +``` +/dbname +``` + +If you do not want to preselect a database, leave `dbname` empty: +``` +/ +``` +This has the same effect as an empty DSN string: +``` + +``` + +#### Password +Passwords can consist of any character. Escaping is **not** necessary. + +#### Protocol +See [net.Dial](http://golang.org/pkg/net/#Dial) for more information which networks are available. +In general you should use an Unix domain socket if available and TCP otherwise for best performance. + +#### Address +For TCP and UDP networks, addresses have the form `host:port`. +If `host` is a literal IPv6 address, it must be enclosed in square brackets. +The functions [net.JoinHostPort](http://golang.org/pkg/net/#JoinHostPort) and [net.SplitHostPort](http://golang.org/pkg/net/#SplitHostPort) manipulate addresses in this form. + +For Unix domain sockets the address is the absolute path to the MySQL-Server-socket, e.g. `/var/run/mysqld/mysqld.sock` or `/tmp/mysql.sock`. + +#### Parameters +*Parameters are case-sensitive!* + +Notice that any of `true`, `TRUE`, `True` or `1` is accepted to stand for a true boolean value. Not surprisingly, false can be specified as any of: `false`, `FALSE`, `False` or `0`. + +##### `allowAllFiles` + +``` +Type: bool +Valid Values: true, false +Default: false +``` + +`allowAllFiles=true` disables the file Whitelist for `LOAD DATA LOCAL INFILE` and allows *all* files. +[*Might be insecure!*](http://dev.mysql.com/doc/refman/5.7/en/load-data-local.html) + +##### `allowCleartextPasswords` + +``` +Type: bool +Valid Values: true, false +Default: false +``` + +`allowCleartextPasswords=true` allows using the [cleartext client side plugin](http://dev.mysql.com/doc/en/cleartext-authentication-plugin.html) if required by an account, such as one defined with the [PAM authentication plugin](http://dev.mysql.com/doc/en/pam-authentication-plugin.html). Sending passwords in clear text may be a security problem in some configurations. To avoid problems if there is any possibility that the password would be intercepted, clients should connect to MySQL Server using a method that protects the password. Possibilities include [TLS / SSL](#tls), IPsec, or a private network. + +##### `allowOldPasswords` + +``` +Type: bool +Valid Values: true, false +Default: false +``` +`allowOldPasswords=true` allows the usage of the insecure old password method. This should be avoided, but is necessary in some cases. See also [the old_passwords wiki page](https://github.com/go-sql-driver/mysql/wiki/old_passwords). + +##### `charset` + +``` +Type: string +Valid Values: +Default: none +``` + +Sets the charset used for client-server interaction (`"SET NAMES "`). If multiple charsets are set (separated by a comma), the following charset is used if setting the charset failes. This enables for example support for `utf8mb4` ([introduced in MySQL 5.5.3](http://dev.mysql.com/doc/refman/5.5/en/charset-unicode-utf8mb4.html)) with fallback to `utf8` for older servers (`charset=utf8mb4,utf8`). + +Usage of the `charset` parameter is discouraged because it issues additional queries to the server. +Unless you need the fallback behavior, please use `collation` instead. + +##### `collation` + +``` +Type: string +Valid Values: +Default: utf8_general_ci +``` + +Sets the collation used for client-server interaction on connection. In contrast to `charset`, `collation` does not issue additional queries. If the specified collation is unavailable on the target server, the connection will fail. + +A list of valid charsets for a server is retrievable with `SHOW COLLATION`. + +##### `clientFoundRows` + +``` +Type: bool +Valid Values: true, false +Default: false +``` + +`clientFoundRows=true` causes an UPDATE to return the number of matching rows instead of the number of rows changed. + +##### `columnsWithAlias` + +``` +Type: bool +Valid Values: true, false +Default: false +``` + +When `columnsWithAlias` is true, calls to `sql.Rows.Columns()` will return the table alias and the column name separated by a dot. For example: + +``` +SELECT u.id FROM users as u +``` + +will return `u.id` instead of just `id` if `columnsWithAlias=true`. + +##### `interpolateParams` + +``` +Type: bool +Valid Values: true, false +Default: false +``` + +If `interpolateParams` is true, placeholders (`?`) in calls to `db.Query()` and `db.Exec()` are interpolated into a single query string with given parameters. This reduces the number of roundtrips, since the driver has to prepare a statement, execute it with given parameters and close the statement again with `interpolateParams=false`. + +*This can not be used together with the multibyte encodings BIG5, CP932, GB2312, GBK or SJIS. These are blacklisted as they may [introduce a SQL injection vulnerability](http://stackoverflow.com/a/12118602/3430118)!* + +##### `loc` + +``` +Type: string +Valid Values: +Default: UTC +``` + +Sets the location for time.Time values (when using `parseTime=true`). *"Local"* sets the system's location. See [time.LoadLocation](http://golang.org/pkg/time/#LoadLocation) for details. + +Note that this sets the location for time.Time values but does not change MySQL's [time_zone setting](https://dev.mysql.com/doc/refman/5.5/en/time-zone-support.html). For that see the [time_zone system variable](#system-variables), which can also be set as a DSN parameter. + +Please keep in mind, that param values must be [url.QueryEscape](http://golang.org/pkg/net/url/#QueryEscape)'ed. Alternatively you can manually replace the `/` with `%2F`. For example `US/Pacific` would be `loc=US%2FPacific`. + + +##### `parseTime` + +``` +Type: bool +Valid Values: true, false +Default: false +``` + +`parseTime=true` changes the output type of `DATE` and `DATETIME` values to `time.Time` instead of `[]byte` / `string` + + +##### `strict` + +``` +Type: bool +Valid Values: true, false +Default: false +``` + +`strict=true` enables the strict mode in which MySQL warnings are treated as errors. + +By default MySQL also treats notes as warnings. Use [`sql_notes=false`](http://dev.mysql.com/doc/refman/5.7/en/server-system-variables.html#sysvar_sql_notes) to ignore notes. See the [examples](#examples) for an DSN example. + + +##### `timeout` + +``` +Type: decimal number +Default: OS default +``` + +*Driver* side connection timeout. The value must be a string of decimal numbers, each with optional fraction and a unit suffix ( *"ms"*, *"s"*, *"m"*, *"h"* ), such as *"30s"*, *"0.5m"* or *"1m30s"*. To set a server side timeout, use the parameter [`wait_timeout`](http://dev.mysql.com/doc/refman/5.6/en/server-system-variables.html#sysvar_wait_timeout). + + +##### `tls` + +``` +Type: bool / string +Valid Values: true, false, skip-verify, +Default: false +``` + +`tls=true` enables TLS / SSL encrypted connection to the server. Use `skip-verify` if you want to use a self-signed or invalid certificate (server side). Use a custom value registered with [`mysql.RegisterTLSConfig`](http://godoc.org/github.com/go-sql-driver/mysql#RegisterTLSConfig). + + +##### System Variables + +All other parameters are interpreted as system variables: + * `autocommit`: `"SET autocommit="` + * [`time_zone`](https://dev.mysql.com/doc/refman/5.5/en/time-zone-support.html): `"SET time_zone="` + * [`tx_isolation`](https://dev.mysql.com/doc/refman/5.5/en/server-system-variables.html#sysvar_tx_isolation): `"SET tx_isolation="` + * `param`: `"SET ="` + +*The values must be [url.QueryEscape](http://golang.org/pkg/net/url/#QueryEscape)'ed!* + +#### Examples +``` +user@unix(/path/to/socket)/dbname +``` + +``` +root:pw@unix(/tmp/mysql.sock)/myDatabase?loc=Local +``` + +``` +user:password@tcp(localhost:5555)/dbname?tls=skip-verify&autocommit=true +``` + +Use the [strict mode](#strict) but ignore notes: +``` +user:password@/dbname?strict=true&sql_notes=false +``` + +TCP via IPv6: +``` +user:password@tcp([de:ad:be:ef::ca:fe]:80)/dbname?timeout=90s&collation=utf8mb4_unicode_ci +``` + +TCP on a remote host, e.g. Amazon RDS: +``` +id:password@tcp(your-amazonaws-uri.com:3306)/dbname +``` + +Google Cloud SQL on App Engine: +``` +user@cloudsql(project-id:instance-name)/dbname +``` + +TCP using default port (3306) on localhost: +``` +user:password@tcp/dbname?charset=utf8mb4,utf8&sys_var=esc%40ped +``` + +Use the default protocol (tcp) and host (localhost:3306): +``` +user:password@/dbname +``` + +No Database preselected: +``` +user:password@/ +``` + +### `LOAD DATA LOCAL INFILE` support +For this feature you need direct access to the package. Therefore you must change the import path (no `_`): +```go +import "github.com/go-sql-driver/mysql" +``` + +Files must be whitelisted by registering them with `mysql.RegisterLocalFile(filepath)` (recommended) or the Whitelist check must be deactivated by using the DSN parameter `allowAllFiles=true` ([*Might be insecure!*](http://dev.mysql.com/doc/refman/5.7/en/load-data-local.html)). + +To use a `io.Reader` a handler function must be registered with `mysql.RegisterReaderHandler(name, handler)` which returns a `io.Reader` or `io.ReadCloser`. The Reader is available with the filepath `Reader::` then. + +See the [godoc of Go-MySQL-Driver](http://godoc.org/github.com/go-sql-driver/mysql "golang mysql driver documentation") for details. + + +### `time.Time` support +The default internal output type of MySQL `DATE` and `DATETIME` values is `[]byte` which allows you to scan the value into a `[]byte`, `string` or `sql.RawBytes` variable in your programm. + +However, many want to scan MySQL `DATE` and `DATETIME` values into `time.Time` variables, which is the logical opposite in Go to `DATE` and `DATETIME` in MySQL. You can do that by changing the internal output type from `[]byte` to `time.Time` with the DSN parameter `parseTime=true`. You can set the default [`time.Time` location](http://golang.org/pkg/time/#Location) with the `loc` DSN parameter. + +**Caution:** As of Go 1.1, this makes `time.Time` the only variable type you can scan `DATE` and `DATETIME` values into. This breaks for example [`sql.RawBytes` support](https://github.com/go-sql-driver/mysql/wiki/Examples#rawbytes). + +Alternatively you can use the [`NullTime`](http://godoc.org/github.com/go-sql-driver/mysql#NullTime) type as the scan destination, which works with both `time.Time` and `string` / `[]byte`. + + +### Unicode support +Since version 1.1 Go-MySQL-Driver automatically uses the collation `utf8_general_ci` by default. + +Other collations / charsets can be set using the [`collation`](#collation) DSN parameter. + +Version 1.0 of the driver recommended adding `&charset=utf8` (alias for `SET NAMES utf8`) to the DSN to enable proper UTF-8 support. This is not necessary anymore. The [`collation`](#collation) parameter should be preferred to set another collation / charset than the default. + +See http://dev.mysql.com/doc/refman/5.7/en/charset-unicode.html for more details on MySQL's Unicode support. + + +## Testing / Development +To run the driver tests you may need to adjust the configuration. See the [Testing Wiki-Page](https://github.com/go-sql-driver/mysql/wiki/Testing "Testing") for details. + +Go-MySQL-Driver is not feature-complete yet. Your help is very appreciated. +If you want to contribute, you can work on an [open issue](https://github.com/go-sql-driver/mysql/issues?state=open) or review a [pull request](https://github.com/go-sql-driver/mysql/pulls). + +See the [Contribution Guidelines](https://github.com/go-sql-driver/mysql/blob/master/CONTRIBUTING.md) for details. + +--------------------------------------- + +## License +Go-MySQL-Driver is licensed under the [Mozilla Public License Version 2.0](https://raw.github.com/go-sql-driver/mysql/master/LICENSE) + +Mozilla summarizes the license scope as follows: +> MPL: The copyleft applies to any files containing MPLed code. + + +That means: + * You can **use** the **unchanged** source code both in private and commercially + * When distributing, you **must publish** the source code of any **changed files** licensed under the MPL 2.0 under a) the MPL 2.0 itself or b) a compatible license (e.g. GPL 3.0 or Apache License 2.0) + * You **needn't publish** the source code of your library as long as the files licensed under the MPL 2.0 are **unchanged** + +Please read the [MPL 2.0 FAQ](http://www.mozilla.org/MPL/2.0/FAQ.html) if you have further questions regarding the license. + +You can read the full terms here: [LICENSE](https://raw.github.com/go-sql-driver/mysql/master/LICENSE) + +![Go Gopher and MySQL Dolphin](https://raw.github.com/wiki/go-sql-driver/mysql/go-mysql-driver_m.jpg "Golang Gopher transporting the MySQL Dolphin in a wheelbarrow") + diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/appengine.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/appengine.go new file mode 100644 index 0000000000000..565614eef7f33 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/appengine.go @@ -0,0 +1,19 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2013 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +// +build appengine + +package mysql + +import ( + "appengine/cloudsql" +) + +func init() { + RegisterDial("cloudsql", cloudsql.Dial) +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/benchmark_test.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/benchmark_test.go new file mode 100644 index 0000000000000..fb8a2f5f3f358 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/benchmark_test.go @@ -0,0 +1,246 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2013 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +import ( + "bytes" + "database/sql" + "database/sql/driver" + "math" + "strings" + "sync" + "sync/atomic" + "testing" + "time" +) + +type TB testing.B + +func (tb *TB) check(err error) { + if err != nil { + tb.Fatal(err) + } +} + +func (tb *TB) checkDB(db *sql.DB, err error) *sql.DB { + tb.check(err) + return db +} + +func (tb *TB) checkRows(rows *sql.Rows, err error) *sql.Rows { + tb.check(err) + return rows +} + +func (tb *TB) checkStmt(stmt *sql.Stmt, err error) *sql.Stmt { + tb.check(err) + return stmt +} + +func initDB(b *testing.B, queries ...string) *sql.DB { + tb := (*TB)(b) + db := tb.checkDB(sql.Open("mysql", dsn)) + for _, query := range queries { + if _, err := db.Exec(query); err != nil { + if w, ok := err.(MySQLWarnings); ok { + b.Logf("Warning on %q: %v", query, w) + } else { + b.Fatalf("Error on %q: %v", query, err) + } + } + } + return db +} + +const concurrencyLevel = 10 + +func BenchmarkQuery(b *testing.B) { + tb := (*TB)(b) + b.StopTimer() + b.ReportAllocs() + db := initDB(b, + "DROP TABLE IF EXISTS foo", + "CREATE TABLE foo (id INT PRIMARY KEY, val CHAR(50))", + `INSERT INTO foo VALUES (1, "one")`, + `INSERT INTO foo VALUES (2, "two")`, + ) + db.SetMaxIdleConns(concurrencyLevel) + defer db.Close() + + stmt := tb.checkStmt(db.Prepare("SELECT val FROM foo WHERE id=?")) + defer stmt.Close() + + remain := int64(b.N) + var wg sync.WaitGroup + wg.Add(concurrencyLevel) + defer wg.Wait() + b.StartTimer() + + for i := 0; i < concurrencyLevel; i++ { + go func() { + for { + if atomic.AddInt64(&remain, -1) < 0 { + wg.Done() + return + } + + var got string + tb.check(stmt.QueryRow(1).Scan(&got)) + if got != "one" { + b.Errorf("query = %q; want one", got) + wg.Done() + return + } + } + }() + } +} + +func BenchmarkExec(b *testing.B) { + tb := (*TB)(b) + b.StopTimer() + b.ReportAllocs() + db := tb.checkDB(sql.Open("mysql", dsn)) + db.SetMaxIdleConns(concurrencyLevel) + defer db.Close() + + stmt := tb.checkStmt(db.Prepare("DO 1")) + defer stmt.Close() + + remain := int64(b.N) + var wg sync.WaitGroup + wg.Add(concurrencyLevel) + defer wg.Wait() + b.StartTimer() + + for i := 0; i < concurrencyLevel; i++ { + go func() { + for { + if atomic.AddInt64(&remain, -1) < 0 { + wg.Done() + return + } + + if _, err := stmt.Exec(); err != nil { + b.Fatal(err.Error()) + } + } + }() + } +} + +// data, but no db writes +var roundtripSample []byte + +func initRoundtripBenchmarks() ([]byte, int, int) { + if roundtripSample == nil { + roundtripSample = []byte(strings.Repeat("0123456789abcdef", 1024*1024)) + } + return roundtripSample, 16, len(roundtripSample) +} + +func BenchmarkRoundtripTxt(b *testing.B) { + b.StopTimer() + sample, min, max := initRoundtripBenchmarks() + sampleString := string(sample) + b.ReportAllocs() + tb := (*TB)(b) + db := tb.checkDB(sql.Open("mysql", dsn)) + defer db.Close() + b.StartTimer() + var result string + for i := 0; i < b.N; i++ { + length := min + i + if length > max { + length = max + } + test := sampleString[0:length] + rows := tb.checkRows(db.Query(`SELECT "` + test + `"`)) + if !rows.Next() { + rows.Close() + b.Fatalf("crashed") + } + err := rows.Scan(&result) + if err != nil { + rows.Close() + b.Fatalf("crashed") + } + if result != test { + rows.Close() + b.Errorf("mismatch") + } + rows.Close() + } +} + +func BenchmarkRoundtripBin(b *testing.B) { + b.StopTimer() + sample, min, max := initRoundtripBenchmarks() + b.ReportAllocs() + tb := (*TB)(b) + db := tb.checkDB(sql.Open("mysql", dsn)) + defer db.Close() + stmt := tb.checkStmt(db.Prepare("SELECT ?")) + defer stmt.Close() + b.StartTimer() + var result sql.RawBytes + for i := 0; i < b.N; i++ { + length := min + i + if length > max { + length = max + } + test := sample[0:length] + rows := tb.checkRows(stmt.Query(test)) + if !rows.Next() { + rows.Close() + b.Fatalf("crashed") + } + err := rows.Scan(&result) + if err != nil { + rows.Close() + b.Fatalf("crashed") + } + if !bytes.Equal(result, test) { + rows.Close() + b.Errorf("mismatch") + } + rows.Close() + } +} + +func BenchmarkInterpolation(b *testing.B) { + mc := &mysqlConn{ + cfg: &config{ + interpolateParams: true, + loc: time.UTC, + }, + maxPacketAllowed: maxPacketSize, + maxWriteSize: maxPacketSize - 1, + buf: newBuffer(nil), + } + + args := []driver.Value{ + int64(42424242), + float64(math.Pi), + false, + time.Unix(1423411542, 807015000), + []byte("bytes containing special chars ' \" \a \x00"), + "string containing special chars ' \" \a \x00", + } + q := "SELECT ?, ?, ?, ?, ?, ?" + + b.ReportAllocs() + b.ResetTimer() + for i := 0; i < b.N; i++ { + _, err := mc.interpolateParams(q, args) + if err != nil { + b.Fatal(err) + } + } +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/buffer.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/buffer.go new file mode 100644 index 0000000000000..509ce89e46855 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/buffer.go @@ -0,0 +1,136 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2013 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +import "io" + +const defaultBufSize = 4096 + +// A buffer which is used for both reading and writing. +// This is possible since communication on each connection is synchronous. +// In other words, we can't write and read simultaneously on the same connection. +// The buffer is similar to bufio.Reader / Writer but zero-copy-ish +// Also highly optimized for this particular use case. +type buffer struct { + buf []byte + rd io.Reader + idx int + length int +} + +func newBuffer(rd io.Reader) buffer { + var b [defaultBufSize]byte + return buffer{ + buf: b[:], + rd: rd, + } +} + +// fill reads into the buffer until at least _need_ bytes are in it +func (b *buffer) fill(need int) error { + n := b.length + + // move existing data to the beginning + if n > 0 && b.idx > 0 { + copy(b.buf[0:n], b.buf[b.idx:]) + } + + // grow buffer if necessary + // TODO: let the buffer shrink again at some point + // Maybe keep the org buf slice and swap back? + if need > len(b.buf) { + // Round up to the next multiple of the default size + newBuf := make([]byte, ((need/defaultBufSize)+1)*defaultBufSize) + copy(newBuf, b.buf) + b.buf = newBuf + } + + b.idx = 0 + + for { + nn, err := b.rd.Read(b.buf[n:]) + n += nn + + switch err { + case nil: + if n < need { + continue + } + b.length = n + return nil + + case io.EOF: + if n >= need { + b.length = n + return nil + } + return io.ErrUnexpectedEOF + + default: + return err + } + } +} + +// returns next N bytes from buffer. +// The returned slice is only guaranteed to be valid until the next read +func (b *buffer) readNext(need int) ([]byte, error) { + if b.length < need { + // refill + if err := b.fill(need); err != nil { + return nil, err + } + } + + offset := b.idx + b.idx += need + b.length -= need + return b.buf[offset:b.idx], nil +} + +// returns a buffer with the requested size. +// If possible, a slice from the existing buffer is returned. +// Otherwise a bigger buffer is made. +// Only one buffer (total) can be used at a time. +func (b *buffer) takeBuffer(length int) []byte { + if b.length > 0 { + return nil + } + + // test (cheap) general case first + if length <= defaultBufSize || length <= cap(b.buf) { + return b.buf[:length] + } + + if length < maxPacketSize { + b.buf = make([]byte, length) + return b.buf + } + return make([]byte, length) +} + +// shortcut which can be used if the requested buffer is guaranteed to be +// smaller than defaultBufSize +// Only one buffer (total) can be used at a time. +func (b *buffer) takeSmallBuffer(length int) []byte { + if b.length == 0 { + return b.buf[:length] + } + return nil +} + +// takeCompleteBuffer returns the complete existing buffer. +// This can be used if the necessary buffer size is unknown. +// Only one buffer (total) can be used at a time. +func (b *buffer) takeCompleteBuffer() []byte { + if b.length == 0 { + return b.buf + } + return nil +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/collations.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/collations.go new file mode 100644 index 0000000000000..6c1d613d5b821 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/collations.go @@ -0,0 +1,250 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2014 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +const defaultCollation byte = 33 // utf8_general_ci + +// A list of available collations mapped to the internal ID. +// To update this map use the following MySQL query: +// SELECT COLLATION_NAME, ID FROM information_schema.COLLATIONS +var collations = map[string]byte{ + "big5_chinese_ci": 1, + "latin2_czech_cs": 2, + "dec8_swedish_ci": 3, + "cp850_general_ci": 4, + "latin1_german1_ci": 5, + "hp8_english_ci": 6, + "koi8r_general_ci": 7, + "latin1_swedish_ci": 8, + "latin2_general_ci": 9, + "swe7_swedish_ci": 10, + "ascii_general_ci": 11, + "ujis_japanese_ci": 12, + "sjis_japanese_ci": 13, + "cp1251_bulgarian_ci": 14, + "latin1_danish_ci": 15, + "hebrew_general_ci": 16, + "tis620_thai_ci": 18, + "euckr_korean_ci": 19, + "latin7_estonian_cs": 20, + "latin2_hungarian_ci": 21, + "koi8u_general_ci": 22, + "cp1251_ukrainian_ci": 23, + "gb2312_chinese_ci": 24, + "greek_general_ci": 25, + "cp1250_general_ci": 26, + "latin2_croatian_ci": 27, + "gbk_chinese_ci": 28, + "cp1257_lithuanian_ci": 29, + "latin5_turkish_ci": 30, + "latin1_german2_ci": 31, + "armscii8_general_ci": 32, + "utf8_general_ci": 33, + "cp1250_czech_cs": 34, + "ucs2_general_ci": 35, + "cp866_general_ci": 36, + "keybcs2_general_ci": 37, + "macce_general_ci": 38, + "macroman_general_ci": 39, + "cp852_general_ci": 40, + "latin7_general_ci": 41, + "latin7_general_cs": 42, + "macce_bin": 43, + "cp1250_croatian_ci": 44, + "utf8mb4_general_ci": 45, + "utf8mb4_bin": 46, + "latin1_bin": 47, + "latin1_general_ci": 48, + "latin1_general_cs": 49, + "cp1251_bin": 50, + "cp1251_general_ci": 51, + "cp1251_general_cs": 52, + "macroman_bin": 53, + "utf16_general_ci": 54, + "utf16_bin": 55, + "utf16le_general_ci": 56, + "cp1256_general_ci": 57, + "cp1257_bin": 58, + "cp1257_general_ci": 59, + "utf32_general_ci": 60, + "utf32_bin": 61, + "utf16le_bin": 62, + "binary": 63, + "armscii8_bin": 64, + "ascii_bin": 65, + "cp1250_bin": 66, + "cp1256_bin": 67, + "cp866_bin": 68, + "dec8_bin": 69, + "greek_bin": 70, + "hebrew_bin": 71, + "hp8_bin": 72, + "keybcs2_bin": 73, + "koi8r_bin": 74, + "koi8u_bin": 75, + "latin2_bin": 77, + "latin5_bin": 78, + "latin7_bin": 79, + "cp850_bin": 80, + "cp852_bin": 81, + "swe7_bin": 82, + "utf8_bin": 83, + "big5_bin": 84, + "euckr_bin": 85, + "gb2312_bin": 86, + "gbk_bin": 87, + "sjis_bin": 88, + "tis620_bin": 89, + "ucs2_bin": 90, + "ujis_bin": 91, + "geostd8_general_ci": 92, + "geostd8_bin": 93, + "latin1_spanish_ci": 94, + "cp932_japanese_ci": 95, + "cp932_bin": 96, + "eucjpms_japanese_ci": 97, + "eucjpms_bin": 98, + "cp1250_polish_ci": 99, + "utf16_unicode_ci": 101, + "utf16_icelandic_ci": 102, + "utf16_latvian_ci": 103, + "utf16_romanian_ci": 104, + "utf16_slovenian_ci": 105, + "utf16_polish_ci": 106, + "utf16_estonian_ci": 107, + "utf16_spanish_ci": 108, + "utf16_swedish_ci": 109, + "utf16_turkish_ci": 110, + "utf16_czech_ci": 111, + "utf16_danish_ci": 112, + "utf16_lithuanian_ci": 113, + "utf16_slovak_ci": 114, + "utf16_spanish2_ci": 115, + "utf16_roman_ci": 116, + "utf16_persian_ci": 117, + "utf16_esperanto_ci": 118, + "utf16_hungarian_ci": 119, + "utf16_sinhala_ci": 120, + "utf16_german2_ci": 121, + "utf16_croatian_ci": 122, + "utf16_unicode_520_ci": 123, + "utf16_vietnamese_ci": 124, + "ucs2_unicode_ci": 128, + "ucs2_icelandic_ci": 129, + "ucs2_latvian_ci": 130, + "ucs2_romanian_ci": 131, + "ucs2_slovenian_ci": 132, + "ucs2_polish_ci": 133, + "ucs2_estonian_ci": 134, + "ucs2_spanish_ci": 135, + "ucs2_swedish_ci": 136, + "ucs2_turkish_ci": 137, + "ucs2_czech_ci": 138, + "ucs2_danish_ci": 139, + "ucs2_lithuanian_ci": 140, + "ucs2_slovak_ci": 141, + "ucs2_spanish2_ci": 142, + "ucs2_roman_ci": 143, + "ucs2_persian_ci": 144, + "ucs2_esperanto_ci": 145, + "ucs2_hungarian_ci": 146, + "ucs2_sinhala_ci": 147, + "ucs2_german2_ci": 148, + "ucs2_croatian_ci": 149, + "ucs2_unicode_520_ci": 150, + "ucs2_vietnamese_ci": 151, + "ucs2_general_mysql500_ci": 159, + "utf32_unicode_ci": 160, + "utf32_icelandic_ci": 161, + "utf32_latvian_ci": 162, + "utf32_romanian_ci": 163, + "utf32_slovenian_ci": 164, + "utf32_polish_ci": 165, + "utf32_estonian_ci": 166, + "utf32_spanish_ci": 167, + "utf32_swedish_ci": 168, + "utf32_turkish_ci": 169, + "utf32_czech_ci": 170, + "utf32_danish_ci": 171, + "utf32_lithuanian_ci": 172, + "utf32_slovak_ci": 173, + "utf32_spanish2_ci": 174, + "utf32_roman_ci": 175, + "utf32_persian_ci": 176, + "utf32_esperanto_ci": 177, + "utf32_hungarian_ci": 178, + "utf32_sinhala_ci": 179, + "utf32_german2_ci": 180, + "utf32_croatian_ci": 181, + "utf32_unicode_520_ci": 182, + "utf32_vietnamese_ci": 183, + "utf8_unicode_ci": 192, + "utf8_icelandic_ci": 193, + "utf8_latvian_ci": 194, + "utf8_romanian_ci": 195, + "utf8_slovenian_ci": 196, + "utf8_polish_ci": 197, + "utf8_estonian_ci": 198, + "utf8_spanish_ci": 199, + "utf8_swedish_ci": 200, + "utf8_turkish_ci": 201, + "utf8_czech_ci": 202, + "utf8_danish_ci": 203, + "utf8_lithuanian_ci": 204, + "utf8_slovak_ci": 205, + "utf8_spanish2_ci": 206, + "utf8_roman_ci": 207, + "utf8_persian_ci": 208, + "utf8_esperanto_ci": 209, + "utf8_hungarian_ci": 210, + "utf8_sinhala_ci": 211, + "utf8_german2_ci": 212, + "utf8_croatian_ci": 213, + "utf8_unicode_520_ci": 214, + "utf8_vietnamese_ci": 215, + "utf8_general_mysql500_ci": 223, + "utf8mb4_unicode_ci": 224, + "utf8mb4_icelandic_ci": 225, + "utf8mb4_latvian_ci": 226, + "utf8mb4_romanian_ci": 227, + "utf8mb4_slovenian_ci": 228, + "utf8mb4_polish_ci": 229, + "utf8mb4_estonian_ci": 230, + "utf8mb4_spanish_ci": 231, + "utf8mb4_swedish_ci": 232, + "utf8mb4_turkish_ci": 233, + "utf8mb4_czech_ci": 234, + "utf8mb4_danish_ci": 235, + "utf8mb4_lithuanian_ci": 236, + "utf8mb4_slovak_ci": 237, + "utf8mb4_spanish2_ci": 238, + "utf8mb4_roman_ci": 239, + "utf8mb4_persian_ci": 240, + "utf8mb4_esperanto_ci": 241, + "utf8mb4_hungarian_ci": 242, + "utf8mb4_sinhala_ci": 243, + "utf8mb4_german2_ci": 244, + "utf8mb4_croatian_ci": 245, + "utf8mb4_unicode_520_ci": 246, + "utf8mb4_vietnamese_ci": 247, +} + +// A blacklist of collations which is unsafe to interpolate parameters. +// These multibyte encodings may contains 0x5c (`\`) in their trailing bytes. +var unsafeCollations = map[byte]bool{ + 1: true, // big5_chinese_ci + 13: true, // sjis_japanese_ci + 28: true, // gbk_chinese_ci + 84: true, // big5_bin + 86: true, // gb2312_bin + 87: true, // gbk_bin + 88: true, // sjis_bin + 95: true, // cp932_japanese_ci + 96: true, // cp932_bin +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/connection.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/connection.go new file mode 100644 index 0000000000000..caaae013f5f2f --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/connection.go @@ -0,0 +1,403 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2012 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +import ( + "crypto/tls" + "database/sql/driver" + "errors" + "net" + "strconv" + "strings" + "time" +) + +type mysqlConn struct { + buf buffer + netConn net.Conn + affectedRows uint64 + insertId uint64 + cfg *config + maxPacketAllowed int + maxWriteSize int + flags clientFlag + status statusFlag + sequence uint8 + parseTime bool + strict bool +} + +type config struct { + user string + passwd string + net string + addr string + dbname string + params map[string]string + loc *time.Location + tls *tls.Config + timeout time.Duration + collation uint8 + allowAllFiles bool + allowOldPasswords bool + allowCleartextPasswords bool + clientFoundRows bool + columnsWithAlias bool + interpolateParams bool +} + +// Handles parameters set in DSN after the connection is established +func (mc *mysqlConn) handleParams() (err error) { + for param, val := range mc.cfg.params { + switch param { + // Charset + case "charset": + charsets := strings.Split(val, ",") + for i := range charsets { + // ignore errors here - a charset may not exist + err = mc.exec("SET NAMES " + charsets[i]) + if err == nil { + break + } + } + if err != nil { + return + } + + // time.Time parsing + case "parseTime": + var isBool bool + mc.parseTime, isBool = readBool(val) + if !isBool { + return errors.New("Invalid Bool value: " + val) + } + + // Strict mode + case "strict": + var isBool bool + mc.strict, isBool = readBool(val) + if !isBool { + return errors.New("Invalid Bool value: " + val) + } + + // Compression + case "compress": + err = errors.New("Compression not implemented yet") + return + + // System Vars + default: + err = mc.exec("SET " + param + "=" + val + "") + if err != nil { + return + } + } + } + + return +} + +func (mc *mysqlConn) Begin() (driver.Tx, error) { + if mc.netConn == nil { + errLog.Print(ErrInvalidConn) + return nil, driver.ErrBadConn + } + err := mc.exec("START TRANSACTION") + if err == nil { + return &mysqlTx{mc}, err + } + + return nil, err +} + +func (mc *mysqlConn) Close() (err error) { + // Makes Close idempotent + if mc.netConn != nil { + err = mc.writeCommandPacket(comQuit) + if err == nil { + err = mc.netConn.Close() + } else { + mc.netConn.Close() + } + mc.netConn = nil + } + + mc.cfg = nil + mc.buf.rd = nil + + return +} + +func (mc *mysqlConn) Prepare(query string) (driver.Stmt, error) { + if mc.netConn == nil { + errLog.Print(ErrInvalidConn) + return nil, driver.ErrBadConn + } + // Send command + err := mc.writeCommandPacketStr(comStmtPrepare, query) + if err != nil { + return nil, err + } + + stmt := &mysqlStmt{ + mc: mc, + } + + // Read Result + columnCount, err := stmt.readPrepareResultPacket() + if err == nil { + if stmt.paramCount > 0 { + if err = mc.readUntilEOF(); err != nil { + return nil, err + } + } + + if columnCount > 0 { + err = mc.readUntilEOF() + } + } + + return stmt, err +} + +func (mc *mysqlConn) interpolateParams(query string, args []driver.Value) (string, error) { + buf := mc.buf.takeCompleteBuffer() + if buf == nil { + // can not take the buffer. Something must be wrong with the connection + errLog.Print(ErrBusyBuffer) + return "", driver.ErrBadConn + } + buf = buf[:0] + argPos := 0 + + for i := 0; i < len(query); i++ { + q := strings.IndexByte(query[i:], '?') + if q == -1 { + buf = append(buf, query[i:]...) + break + } + buf = append(buf, query[i:i+q]...) + i += q + + arg := args[argPos] + argPos++ + + if arg == nil { + buf = append(buf, "NULL"...) + continue + } + + switch v := arg.(type) { + case int64: + buf = strconv.AppendInt(buf, v, 10) + case float64: + buf = strconv.AppendFloat(buf, v, 'g', -1, 64) + case bool: + if v { + buf = append(buf, '1') + } else { + buf = append(buf, '0') + } + case time.Time: + if v.IsZero() { + buf = append(buf, "'0000-00-00'"...) + } else { + v := v.In(mc.cfg.loc) + v = v.Add(time.Nanosecond * 500) // To round under microsecond + year := v.Year() + year100 := year / 100 + year1 := year % 100 + month := v.Month() + day := v.Day() + hour := v.Hour() + minute := v.Minute() + second := v.Second() + micro := v.Nanosecond() / 1000 + + buf = append(buf, []byte{ + '\'', + digits10[year100], digits01[year100], + digits10[year1], digits01[year1], + '-', + digits10[month], digits01[month], + '-', + digits10[day], digits01[day], + ' ', + digits10[hour], digits01[hour], + ':', + digits10[minute], digits01[minute], + ':', + digits10[second], digits01[second], + }...) + + if micro != 0 { + micro10000 := micro / 10000 + micro100 := micro / 100 % 100 + micro1 := micro % 100 + buf = append(buf, []byte{ + '.', + digits10[micro10000], digits01[micro10000], + digits10[micro100], digits01[micro100], + digits10[micro1], digits01[micro1], + }...) + } + buf = append(buf, '\'') + } + case []byte: + if v == nil { + buf = append(buf, "NULL"...) + } else { + buf = append(buf, '\'') + if mc.status&statusNoBackslashEscapes == 0 { + buf = escapeBytesBackslash(buf, v) + } else { + buf = escapeBytesQuotes(buf, v) + } + buf = append(buf, '\'') + } + case string: + buf = append(buf, '\'') + if mc.status&statusNoBackslashEscapes == 0 { + buf = escapeStringBackslash(buf, v) + } else { + buf = escapeStringQuotes(buf, v) + } + buf = append(buf, '\'') + default: + return "", driver.ErrSkip + } + + if len(buf)+4 > mc.maxPacketAllowed { + return "", driver.ErrSkip + } + } + if argPos != len(args) { + return "", driver.ErrSkip + } + return string(buf), nil +} + +func (mc *mysqlConn) Exec(query string, args []driver.Value) (driver.Result, error) { + if mc.netConn == nil { + errLog.Print(ErrInvalidConn) + return nil, driver.ErrBadConn + } + if len(args) != 0 { + if !mc.cfg.interpolateParams { + return nil, driver.ErrSkip + } + // try to interpolate the parameters to save extra roundtrips for preparing and closing a statement + prepared, err := mc.interpolateParams(query, args) + if err != nil { + return nil, err + } + query = prepared + args = nil + } + mc.affectedRows = 0 + mc.insertId = 0 + + err := mc.exec(query) + if err == nil { + return &mysqlResult{ + affectedRows: int64(mc.affectedRows), + insertId: int64(mc.insertId), + }, err + } + return nil, err +} + +// Internal function to execute commands +func (mc *mysqlConn) exec(query string) error { + // Send command + err := mc.writeCommandPacketStr(comQuery, query) + if err != nil { + return err + } + + // Read Result + resLen, err := mc.readResultSetHeaderPacket() + if err == nil && resLen > 0 { + if err = mc.readUntilEOF(); err != nil { + return err + } + + err = mc.readUntilEOF() + } + + return err +} + +func (mc *mysqlConn) Query(query string, args []driver.Value) (driver.Rows, error) { + if mc.netConn == nil { + errLog.Print(ErrInvalidConn) + return nil, driver.ErrBadConn + } + if len(args) != 0 { + if !mc.cfg.interpolateParams { + return nil, driver.ErrSkip + } + // try client-side prepare to reduce roundtrip + prepared, err := mc.interpolateParams(query, args) + if err != nil { + return nil, err + } + query = prepared + args = nil + } + // Send command + err := mc.writeCommandPacketStr(comQuery, query) + if err == nil { + // Read Result + var resLen int + resLen, err = mc.readResultSetHeaderPacket() + if err == nil { + rows := new(textRows) + rows.mc = mc + + if resLen == 0 { + // no columns, no more data + return emptyRows{}, nil + } + // Columns + rows.columns, err = mc.readColumns(resLen) + return rows, err + } + } + return nil, err +} + +// Gets the value of the given MySQL System Variable +// The returned byte slice is only valid until the next read +func (mc *mysqlConn) getSystemVar(name string) ([]byte, error) { + // Send command + if err := mc.writeCommandPacketStr(comQuery, "SELECT @@"+name); err != nil { + return nil, err + } + + // Read Result + resLen, err := mc.readResultSetHeaderPacket() + if err == nil { + rows := new(textRows) + rows.mc = mc + + if resLen > 0 { + // Columns + if err := mc.readUntilEOF(); err != nil { + return nil, err + } + } + + dest := make([]driver.Value, resLen) + if err = rows.readRow(dest); err == nil { + return dest[0].([]byte), mc.readUntilEOF() + } + } + return nil, err +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/const.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/const.go new file mode 100644 index 0000000000000..dddc12908f7f0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/const.go @@ -0,0 +1,162 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2012 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +const ( + minProtocolVersion byte = 10 + maxPacketSize = 1<<24 - 1 + timeFormat = "2006-01-02 15:04:05.999999" +) + +// MySQL constants documentation: +// http://dev.mysql.com/doc/internals/en/client-server-protocol.html + +const ( + iOK byte = 0x00 + iLocalInFile byte = 0xfb + iEOF byte = 0xfe + iERR byte = 0xff +) + +// https://dev.mysql.com/doc/internals/en/capability-flags.html#packet-Protocol::CapabilityFlags +type clientFlag uint32 + +const ( + clientLongPassword clientFlag = 1 << iota + clientFoundRows + clientLongFlag + clientConnectWithDB + clientNoSchema + clientCompress + clientODBC + clientLocalFiles + clientIgnoreSpace + clientProtocol41 + clientInteractive + clientSSL + clientIgnoreSIGPIPE + clientTransactions + clientReserved + clientSecureConn + clientMultiStatements + clientMultiResults + clientPSMultiResults + clientPluginAuth + clientConnectAttrs + clientPluginAuthLenEncClientData + clientCanHandleExpiredPasswords + clientSessionTrack + clientDeprecateEOF +) + +const ( + comQuit byte = iota + 1 + comInitDB + comQuery + comFieldList + comCreateDB + comDropDB + comRefresh + comShutdown + comStatistics + comProcessInfo + comConnect + comProcessKill + comDebug + comPing + comTime + comDelayedInsert + comChangeUser + comBinlogDump + comTableDump + comConnectOut + comRegisterSlave + comStmtPrepare + comStmtExecute + comStmtSendLongData + comStmtClose + comStmtReset + comSetOption + comStmtFetch +) + +// https://dev.mysql.com/doc/internals/en/com-query-response.html#packet-Protocol::ColumnType +const ( + fieldTypeDecimal byte = iota + fieldTypeTiny + fieldTypeShort + fieldTypeLong + fieldTypeFloat + fieldTypeDouble + fieldTypeNULL + fieldTypeTimestamp + fieldTypeLongLong + fieldTypeInt24 + fieldTypeDate + fieldTypeTime + fieldTypeDateTime + fieldTypeYear + fieldTypeNewDate + fieldTypeVarChar + fieldTypeBit +) +const ( + fieldTypeNewDecimal byte = iota + 0xf6 + fieldTypeEnum + fieldTypeSet + fieldTypeTinyBLOB + fieldTypeMediumBLOB + fieldTypeLongBLOB + fieldTypeBLOB + fieldTypeVarString + fieldTypeString + fieldTypeGeometry +) + +type fieldFlag uint16 + +const ( + flagNotNULL fieldFlag = 1 << iota + flagPriKey + flagUniqueKey + flagMultipleKey + flagBLOB + flagUnsigned + flagZeroFill + flagBinary + flagEnum + flagAutoIncrement + flagTimestamp + flagSet + flagUnknown1 + flagUnknown2 + flagUnknown3 + flagUnknown4 +) + +// http://dev.mysql.com/doc/internals/en/status-flags.html +type statusFlag uint16 + +const ( + statusInTrans statusFlag = 1 << iota + statusInAutocommit + statusReserved // Not in documentation + statusMoreResultsExists + statusNoGoodIndexUsed + statusNoIndexUsed + statusCursorExists + statusLastRowSent + statusDbDropped + statusNoBackslashEscapes + statusMetadataChanged + statusQueryWasSlow + statusPsOutParams + statusInTransReadonly + statusSessionStateChanged +) diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/driver.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/driver.go new file mode 100644 index 0000000000000..d310624ad1168 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/driver.go @@ -0,0 +1,149 @@ +// Copyright 2012 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// The driver should be used via the database/sql package: +// +// import "database/sql" +// import _ "github.com/go-sql-driver/mysql" +// +// db, err := sql.Open("mysql", "user:password@/dbname") +// +// See https://github.com/go-sql-driver/mysql#usage for details +package mysql + +import ( + "database/sql" + "database/sql/driver" + "net" +) + +// This struct is exported to make the driver directly accessible. +// In general the driver is used via the database/sql package. +type MySQLDriver struct{} + +// DialFunc is a function which can be used to establish the network connection. +// Custom dial functions must be registered with RegisterDial +type DialFunc func(addr string) (net.Conn, error) + +var dials map[string]DialFunc + +// RegisterDial registers a custom dial function. It can then be used by the +// network address mynet(addr), where mynet is the registered new network. +// addr is passed as a parameter to the dial function. +func RegisterDial(net string, dial DialFunc) { + if dials == nil { + dials = make(map[string]DialFunc) + } + dials[net] = dial +} + +// Open new Connection. +// See https://github.com/go-sql-driver/mysql#dsn-data-source-name for how +// the DSN string is formated +func (d MySQLDriver) Open(dsn string) (driver.Conn, error) { + var err error + + // New mysqlConn + mc := &mysqlConn{ + maxPacketAllowed: maxPacketSize, + maxWriteSize: maxPacketSize - 1, + } + mc.cfg, err = parseDSN(dsn) + if err != nil { + return nil, err + } + + // Connect to Server + if dial, ok := dials[mc.cfg.net]; ok { + mc.netConn, err = dial(mc.cfg.addr) + } else { + nd := net.Dialer{Timeout: mc.cfg.timeout} + mc.netConn, err = nd.Dial(mc.cfg.net, mc.cfg.addr) + } + if err != nil { + return nil, err + } + + // Enable TCP Keepalives on TCP connections + if tc, ok := mc.netConn.(*net.TCPConn); ok { + if err := tc.SetKeepAlive(true); err != nil { + // Don't send COM_QUIT before handshake. + mc.netConn.Close() + mc.netConn = nil + return nil, err + } + } + + mc.buf = newBuffer(mc.netConn) + + // Reading Handshake Initialization Packet + cipher, err := mc.readInitPacket() + if err != nil { + mc.Close() + return nil, err + } + + // Send Client Authentication Packet + if err = mc.writeAuthPacket(cipher); err != nil { + mc.Close() + return nil, err + } + + // Read Result Packet + err = mc.readResultOK() + if err != nil { + // Retry with old authentication method, if allowed + if mc.cfg != nil && mc.cfg.allowOldPasswords && err == ErrOldPassword { + if err = mc.writeOldAuthPacket(cipher); err != nil { + mc.Close() + return nil, err + } + if err = mc.readResultOK(); err != nil { + mc.Close() + return nil, err + } + } else if mc.cfg != nil && mc.cfg.allowCleartextPasswords && err == ErrCleartextPassword { + if err = mc.writeClearAuthPacket(); err != nil { + mc.Close() + return nil, err + } + if err = mc.readResultOK(); err != nil { + mc.Close() + return nil, err + } + } else { + mc.Close() + return nil, err + } + + } + + // Get max allowed packet size + maxap, err := mc.getSystemVar("max_allowed_packet") + if err != nil { + mc.Close() + return nil, err + } + mc.maxPacketAllowed = stringToInt(maxap) - 1 + if mc.maxPacketAllowed < maxPacketSize { + mc.maxWriteSize = mc.maxPacketAllowed + } + + // Handle DSN Params + err = mc.handleParams() + if err != nil { + mc.Close() + return nil, err + } + + return mc, nil +} + +func init() { + sql.Register("mysql", &MySQLDriver{}) +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/driver_test.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/driver_test.go new file mode 100644 index 0000000000000..f9da416ec344a --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/driver_test.go @@ -0,0 +1,1681 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2013 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +import ( + "crypto/tls" + "database/sql" + "database/sql/driver" + "fmt" + "io" + "io/ioutil" + "net" + "net/url" + "os" + "strings" + "sync" + "sync/atomic" + "testing" + "time" +) + +var ( + user string + pass string + prot string + addr string + dbname string + dsn string + netAddr string + available bool +) + +var ( + tDate = time.Date(2012, 6, 14, 0, 0, 0, 0, time.UTC) + sDate = "2012-06-14" + tDateTime = time.Date(2011, 11, 20, 21, 27, 37, 0, time.UTC) + sDateTime = "2011-11-20 21:27:37" + tDate0 = time.Time{} + sDate0 = "0000-00-00" + sDateTime0 = "0000-00-00 00:00:00" +) + +// See https://github.com/go-sql-driver/mysql/wiki/Testing +func init() { + // get environment variables + env := func(key, defaultValue string) string { + if value := os.Getenv(key); value != "" { + return value + } + return defaultValue + } + user = env("MYSQL_TEST_USER", "root") + pass = env("MYSQL_TEST_PASS", "") + prot = env("MYSQL_TEST_PROT", "tcp") + addr = env("MYSQL_TEST_ADDR", "localhost:3306") + dbname = env("MYSQL_TEST_DBNAME", "gotest") + netAddr = fmt.Sprintf("%s(%s)", prot, addr) + dsn = fmt.Sprintf("%s:%s@%s/%s?timeout=30s&strict=true", user, pass, netAddr, dbname) + c, err := net.Dial(prot, addr) + if err == nil { + available = true + c.Close() + } +} + +type DBTest struct { + *testing.T + db *sql.DB +} + +func runTests(t *testing.T, dsn string, tests ...func(dbt *DBTest)) { + if !available { + t.Skipf("MySQL-Server not running on %s", netAddr) + } + + db, err := sql.Open("mysql", dsn) + if err != nil { + t.Fatalf("Error connecting: %s", err.Error()) + } + defer db.Close() + + db.Exec("DROP TABLE IF EXISTS test") + + dsn2 := dsn + "&interpolateParams=true" + var db2 *sql.DB + if _, err := parseDSN(dsn2); err != errInvalidDSNUnsafeCollation { + db2, err = sql.Open("mysql", dsn2) + if err != nil { + t.Fatalf("Error connecting: %s", err.Error()) + } + defer db2.Close() + } + + dbt := &DBTest{t, db} + dbt2 := &DBTest{t, db2} + for _, test := range tests { + test(dbt) + dbt.db.Exec("DROP TABLE IF EXISTS test") + if db2 != nil { + test(dbt2) + dbt2.db.Exec("DROP TABLE IF EXISTS test") + } + } +} + +func (dbt *DBTest) fail(method, query string, err error) { + if len(query) > 300 { + query = "[query too large to print]" + } + dbt.Fatalf("Error on %s %s: %s", method, query, err.Error()) +} + +func (dbt *DBTest) mustExec(query string, args ...interface{}) (res sql.Result) { + res, err := dbt.db.Exec(query, args...) + if err != nil { + dbt.fail("Exec", query, err) + } + return res +} + +func (dbt *DBTest) mustQuery(query string, args ...interface{}) (rows *sql.Rows) { + rows, err := dbt.db.Query(query, args...) + if err != nil { + dbt.fail("Query", query, err) + } + return rows +} + +func TestEmptyQuery(t *testing.T) { + runTests(t, dsn, func(dbt *DBTest) { + // just a comment, no query + rows := dbt.mustQuery("--") + // will hang before #255 + if rows.Next() { + dbt.Errorf("Next on rows must be false") + } + }) +} + +func TestCRUD(t *testing.T) { + runTests(t, dsn, func(dbt *DBTest) { + // Create Table + dbt.mustExec("CREATE TABLE test (value BOOL)") + + // Test for unexpected data + var out bool + rows := dbt.mustQuery("SELECT * FROM test") + if rows.Next() { + dbt.Error("unexpected data in empty table") + } + + // Create Data + res := dbt.mustExec("INSERT INTO test VALUES (1)") + count, err := res.RowsAffected() + if err != nil { + dbt.Fatalf("res.RowsAffected() returned error: %s", err.Error()) + } + if count != 1 { + dbt.Fatalf("Expected 1 affected row, got %d", count) + } + + id, err := res.LastInsertId() + if err != nil { + dbt.Fatalf("res.LastInsertId() returned error: %s", err.Error()) + } + if id != 0 { + dbt.Fatalf("Expected InsertID 0, got %d", id) + } + + // Read + rows = dbt.mustQuery("SELECT value FROM test") + if rows.Next() { + rows.Scan(&out) + if true != out { + dbt.Errorf("true != %t", out) + } + + if rows.Next() { + dbt.Error("unexpected data") + } + } else { + dbt.Error("no data") + } + + // Update + res = dbt.mustExec("UPDATE test SET value = ? WHERE value = ?", false, true) + count, err = res.RowsAffected() + if err != nil { + dbt.Fatalf("res.RowsAffected() returned error: %s", err.Error()) + } + if count != 1 { + dbt.Fatalf("Expected 1 affected row, got %d", count) + } + + // Check Update + rows = dbt.mustQuery("SELECT value FROM test") + if rows.Next() { + rows.Scan(&out) + if false != out { + dbt.Errorf("false != %t", out) + } + + if rows.Next() { + dbt.Error("unexpected data") + } + } else { + dbt.Error("no data") + } + + // Delete + res = dbt.mustExec("DELETE FROM test WHERE value = ?", false) + count, err = res.RowsAffected() + if err != nil { + dbt.Fatalf("res.RowsAffected() returned error: %s", err.Error()) + } + if count != 1 { + dbt.Fatalf("Expected 1 affected row, got %d", count) + } + + // Check for unexpected rows + res = dbt.mustExec("DELETE FROM test") + count, err = res.RowsAffected() + if err != nil { + dbt.Fatalf("res.RowsAffected() returned error: %s", err.Error()) + } + if count != 0 { + dbt.Fatalf("Expected 0 affected row, got %d", count) + } + }) +} + +func TestInt(t *testing.T) { + runTests(t, dsn, func(dbt *DBTest) { + types := [5]string{"TINYINT", "SMALLINT", "MEDIUMINT", "INT", "BIGINT"} + in := int64(42) + var out int64 + var rows *sql.Rows + + // SIGNED + for _, v := range types { + dbt.mustExec("CREATE TABLE test (value " + v + ")") + + dbt.mustExec("INSERT INTO test VALUES (?)", in) + + rows = dbt.mustQuery("SELECT value FROM test") + if rows.Next() { + rows.Scan(&out) + if in != out { + dbt.Errorf("%s: %d != %d", v, in, out) + } + } else { + dbt.Errorf("%s: no data", v) + } + + dbt.mustExec("DROP TABLE IF EXISTS test") + } + + // UNSIGNED ZEROFILL + for _, v := range types { + dbt.mustExec("CREATE TABLE test (value " + v + " ZEROFILL)") + + dbt.mustExec("INSERT INTO test VALUES (?)", in) + + rows = dbt.mustQuery("SELECT value FROM test") + if rows.Next() { + rows.Scan(&out) + if in != out { + dbt.Errorf("%s ZEROFILL: %d != %d", v, in, out) + } + } else { + dbt.Errorf("%s ZEROFILL: no data", v) + } + + dbt.mustExec("DROP TABLE IF EXISTS test") + } + }) +} + +func TestFloat(t *testing.T) { + runTests(t, dsn, func(dbt *DBTest) { + types := [2]string{"FLOAT", "DOUBLE"} + in := float32(42.23) + var out float32 + var rows *sql.Rows + for _, v := range types { + dbt.mustExec("CREATE TABLE test (value " + v + ")") + dbt.mustExec("INSERT INTO test VALUES (?)", in) + rows = dbt.mustQuery("SELECT value FROM test") + if rows.Next() { + rows.Scan(&out) + if in != out { + dbt.Errorf("%s: %g != %g", v, in, out) + } + } else { + dbt.Errorf("%s: no data", v) + } + dbt.mustExec("DROP TABLE IF EXISTS test") + } + }) +} + +func TestString(t *testing.T) { + runTests(t, dsn, func(dbt *DBTest) { + types := [6]string{"CHAR(255)", "VARCHAR(255)", "TINYTEXT", "TEXT", "MEDIUMTEXT", "LONGTEXT"} + in := "κόσμε üöäßñóùéàâÿœ'îë Árvíztűrő いろはにほへとちりぬるを イロハニホヘト דג סקרן чащах น่าฟังเอย" + var out string + var rows *sql.Rows + + for _, v := range types { + dbt.mustExec("CREATE TABLE test (value " + v + ") CHARACTER SET utf8") + + dbt.mustExec("INSERT INTO test VALUES (?)", in) + + rows = dbt.mustQuery("SELECT value FROM test") + if rows.Next() { + rows.Scan(&out) + if in != out { + dbt.Errorf("%s: %s != %s", v, in, out) + } + } else { + dbt.Errorf("%s: no data", v) + } + + dbt.mustExec("DROP TABLE IF EXISTS test") + } + + // BLOB + dbt.mustExec("CREATE TABLE test (id int, value BLOB) CHARACTER SET utf8") + + id := 2 + in = "Lorem ipsum dolor sit amet, consetetur sadipscing elitr, " + + "sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, " + + "sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. " + + "Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet. " + + "Lorem ipsum dolor sit amet, consetetur sadipscing elitr, " + + "sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, " + + "sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. " + + "Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet." + dbt.mustExec("INSERT INTO test VALUES (?, ?)", id, in) + + err := dbt.db.QueryRow("SELECT value FROM test WHERE id = ?", id).Scan(&out) + if err != nil { + dbt.Fatalf("Error on BLOB-Query: %s", err.Error()) + } else if out != in { + dbt.Errorf("BLOB: %s != %s", in, out) + } + }) +} + +type timeTests struct { + dbtype string + tlayout string + tests []timeTest +} + +type timeTest struct { + s string // leading "!": do not use t as value in queries + t time.Time +} + +type timeMode byte + +func (t timeMode) String() string { + switch t { + case binaryString: + return "binary:string" + case binaryTime: + return "binary:time.Time" + case textString: + return "text:string" + } + panic("unsupported timeMode") +} + +func (t timeMode) Binary() bool { + switch t { + case binaryString, binaryTime: + return true + } + return false +} + +const ( + binaryString timeMode = iota + binaryTime + textString +) + +func (t timeTest) genQuery(dbtype string, mode timeMode) string { + var inner string + if mode.Binary() { + inner = "?" + } else { + inner = `"%s"` + } + return `SELECT cast(` + inner + ` as ` + dbtype + `)` +} + +func (t timeTest) run(dbt *DBTest, dbtype, tlayout string, mode timeMode) { + var rows *sql.Rows + query := t.genQuery(dbtype, mode) + switch mode { + case binaryString: + rows = dbt.mustQuery(query, t.s) + case binaryTime: + rows = dbt.mustQuery(query, t.t) + case textString: + query = fmt.Sprintf(query, t.s) + rows = dbt.mustQuery(query) + default: + panic("unsupported mode") + } + defer rows.Close() + var err error + if !rows.Next() { + err = rows.Err() + if err == nil { + err = fmt.Errorf("no data") + } + dbt.Errorf("%s [%s]: %s", dbtype, mode, err) + return + } + var dst interface{} + err = rows.Scan(&dst) + if err != nil { + dbt.Errorf("%s [%s]: %s", dbtype, mode, err) + return + } + switch val := dst.(type) { + case []uint8: + str := string(val) + if str == t.s { + return + } + if mode.Binary() && dbtype == "DATETIME" && len(str) == 26 && str[:19] == t.s { + // a fix mainly for TravisCI: + // accept full microsecond resolution in result for DATETIME columns + // where the binary protocol was used + return + } + dbt.Errorf("%s [%s] to string: expected %q, got %q", + dbtype, mode, + t.s, str, + ) + case time.Time: + if val == t.t { + return + } + dbt.Errorf("%s [%s] to string: expected %q, got %q", + dbtype, mode, + t.s, val.Format(tlayout), + ) + default: + fmt.Printf("%#v\n", []interface{}{dbtype, tlayout, mode, t.s, t.t}) + dbt.Errorf("%s [%s]: unhandled type %T (is '%v')", + dbtype, mode, + val, val, + ) + } +} + +func TestDateTime(t *testing.T) { + afterTime := func(t time.Time, d string) time.Time { + dur, err := time.ParseDuration(d) + if err != nil { + panic(err) + } + return t.Add(dur) + } + // NOTE: MySQL rounds DATETIME(x) up - but that's not included in the tests + format := "2006-01-02 15:04:05.999999" + t0 := time.Time{} + tstr0 := "0000-00-00 00:00:00.000000" + testcases := []timeTests{ + {"DATE", format[:10], []timeTest{ + {t: time.Date(2011, 11, 20, 0, 0, 0, 0, time.UTC)}, + {t: t0, s: tstr0[:10]}, + }}, + {"DATETIME", format[:19], []timeTest{ + {t: time.Date(2011, 11, 20, 21, 27, 37, 0, time.UTC)}, + {t: t0, s: tstr0[:19]}, + }}, + {"DATETIME(0)", format[:21], []timeTest{ + {t: time.Date(2011, 11, 20, 21, 27, 37, 0, time.UTC)}, + {t: t0, s: tstr0[:19]}, + }}, + {"DATETIME(1)", format[:21], []timeTest{ + {t: time.Date(2011, 11, 20, 21, 27, 37, 100000000, time.UTC)}, + {t: t0, s: tstr0[:21]}, + }}, + {"DATETIME(6)", format, []timeTest{ + {t: time.Date(2011, 11, 20, 21, 27, 37, 123456000, time.UTC)}, + {t: t0, s: tstr0}, + }}, + {"TIME", format[11:19], []timeTest{ + {t: afterTime(t0, "12345s")}, + {s: "!-12:34:56"}, + {s: "!-838:59:59"}, + {s: "!838:59:59"}, + {t: t0, s: tstr0[11:19]}, + }}, + {"TIME(0)", format[11:19], []timeTest{ + {t: afterTime(t0, "12345s")}, + {s: "!-12:34:56"}, + {s: "!-838:59:59"}, + {s: "!838:59:59"}, + {t: t0, s: tstr0[11:19]}, + }}, + {"TIME(1)", format[11:21], []timeTest{ + {t: afterTime(t0, "12345600ms")}, + {s: "!-12:34:56.7"}, + {s: "!-838:59:58.9"}, + {s: "!838:59:58.9"}, + {t: t0, s: tstr0[11:21]}, + }}, + {"TIME(6)", format[11:], []timeTest{ + {t: afterTime(t0, "1234567890123000ns")}, + {s: "!-12:34:56.789012"}, + {s: "!-838:59:58.999999"}, + {s: "!838:59:58.999999"}, + {t: t0, s: tstr0[11:]}, + }}, + } + dsns := []string{ + dsn + "&parseTime=true", + dsn + "&parseTime=false", + } + for _, testdsn := range dsns { + runTests(t, testdsn, func(dbt *DBTest) { + microsecsSupported := false + zeroDateSupported := false + var rows *sql.Rows + var err error + rows, err = dbt.db.Query(`SELECT cast("00:00:00.1" as TIME(1)) = "00:00:00.1"`) + if err == nil { + rows.Scan(µsecsSupported) + rows.Close() + } + rows, err = dbt.db.Query(`SELECT cast("0000-00-00" as DATE) = "0000-00-00"`) + if err == nil { + rows.Scan(&zeroDateSupported) + rows.Close() + } + for _, setups := range testcases { + if t := setups.dbtype; !microsecsSupported && t[len(t)-1:] == ")" { + // skip fractional second tests if unsupported by server + continue + } + for _, setup := range setups.tests { + allowBinTime := true + if setup.s == "" { + // fill time string whereever Go can reliable produce it + setup.s = setup.t.Format(setups.tlayout) + } else if setup.s[0] == '!' { + // skip tests using setup.t as source in queries + allowBinTime = false + // fix setup.s - remove the "!" + setup.s = setup.s[1:] + } + if !zeroDateSupported && setup.s == tstr0[:len(setup.s)] { + // skip disallowed 0000-00-00 date + continue + } + setup.run(dbt, setups.dbtype, setups.tlayout, textString) + setup.run(dbt, setups.dbtype, setups.tlayout, binaryString) + if allowBinTime { + setup.run(dbt, setups.dbtype, setups.tlayout, binaryTime) + } + } + } + }) + } +} + +func TestTimestampMicros(t *testing.T) { + format := "2006-01-02 15:04:05.999999" + f0 := format[:19] + f1 := format[:21] + f6 := format[:26] + runTests(t, dsn, func(dbt *DBTest) { + // check if microseconds are supported. + // Do not use timestamp(x) for that check - before 5.5.6, x would mean display width + // and not precision. + // Se last paragraph at http://dev.mysql.com/doc/refman/5.6/en/fractional-seconds.html + microsecsSupported := false + if rows, err := dbt.db.Query(`SELECT cast("00:00:00.1" as TIME(1)) = "00:00:00.1"`); err == nil { + rows.Scan(µsecsSupported) + rows.Close() + } + if !microsecsSupported { + // skip test + return + } + _, err := dbt.db.Exec(` + CREATE TABLE test ( + value0 TIMESTAMP NOT NULL DEFAULT '` + f0 + `', + value1 TIMESTAMP(1) NOT NULL DEFAULT '` + f1 + `', + value6 TIMESTAMP(6) NOT NULL DEFAULT '` + f6 + `' + )`, + ) + if err != nil { + dbt.Error(err) + } + defer dbt.mustExec("DROP TABLE IF EXISTS test") + dbt.mustExec("INSERT INTO test SET value0=?, value1=?, value6=?", f0, f1, f6) + var res0, res1, res6 string + rows := dbt.mustQuery("SELECT * FROM test") + if !rows.Next() { + dbt.Errorf("test contained no selectable values") + } + err = rows.Scan(&res0, &res1, &res6) + if err != nil { + dbt.Error(err) + } + if res0 != f0 { + dbt.Errorf("expected %q, got %q", f0, res0) + } + if res1 != f1 { + dbt.Errorf("expected %q, got %q", f1, res1) + } + if res6 != f6 { + dbt.Errorf("expected %q, got %q", f6, res6) + } + }) +} + +func TestNULL(t *testing.T) { + runTests(t, dsn, func(dbt *DBTest) { + nullStmt, err := dbt.db.Prepare("SELECT NULL") + if err != nil { + dbt.Fatal(err) + } + defer nullStmt.Close() + + nonNullStmt, err := dbt.db.Prepare("SELECT 1") + if err != nil { + dbt.Fatal(err) + } + defer nonNullStmt.Close() + + // NullBool + var nb sql.NullBool + // Invalid + if err = nullStmt.QueryRow().Scan(&nb); err != nil { + dbt.Fatal(err) + } + if nb.Valid { + dbt.Error("Valid NullBool which should be invalid") + } + // Valid + if err = nonNullStmt.QueryRow().Scan(&nb); err != nil { + dbt.Fatal(err) + } + if !nb.Valid { + dbt.Error("Invalid NullBool which should be valid") + } else if nb.Bool != true { + dbt.Errorf("Unexpected NullBool value: %t (should be true)", nb.Bool) + } + + // NullFloat64 + var nf sql.NullFloat64 + // Invalid + if err = nullStmt.QueryRow().Scan(&nf); err != nil { + dbt.Fatal(err) + } + if nf.Valid { + dbt.Error("Valid NullFloat64 which should be invalid") + } + // Valid + if err = nonNullStmt.QueryRow().Scan(&nf); err != nil { + dbt.Fatal(err) + } + if !nf.Valid { + dbt.Error("Invalid NullFloat64 which should be valid") + } else if nf.Float64 != float64(1) { + dbt.Errorf("Unexpected NullFloat64 value: %f (should be 1.0)", nf.Float64) + } + + // NullInt64 + var ni sql.NullInt64 + // Invalid + if err = nullStmt.QueryRow().Scan(&ni); err != nil { + dbt.Fatal(err) + } + if ni.Valid { + dbt.Error("Valid NullInt64 which should be invalid") + } + // Valid + if err = nonNullStmt.QueryRow().Scan(&ni); err != nil { + dbt.Fatal(err) + } + if !ni.Valid { + dbt.Error("Invalid NullInt64 which should be valid") + } else if ni.Int64 != int64(1) { + dbt.Errorf("Unexpected NullInt64 value: %d (should be 1)", ni.Int64) + } + + // NullString + var ns sql.NullString + // Invalid + if err = nullStmt.QueryRow().Scan(&ns); err != nil { + dbt.Fatal(err) + } + if ns.Valid { + dbt.Error("Valid NullString which should be invalid") + } + // Valid + if err = nonNullStmt.QueryRow().Scan(&ns); err != nil { + dbt.Fatal(err) + } + if !ns.Valid { + dbt.Error("Invalid NullString which should be valid") + } else if ns.String != `1` { + dbt.Error("Unexpected NullString value:" + ns.String + " (should be `1`)") + } + + // nil-bytes + var b []byte + // Read nil + if err = nullStmt.QueryRow().Scan(&b); err != nil { + dbt.Fatal(err) + } + if b != nil { + dbt.Error("Non-nil []byte wich should be nil") + } + // Read non-nil + if err = nonNullStmt.QueryRow().Scan(&b); err != nil { + dbt.Fatal(err) + } + if b == nil { + dbt.Error("Nil []byte wich should be non-nil") + } + // Insert nil + b = nil + success := false + if err = dbt.db.QueryRow("SELECT ? IS NULL", b).Scan(&success); err != nil { + dbt.Fatal(err) + } + if !success { + dbt.Error("Inserting []byte(nil) as NULL failed") + } + // Check input==output with input==nil + b = nil + if err = dbt.db.QueryRow("SELECT ?", b).Scan(&b); err != nil { + dbt.Fatal(err) + } + if b != nil { + dbt.Error("Non-nil echo from nil input") + } + // Check input==output with input!=nil + b = []byte("") + if err = dbt.db.QueryRow("SELECT ?", b).Scan(&b); err != nil { + dbt.Fatal(err) + } + if b == nil { + dbt.Error("nil echo from non-nil input") + } + + // Insert NULL + dbt.mustExec("CREATE TABLE test (dummmy1 int, value int, dummy2 int)") + + dbt.mustExec("INSERT INTO test VALUES (?, ?, ?)", 1, nil, 2) + + var out interface{} + rows := dbt.mustQuery("SELECT * FROM test") + if rows.Next() { + rows.Scan(&out) + if out != nil { + dbt.Errorf("%v != nil", out) + } + } else { + dbt.Error("no data") + } + }) +} + +func TestUint64(t *testing.T) { + const ( + u0 = uint64(0) + uall = ^u0 + uhigh = uall >> 1 + utop = ^uhigh + s0 = int64(0) + sall = ^s0 + shigh = int64(uhigh) + stop = ^shigh + ) + runTests(t, dsn, func(dbt *DBTest) { + stmt, err := dbt.db.Prepare(`SELECT ?, ?, ? ,?, ?, ?, ?, ?`) + if err != nil { + dbt.Fatal(err) + } + defer stmt.Close() + row := stmt.QueryRow( + u0, uhigh, utop, uall, + s0, shigh, stop, sall, + ) + + var ua, ub, uc, ud uint64 + var sa, sb, sc, sd int64 + + err = row.Scan(&ua, &ub, &uc, &ud, &sa, &sb, &sc, &sd) + if err != nil { + dbt.Fatal(err) + } + switch { + case ua != u0, + ub != uhigh, + uc != utop, + ud != uall, + sa != s0, + sb != shigh, + sc != stop, + sd != sall: + dbt.Fatal("Unexpected result value") + } + }) +} + +func TestLongData(t *testing.T) { + runTests(t, dsn, func(dbt *DBTest) { + var maxAllowedPacketSize int + err := dbt.db.QueryRow("select @@max_allowed_packet").Scan(&maxAllowedPacketSize) + if err != nil { + dbt.Fatal(err) + } + maxAllowedPacketSize-- + + // don't get too ambitious + if maxAllowedPacketSize > 1<<25 { + maxAllowedPacketSize = 1 << 25 + } + + dbt.mustExec("CREATE TABLE test (value LONGBLOB)") + + in := strings.Repeat(`a`, maxAllowedPacketSize+1) + var out string + var rows *sql.Rows + + // Long text data + const nonDataQueryLen = 28 // length query w/o value + inS := in[:maxAllowedPacketSize-nonDataQueryLen] + dbt.mustExec("INSERT INTO test VALUES('" + inS + "')") + rows = dbt.mustQuery("SELECT value FROM test") + if rows.Next() { + rows.Scan(&out) + if inS != out { + dbt.Fatalf("LONGBLOB: length in: %d, length out: %d", len(inS), len(out)) + } + if rows.Next() { + dbt.Error("LONGBLOB: unexpexted row") + } + } else { + dbt.Fatalf("LONGBLOB: no data") + } + + // Empty table + dbt.mustExec("TRUNCATE TABLE test") + + // Long binary data + dbt.mustExec("INSERT INTO test VALUES(?)", in) + rows = dbt.mustQuery("SELECT value FROM test WHERE 1=?", 1) + if rows.Next() { + rows.Scan(&out) + if in != out { + dbt.Fatalf("LONGBLOB: length in: %d, length out: %d", len(in), len(out)) + } + if rows.Next() { + dbt.Error("LONGBLOB: unexpexted row") + } + } else { + if err = rows.Err(); err != nil { + dbt.Fatalf("LONGBLOB: no data (err: %s)", err.Error()) + } else { + dbt.Fatal("LONGBLOB: no data (err: )") + } + } + }) +} + +func TestLoadData(t *testing.T) { + runTests(t, dsn, func(dbt *DBTest) { + verifyLoadDataResult := func() { + rows, err := dbt.db.Query("SELECT * FROM test") + if err != nil { + dbt.Fatal(err.Error()) + } + + i := 0 + values := [4]string{ + "a string", + "a string containing a \t", + "a string containing a \n", + "a string containing both \t\n", + } + + var id int + var value string + + for rows.Next() { + i++ + err = rows.Scan(&id, &value) + if err != nil { + dbt.Fatal(err.Error()) + } + if i != id { + dbt.Fatalf("%d != %d", i, id) + } + if values[i-1] != value { + dbt.Fatalf("%q != %q", values[i-1], value) + } + } + err = rows.Err() + if err != nil { + dbt.Fatal(err.Error()) + } + + if i != 4 { + dbt.Fatalf("Rows count mismatch. Got %d, want 4", i) + } + } + file, err := ioutil.TempFile("", "gotest") + defer os.Remove(file.Name()) + if err != nil { + dbt.Fatal(err) + } + file.WriteString("1\ta string\n2\ta string containing a \\t\n3\ta string containing a \\n\n4\ta string containing both \\t\\n\n") + file.Close() + + dbt.db.Exec("DROP TABLE IF EXISTS test") + dbt.mustExec("CREATE TABLE test (id INT NOT NULL PRIMARY KEY, value TEXT NOT NULL) CHARACTER SET utf8") + + // Local File + RegisterLocalFile(file.Name()) + dbt.mustExec(fmt.Sprintf("LOAD DATA LOCAL INFILE %q INTO TABLE test", file.Name())) + verifyLoadDataResult() + // negative test + _, err = dbt.db.Exec("LOAD DATA LOCAL INFILE 'doesnotexist' INTO TABLE test") + if err == nil { + dbt.Fatal("Load non-existent file didn't fail") + } else if err.Error() != "Local File 'doesnotexist' is not registered. Use the DSN parameter 'allowAllFiles=true' to allow all files" { + dbt.Fatal(err.Error()) + } + + // Empty table + dbt.mustExec("TRUNCATE TABLE test") + + // Reader + RegisterReaderHandler("test", func() io.Reader { + file, err = os.Open(file.Name()) + if err != nil { + dbt.Fatal(err) + } + return file + }) + dbt.mustExec("LOAD DATA LOCAL INFILE 'Reader::test' INTO TABLE test") + verifyLoadDataResult() + // negative test + _, err = dbt.db.Exec("LOAD DATA LOCAL INFILE 'Reader::doesnotexist' INTO TABLE test") + if err == nil { + dbt.Fatal("Load non-existent Reader didn't fail") + } else if err.Error() != "Reader 'doesnotexist' is not registered" { + dbt.Fatal(err.Error()) + } + }) +} + +func TestFoundRows(t *testing.T) { + runTests(t, dsn, func(dbt *DBTest) { + dbt.mustExec("CREATE TABLE test (id INT NOT NULL ,data INT NOT NULL)") + dbt.mustExec("INSERT INTO test (id, data) VALUES (0, 0),(0, 0),(1, 0),(1, 0),(1, 1)") + + res := dbt.mustExec("UPDATE test SET data = 1 WHERE id = 0") + count, err := res.RowsAffected() + if err != nil { + dbt.Fatalf("res.RowsAffected() returned error: %s", err.Error()) + } + if count != 2 { + dbt.Fatalf("Expected 2 affected rows, got %d", count) + } + res = dbt.mustExec("UPDATE test SET data = 1 WHERE id = 1") + count, err = res.RowsAffected() + if err != nil { + dbt.Fatalf("res.RowsAffected() returned error: %s", err.Error()) + } + if count != 2 { + dbt.Fatalf("Expected 2 affected rows, got %d", count) + } + }) + runTests(t, dsn+"&clientFoundRows=true", func(dbt *DBTest) { + dbt.mustExec("CREATE TABLE test (id INT NOT NULL ,data INT NOT NULL)") + dbt.mustExec("INSERT INTO test (id, data) VALUES (0, 0),(0, 0),(1, 0),(1, 0),(1, 1)") + + res := dbt.mustExec("UPDATE test SET data = 1 WHERE id = 0") + count, err := res.RowsAffected() + if err != nil { + dbt.Fatalf("res.RowsAffected() returned error: %s", err.Error()) + } + if count != 2 { + dbt.Fatalf("Expected 2 matched rows, got %d", count) + } + res = dbt.mustExec("UPDATE test SET data = 1 WHERE id = 1") + count, err = res.RowsAffected() + if err != nil { + dbt.Fatalf("res.RowsAffected() returned error: %s", err.Error()) + } + if count != 3 { + dbt.Fatalf("Expected 3 matched rows, got %d", count) + } + }) +} + +func TestStrict(t *testing.T) { + // ALLOW_INVALID_DATES to get rid of stricter modes - we want to test for warnings, not errors + relaxedDsn := dsn + "&sql_mode=ALLOW_INVALID_DATES" + // make sure the MySQL version is recent enough with a separate connection + // before running the test + conn, err := MySQLDriver{}.Open(relaxedDsn) + if conn != nil { + conn.Close() + } + if me, ok := err.(*MySQLError); ok && me.Number == 1231 { + // Error 1231: Variable 'sql_mode' can't be set to the value of 'ALLOW_INVALID_DATES' + // => skip test, MySQL server version is too old + return + } + runTests(t, relaxedDsn, func(dbt *DBTest) { + dbt.mustExec("CREATE TABLE test (a TINYINT NOT NULL, b CHAR(4))") + + var queries = [...]struct { + in string + codes []string + }{ + {"DROP TABLE IF EXISTS no_such_table", []string{"1051"}}, + {"INSERT INTO test VALUES(10,'mysql'),(NULL,'test'),(300,'Open Source')", []string{"1265", "1048", "1264", "1265"}}, + } + var err error + + var checkWarnings = func(err error, mode string, idx int) { + if err == nil { + dbt.Errorf("Expected STRICT error on query [%s] %s", mode, queries[idx].in) + } + + if warnings, ok := err.(MySQLWarnings); ok { + var codes = make([]string, len(warnings)) + for i := range warnings { + codes[i] = warnings[i].Code + } + if len(codes) != len(queries[idx].codes) { + dbt.Errorf("Unexpected STRICT error count on query [%s] %s: Wanted %v, Got %v", mode, queries[idx].in, queries[idx].codes, codes) + } + + for i := range warnings { + if codes[i] != queries[idx].codes[i] { + dbt.Errorf("Unexpected STRICT error codes on query [%s] %s: Wanted %v, Got %v", mode, queries[idx].in, queries[idx].codes, codes) + return + } + } + + } else { + dbt.Errorf("Unexpected error on query [%s] %s: %s", mode, queries[idx].in, err.Error()) + } + } + + // text protocol + for i := range queries { + _, err = dbt.db.Exec(queries[i].in) + checkWarnings(err, "text", i) + } + + var stmt *sql.Stmt + + // binary protocol + for i := range queries { + stmt, err = dbt.db.Prepare(queries[i].in) + if err != nil { + dbt.Errorf("Error on preparing query %s: %s", queries[i].in, err.Error()) + } + + _, err = stmt.Exec() + checkWarnings(err, "binary", i) + + err = stmt.Close() + if err != nil { + dbt.Errorf("Error on closing stmt for query %s: %s", queries[i].in, err.Error()) + } + } + }) +} + +func TestTLS(t *testing.T) { + tlsTest := func(dbt *DBTest) { + if err := dbt.db.Ping(); err != nil { + if err == ErrNoTLS { + dbt.Skip("Server does not support TLS") + } else { + dbt.Fatalf("Error on Ping: %s", err.Error()) + } + } + + rows := dbt.mustQuery("SHOW STATUS LIKE 'Ssl_cipher'") + + var variable, value *sql.RawBytes + for rows.Next() { + if err := rows.Scan(&variable, &value); err != nil { + dbt.Fatal(err.Error()) + } + + if value == nil { + dbt.Fatal("No Cipher") + } + } + } + + runTests(t, dsn+"&tls=skip-verify", tlsTest) + + // Verify that registering / using a custom cfg works + RegisterTLSConfig("custom-skip-verify", &tls.Config{ + InsecureSkipVerify: true, + }) + runTests(t, dsn+"&tls=custom-skip-verify", tlsTest) +} + +func TestReuseClosedConnection(t *testing.T) { + // this test does not use sql.database, it uses the driver directly + if !available { + t.Skipf("MySQL-Server not running on %s", netAddr) + } + + md := &MySQLDriver{} + conn, err := md.Open(dsn) + if err != nil { + t.Fatalf("Error connecting: %s", err.Error()) + } + stmt, err := conn.Prepare("DO 1") + if err != nil { + t.Fatalf("Error preparing statement: %s", err.Error()) + } + _, err = stmt.Exec(nil) + if err != nil { + t.Fatalf("Error executing statement: %s", err.Error()) + } + err = conn.Close() + if err != nil { + t.Fatalf("Error closing connection: %s", err.Error()) + } + + defer func() { + if err := recover(); err != nil { + t.Errorf("Panic after reusing a closed connection: %v", err) + } + }() + _, err = stmt.Exec(nil) + if err != nil && err != driver.ErrBadConn { + t.Errorf("Unexpected error '%s', expected '%s'", + err.Error(), driver.ErrBadConn.Error()) + } +} + +func TestCharset(t *testing.T) { + if !available { + t.Skipf("MySQL-Server not running on %s", netAddr) + } + + mustSetCharset := func(charsetParam, expected string) { + runTests(t, dsn+"&"+charsetParam, func(dbt *DBTest) { + rows := dbt.mustQuery("SELECT @@character_set_connection") + defer rows.Close() + + if !rows.Next() { + dbt.Fatalf("Error getting connection charset: %s", rows.Err()) + } + + var got string + rows.Scan(&got) + + if got != expected { + dbt.Fatalf("Expected connection charset %s but got %s", expected, got) + } + }) + } + + // non utf8 test + mustSetCharset("charset=ascii", "ascii") + + // when the first charset is invalid, use the second + mustSetCharset("charset=none,utf8", "utf8") + + // when the first charset is valid, use it + mustSetCharset("charset=ascii,utf8", "ascii") + mustSetCharset("charset=utf8,ascii", "utf8") +} + +func TestFailingCharset(t *testing.T) { + runTests(t, dsn+"&charset=none", func(dbt *DBTest) { + // run query to really establish connection... + _, err := dbt.db.Exec("SELECT 1") + if err == nil { + dbt.db.Close() + t.Fatalf("Connection must not succeed without a valid charset") + } + }) +} + +func TestCollation(t *testing.T) { + if !available { + t.Skipf("MySQL-Server not running on %s", netAddr) + } + + defaultCollation := "utf8_general_ci" + testCollations := []string{ + "", // do not set + defaultCollation, // driver default + "latin1_general_ci", + "binary", + "utf8_unicode_ci", + "cp1257_bin", + } + + for _, collation := range testCollations { + var expected, tdsn string + if collation != "" { + tdsn = dsn + "&collation=" + collation + expected = collation + } else { + tdsn = dsn + expected = defaultCollation + } + + runTests(t, tdsn, func(dbt *DBTest) { + var got string + if err := dbt.db.QueryRow("SELECT @@collation_connection").Scan(&got); err != nil { + dbt.Fatal(err) + } + + if got != expected { + dbt.Fatalf("Expected connection collation %s but got %s", expected, got) + } + }) + } +} + +func TestColumnsWithAlias(t *testing.T) { + runTests(t, dsn+"&columnsWithAlias=true", func(dbt *DBTest) { + rows := dbt.mustQuery("SELECT 1 AS A") + defer rows.Close() + cols, _ := rows.Columns() + if len(cols) != 1 { + t.Fatalf("expected 1 column, got %d", len(cols)) + } + if cols[0] != "A" { + t.Fatalf("expected column name \"A\", got \"%s\"", cols[0]) + } + rows.Close() + + rows = dbt.mustQuery("SELECT * FROM (SELECT 1 AS one) AS A") + cols, _ = rows.Columns() + if len(cols) != 1 { + t.Fatalf("expected 1 column, got %d", len(cols)) + } + if cols[0] != "A.one" { + t.Fatalf("expected column name \"A.one\", got \"%s\"", cols[0]) + } + }) +} + +func TestRawBytesResultExceedsBuffer(t *testing.T) { + runTests(t, dsn, func(dbt *DBTest) { + // defaultBufSize from buffer.go + expected := strings.Repeat("abc", defaultBufSize) + + rows := dbt.mustQuery("SELECT '" + expected + "'") + defer rows.Close() + if !rows.Next() { + dbt.Error("expected result, got none") + } + var result sql.RawBytes + rows.Scan(&result) + if expected != string(result) { + dbt.Error("result did not match expected value") + } + }) +} + +func TestTimezoneConversion(t *testing.T) { + zones := []string{"UTC", "US/Central", "US/Pacific", "Local"} + + // Regression test for timezone handling + tzTest := func(dbt *DBTest) { + + // Create table + dbt.mustExec("CREATE TABLE test (ts TIMESTAMP)") + + // Insert local time into database (should be converted) + usCentral, _ := time.LoadLocation("US/Central") + reftime := time.Date(2014, 05, 30, 18, 03, 17, 0, time.UTC).In(usCentral) + dbt.mustExec("INSERT INTO test VALUE (?)", reftime) + + // Retrieve time from DB + rows := dbt.mustQuery("SELECT ts FROM test") + if !rows.Next() { + dbt.Fatal("Didn't get any rows out") + } + + var dbTime time.Time + err := rows.Scan(&dbTime) + if err != nil { + dbt.Fatal("Err", err) + } + + // Check that dates match + if reftime.Unix() != dbTime.Unix() { + dbt.Errorf("Times don't match.\n") + dbt.Errorf(" Now(%v)=%v\n", usCentral, reftime) + dbt.Errorf(" Now(UTC)=%v\n", dbTime) + } + } + + for _, tz := range zones { + runTests(t, dsn+"&parseTime=true&loc="+url.QueryEscape(tz), tzTest) + } +} + +// Special cases + +func TestRowsClose(t *testing.T) { + runTests(t, dsn, func(dbt *DBTest) { + rows, err := dbt.db.Query("SELECT 1") + if err != nil { + dbt.Fatal(err) + } + + err = rows.Close() + if err != nil { + dbt.Fatal(err) + } + + if rows.Next() { + dbt.Fatal("Unexpected row after rows.Close()") + } + + err = rows.Err() + if err != nil { + dbt.Fatal(err) + } + }) +} + +// dangling statements +// http://code.google.com/p/go/issues/detail?id=3865 +func TestCloseStmtBeforeRows(t *testing.T) { + runTests(t, dsn, func(dbt *DBTest) { + stmt, err := dbt.db.Prepare("SELECT 1") + if err != nil { + dbt.Fatal(err) + } + + rows, err := stmt.Query() + if err != nil { + stmt.Close() + dbt.Fatal(err) + } + defer rows.Close() + + err = stmt.Close() + if err != nil { + dbt.Fatal(err) + } + + if !rows.Next() { + dbt.Fatal("Getting row failed") + } else { + err = rows.Err() + if err != nil { + dbt.Fatal(err) + } + + var out bool + err = rows.Scan(&out) + if err != nil { + dbt.Fatalf("Error on rows.Scan(): %s", err.Error()) + } + if out != true { + dbt.Errorf("true != %t", out) + } + } + }) +} + +// It is valid to have multiple Rows for the same Stmt +// http://code.google.com/p/go/issues/detail?id=3734 +func TestStmtMultiRows(t *testing.T) { + runTests(t, dsn, func(dbt *DBTest) { + stmt, err := dbt.db.Prepare("SELECT 1 UNION SELECT 0") + if err != nil { + dbt.Fatal(err) + } + + rows1, err := stmt.Query() + if err != nil { + stmt.Close() + dbt.Fatal(err) + } + defer rows1.Close() + + rows2, err := stmt.Query() + if err != nil { + stmt.Close() + dbt.Fatal(err) + } + defer rows2.Close() + + var out bool + + // 1 + if !rows1.Next() { + dbt.Fatal("1st rows1.Next failed") + } else { + err = rows1.Err() + if err != nil { + dbt.Fatal(err) + } + + err = rows1.Scan(&out) + if err != nil { + dbt.Fatalf("Error on rows.Scan(): %s", err.Error()) + } + if out != true { + dbt.Errorf("true != %t", out) + } + } + + if !rows2.Next() { + dbt.Fatal("1st rows2.Next failed") + } else { + err = rows2.Err() + if err != nil { + dbt.Fatal(err) + } + + err = rows2.Scan(&out) + if err != nil { + dbt.Fatalf("Error on rows.Scan(): %s", err.Error()) + } + if out != true { + dbt.Errorf("true != %t", out) + } + } + + // 2 + if !rows1.Next() { + dbt.Fatal("2nd rows1.Next failed") + } else { + err = rows1.Err() + if err != nil { + dbt.Fatal(err) + } + + err = rows1.Scan(&out) + if err != nil { + dbt.Fatalf("Error on rows.Scan(): %s", err.Error()) + } + if out != false { + dbt.Errorf("false != %t", out) + } + + if rows1.Next() { + dbt.Fatal("Unexpected row on rows1") + } + err = rows1.Close() + if err != nil { + dbt.Fatal(err) + } + } + + if !rows2.Next() { + dbt.Fatal("2nd rows2.Next failed") + } else { + err = rows2.Err() + if err != nil { + dbt.Fatal(err) + } + + err = rows2.Scan(&out) + if err != nil { + dbt.Fatalf("Error on rows.Scan(): %s", err.Error()) + } + if out != false { + dbt.Errorf("false != %t", out) + } + + if rows2.Next() { + dbt.Fatal("Unexpected row on rows2") + } + err = rows2.Close() + if err != nil { + dbt.Fatal(err) + } + } + }) +} + +// Regression test for +// * more than 32 NULL parameters (issue 209) +// * more parameters than fit into the buffer (issue 201) +func TestPreparedManyCols(t *testing.T) { + const numParams = defaultBufSize + runTests(t, dsn, func(dbt *DBTest) { + query := "SELECT ?" + strings.Repeat(",?", numParams-1) + stmt, err := dbt.db.Prepare(query) + if err != nil { + dbt.Fatal(err) + } + defer stmt.Close() + // create more parameters than fit into the buffer + // which will take nil-values + params := make([]interface{}, numParams) + rows, err := stmt.Query(params...) + if err != nil { + stmt.Close() + dbt.Fatal(err) + } + defer rows.Close() + }) +} + +func TestConcurrent(t *testing.T) { + if enabled, _ := readBool(os.Getenv("MYSQL_TEST_CONCURRENT")); !enabled { + t.Skip("MYSQL_TEST_CONCURRENT env var not set") + } + + runTests(t, dsn, func(dbt *DBTest) { + var max int + err := dbt.db.QueryRow("SELECT @@max_connections").Scan(&max) + if err != nil { + dbt.Fatalf("%s", err.Error()) + } + dbt.Logf("Testing up to %d concurrent connections \r\n", max) + + var remaining, succeeded int32 = int32(max), 0 + + var wg sync.WaitGroup + wg.Add(max) + + var fatalError string + var once sync.Once + fatalf := func(s string, vals ...interface{}) { + once.Do(func() { + fatalError = fmt.Sprintf(s, vals...) + }) + } + + for i := 0; i < max; i++ { + go func(id int) { + defer wg.Done() + + tx, err := dbt.db.Begin() + atomic.AddInt32(&remaining, -1) + + if err != nil { + if err.Error() != "Error 1040: Too many connections" { + fatalf("Error on Conn %d: %s", id, err.Error()) + } + return + } + + // keep the connection busy until all connections are open + for remaining > 0 { + if _, err = tx.Exec("DO 1"); err != nil { + fatalf("Error on Conn %d: %s", id, err.Error()) + return + } + } + + if err = tx.Commit(); err != nil { + fatalf("Error on Conn %d: %s", id, err.Error()) + return + } + + // everything went fine with this connection + atomic.AddInt32(&succeeded, 1) + }(i) + } + + // wait until all conections are open + wg.Wait() + + if fatalError != "" { + dbt.Fatal(fatalError) + } + + dbt.Logf("Reached %d concurrent connections\r\n", succeeded) + }) +} + +// Tests custom dial functions +func TestCustomDial(t *testing.T) { + if !available { + t.Skipf("MySQL-Server not running on %s", netAddr) + } + + // our custom dial function which justs wraps net.Dial here + RegisterDial("mydial", func(addr string) (net.Conn, error) { + return net.Dial(prot, addr) + }) + + db, err := sql.Open("mysql", fmt.Sprintf("%s:%s@mydial(%s)/%s?timeout=30s&strict=true", user, pass, addr, dbname)) + if err != nil { + t.Fatalf("Error connecting: %s", err.Error()) + } + defer db.Close() + + if _, err = db.Exec("DO 1"); err != nil { + t.Fatalf("Connection failed: %s", err.Error()) + } +} + +func TestSqlInjection(t *testing.T) { + createTest := func(arg string) func(dbt *DBTest) { + return func(dbt *DBTest) { + dbt.mustExec("CREATE TABLE test (v INTEGER)") + dbt.mustExec("INSERT INTO test VALUES (?)", 1) + + var v int + // NULL can't be equal to anything, the idea here is to inject query so it returns row + // This test verifies that escapeQuotes and escapeBackslash are working properly + err := dbt.db.QueryRow("SELECT v FROM test WHERE NULL = ?", arg).Scan(&v) + if err == sql.ErrNoRows { + return // success, sql injection failed + } else if err == nil { + dbt.Errorf("Sql injection successful with arg: %s", arg) + } else { + dbt.Errorf("Error running query with arg: %s; err: %s", arg, err.Error()) + } + } + } + + dsns := []string{ + dsn, + dsn + "&sql_mode=NO_BACKSLASH_ESCAPES", + } + for _, testdsn := range dsns { + runTests(t, testdsn, createTest("1 OR 1=1")) + runTests(t, testdsn, createTest("' OR '1'='1")) + } +} + +// Test if inserted data is correctly retrieved after being escaped +func TestInsertRetrieveEscapedData(t *testing.T) { + testData := func(dbt *DBTest) { + dbt.mustExec("CREATE TABLE test (v VARCHAR(255))") + + // All sequences that are escaped by escapeQuotes and escapeBackslash + v := "foo \x00\n\r\x1a\"'\\" + dbt.mustExec("INSERT INTO test VALUES (?)", v) + + var out string + err := dbt.db.QueryRow("SELECT v FROM test").Scan(&out) + if err != nil { + dbt.Fatalf("%s", err.Error()) + } + + if out != v { + dbt.Errorf("%q != %q", out, v) + } + } + + dsns := []string{ + dsn, + dsn + "&sql_mode=NO_BACKSLASH_ESCAPES", + } + for _, testdsn := range dsns { + runTests(t, testdsn, testData) + } +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/errors.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/errors.go new file mode 100644 index 0000000000000..44cf30db60308 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/errors.go @@ -0,0 +1,131 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2013 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +import ( + "database/sql/driver" + "errors" + "fmt" + "io" + "log" + "os" +) + +// Various errors the driver might return. Can change between driver versions. +var ( + ErrInvalidConn = errors.New("Invalid Connection") + ErrMalformPkt = errors.New("Malformed Packet") + ErrNoTLS = errors.New("TLS encryption requested but server does not support TLS") + ErrOldPassword = errors.New("This user requires old password authentication. If you still want to use it, please add 'allowOldPasswords=1' to your DSN. See also https://github.com/go-sql-driver/mysql/wiki/old_passwords") + ErrCleartextPassword = errors.New("This user requires clear text authentication. If you still want to use it, please add 'allowCleartextPasswords=1' to your DSN.") + ErrUnknownPlugin = errors.New("The authentication plugin is not supported.") + ErrOldProtocol = errors.New("MySQL-Server does not support required Protocol 41+") + ErrPktSync = errors.New("Commands out of sync. You can't run this command now") + ErrPktSyncMul = errors.New("Commands out of sync. Did you run multiple statements at once?") + ErrPktTooLarge = errors.New("Packet for query is too large. You can change this value on the server by adjusting the 'max_allowed_packet' variable.") + ErrBusyBuffer = errors.New("Busy buffer") +) + +var errLog Logger = log.New(os.Stderr, "[MySQL] ", log.Ldate|log.Ltime|log.Lshortfile) + +// Logger is used to log critical error messages. +type Logger interface { + Print(v ...interface{}) +} + +// SetLogger is used to set the logger for critical errors. +// The initial logger is os.Stderr. +func SetLogger(logger Logger) error { + if logger == nil { + return errors.New("logger is nil") + } + errLog = logger + return nil +} + +// MySQLError is an error type which represents a single MySQL error +type MySQLError struct { + Number uint16 + Message string +} + +func (me *MySQLError) Error() string { + return fmt.Sprintf("Error %d: %s", me.Number, me.Message) +} + +// MySQLWarnings is an error type which represents a group of one or more MySQL +// warnings +type MySQLWarnings []MySQLWarning + +func (mws MySQLWarnings) Error() string { + var msg string + for i, warning := range mws { + if i > 0 { + msg += "\r\n" + } + msg += fmt.Sprintf( + "%s %s: %s", + warning.Level, + warning.Code, + warning.Message, + ) + } + return msg +} + +// MySQLWarning is an error type which represents a single MySQL warning. +// Warnings are returned in groups only. See MySQLWarnings +type MySQLWarning struct { + Level string + Code string + Message string +} + +func (mc *mysqlConn) getWarnings() (err error) { + rows, err := mc.Query("SHOW WARNINGS", nil) + if err != nil { + return + } + + var warnings = MySQLWarnings{} + var values = make([]driver.Value, 3) + + for { + err = rows.Next(values) + switch err { + case nil: + warning := MySQLWarning{} + + if raw, ok := values[0].([]byte); ok { + warning.Level = string(raw) + } else { + warning.Level = fmt.Sprintf("%s", values[0]) + } + if raw, ok := values[1].([]byte); ok { + warning.Code = string(raw) + } else { + warning.Code = fmt.Sprintf("%s", values[1]) + } + if raw, ok := values[2].([]byte); ok { + warning.Message = string(raw) + } else { + warning.Message = fmt.Sprintf("%s", values[0]) + } + + warnings = append(warnings, warning) + + case io.EOF: + return warnings + + default: + rows.Close() + return + } + } +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/errors_test.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/errors_test.go new file mode 100644 index 0000000000000..96f9126d679a4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/errors_test.go @@ -0,0 +1,42 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2013 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +import ( + "bytes" + "log" + "testing" +) + +func TestErrorsSetLogger(t *testing.T) { + previous := errLog + defer func() { + errLog = previous + }() + + // set up logger + const expected = "prefix: test\n" + buffer := bytes.NewBuffer(make([]byte, 0, 64)) + logger := log.New(buffer, "prefix: ", 0) + + // print + SetLogger(logger) + errLog.Print("test") + + // check result + if actual := buffer.String(); actual != expected { + t.Errorf("expected %q, got %q", expected, actual) + } +} + +func TestErrorsStrictIgnoreNotes(t *testing.T) { + runTests(t, dsn+"&sql_notes=false", func(dbt *DBTest) { + dbt.mustExec("DROP TABLE IF EXISTS does_not_exist") + }) +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/infile.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/infile.go new file mode 100644 index 0000000000000..a2dedb3c0aa73 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/infile.go @@ -0,0 +1,164 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2013 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +import ( + "fmt" + "io" + "os" + "strings" +) + +var ( + fileRegister map[string]bool + readerRegister map[string]func() io.Reader +) + +// RegisterLocalFile adds the given file to the file whitelist, +// so that it can be used by "LOAD DATA LOCAL INFILE ". +// Alternatively you can allow the use of all local files with +// the DSN parameter 'allowAllFiles=true' +// +// filePath := "/home/gopher/data.csv" +// mysql.RegisterLocalFile(filePath) +// err := db.Exec("LOAD DATA LOCAL INFILE '" + filePath + "' INTO TABLE foo") +// if err != nil { +// ... +// +func RegisterLocalFile(filePath string) { + // lazy map init + if fileRegister == nil { + fileRegister = make(map[string]bool) + } + + fileRegister[strings.Trim(filePath, `"`)] = true +} + +// DeregisterLocalFile removes the given filepath from the whitelist. +func DeregisterLocalFile(filePath string) { + delete(fileRegister, strings.Trim(filePath, `"`)) +} + +// RegisterReaderHandler registers a handler function which is used +// to receive a io.Reader. +// The Reader can be used by "LOAD DATA LOCAL INFILE Reader::". +// If the handler returns a io.ReadCloser Close() is called when the +// request is finished. +// +// mysql.RegisterReaderHandler("data", func() io.Reader { +// var csvReader io.Reader // Some Reader that returns CSV data +// ... // Open Reader here +// return csvReader +// }) +// err := db.Exec("LOAD DATA LOCAL INFILE 'Reader::data' INTO TABLE foo") +// if err != nil { +// ... +// +func RegisterReaderHandler(name string, handler func() io.Reader) { + // lazy map init + if readerRegister == nil { + readerRegister = make(map[string]func() io.Reader) + } + + readerRegister[name] = handler +} + +// DeregisterReaderHandler removes the ReaderHandler function with +// the given name from the registry. +func DeregisterReaderHandler(name string) { + delete(readerRegister, name) +} + +func deferredClose(err *error, closer io.Closer) { + closeErr := closer.Close() + if *err == nil { + *err = closeErr + } +} + +func (mc *mysqlConn) handleInFileRequest(name string) (err error) { + var rdr io.Reader + var data []byte + + if idx := strings.Index(name, "Reader::"); idx == 0 || (idx > 0 && name[idx-1] == '/') { // io.Reader + // The server might return an an absolute path. See issue #355. + name = name[idx+8:] + + if handler, inMap := readerRegister[name]; inMap { + rdr = handler() + if rdr != nil { + data = make([]byte, 4+mc.maxWriteSize) + + if cl, ok := rdr.(io.Closer); ok { + defer deferredClose(&err, cl) + } + } else { + err = fmt.Errorf("Reader '%s' is ", name) + } + } else { + err = fmt.Errorf("Reader '%s' is not registered", name) + } + } else { // File + name = strings.Trim(name, `"`) + if mc.cfg.allowAllFiles || fileRegister[name] { + var file *os.File + var fi os.FileInfo + + if file, err = os.Open(name); err == nil { + defer deferredClose(&err, file) + + // get file size + if fi, err = file.Stat(); err == nil { + rdr = file + if fileSize := int(fi.Size()); fileSize <= mc.maxWriteSize { + data = make([]byte, 4+fileSize) + } else if fileSize <= mc.maxPacketAllowed { + data = make([]byte, 4+mc.maxWriteSize) + } else { + err = fmt.Errorf("Local File '%s' too large: Size: %d, Max: %d", name, fileSize, mc.maxPacketAllowed) + } + } + } + } else { + err = fmt.Errorf("Local File '%s' is not registered. Use the DSN parameter 'allowAllFiles=true' to allow all files", name) + } + } + + // send content packets + if err == nil { + var n int + for err == nil { + n, err = rdr.Read(data[4:]) + if n > 0 { + if ioErr := mc.writePacket(data[:4+n]); ioErr != nil { + return ioErr + } + } + } + if err == io.EOF { + err = nil + } + } + + // send empty packet (termination) + if data == nil { + data = make([]byte, 4) + } + if ioErr := mc.writePacket(data[:4]); ioErr != nil { + return ioErr + } + + // read OK packet + if err == nil { + return mc.readResultOK() + } else { + mc.readPacket() + } + return err +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/packets.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/packets.go new file mode 100644 index 0000000000000..14395bf9ab524 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/packets.go @@ -0,0 +1,1179 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2012 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +import ( + "bytes" + "crypto/tls" + "database/sql/driver" + "encoding/binary" + "fmt" + "io" + "math" + "time" +) + +// Packets documentation: +// http://dev.mysql.com/doc/internals/en/client-server-protocol.html + +// Read packet to buffer 'data' +func (mc *mysqlConn) readPacket() ([]byte, error) { + var payload []byte + for { + // Read packet header + data, err := mc.buf.readNext(4) + if err != nil { + errLog.Print(err) + mc.Close() + return nil, driver.ErrBadConn + } + + // Packet Length [24 bit] + pktLen := int(uint32(data[0]) | uint32(data[1])<<8 | uint32(data[2])<<16) + + if pktLen < 1 { + errLog.Print(ErrMalformPkt) + mc.Close() + return nil, driver.ErrBadConn + } + + // Check Packet Sync [8 bit] + if data[3] != mc.sequence { + if data[3] > mc.sequence { + return nil, ErrPktSyncMul + } else { + return nil, ErrPktSync + } + } + mc.sequence++ + + // Read packet body [pktLen bytes] + data, err = mc.buf.readNext(pktLen) + if err != nil { + errLog.Print(err) + mc.Close() + return nil, driver.ErrBadConn + } + + isLastPacket := (pktLen < maxPacketSize) + + // Zero allocations for non-splitting packets + if isLastPacket && payload == nil { + return data, nil + } + + payload = append(payload, data...) + + if isLastPacket { + return payload, nil + } + } +} + +// Write packet buffer 'data' +func (mc *mysqlConn) writePacket(data []byte) error { + pktLen := len(data) - 4 + + if pktLen > mc.maxPacketAllowed { + return ErrPktTooLarge + } + + for { + var size int + if pktLen >= maxPacketSize { + data[0] = 0xff + data[1] = 0xff + data[2] = 0xff + size = maxPacketSize + } else { + data[0] = byte(pktLen) + data[1] = byte(pktLen >> 8) + data[2] = byte(pktLen >> 16) + size = pktLen + } + data[3] = mc.sequence + + // Write packet + n, err := mc.netConn.Write(data[:4+size]) + if err == nil && n == 4+size { + mc.sequence++ + if size != maxPacketSize { + return nil + } + pktLen -= size + data = data[size:] + continue + } + + // Handle error + if err == nil { // n != len(data) + errLog.Print(ErrMalformPkt) + } else { + errLog.Print(err) + } + return driver.ErrBadConn + } +} + +/****************************************************************************** +* Initialisation Process * +******************************************************************************/ + +// Handshake Initialization Packet +// http://dev.mysql.com/doc/internals/en/connection-phase-packets.html#packet-Protocol::Handshake +func (mc *mysqlConn) readInitPacket() ([]byte, error) { + data, err := mc.readPacket() + if err != nil { + return nil, err + } + + if data[0] == iERR { + return nil, mc.handleErrorPacket(data) + } + + // protocol version [1 byte] + if data[0] < minProtocolVersion { + return nil, fmt.Errorf( + "Unsupported MySQL Protocol Version %d. Protocol Version %d or higher is required", + data[0], + minProtocolVersion, + ) + } + + // server version [null terminated string] + // connection id [4 bytes] + pos := 1 + bytes.IndexByte(data[1:], 0x00) + 1 + 4 + + // first part of the password cipher [8 bytes] + cipher := data[pos : pos+8] + + // (filler) always 0x00 [1 byte] + pos += 8 + 1 + + // capability flags (lower 2 bytes) [2 bytes] + mc.flags = clientFlag(binary.LittleEndian.Uint16(data[pos : pos+2])) + if mc.flags&clientProtocol41 == 0 { + return nil, ErrOldProtocol + } + if mc.flags&clientSSL == 0 && mc.cfg.tls != nil { + return nil, ErrNoTLS + } + pos += 2 + + if len(data) > pos { + // character set [1 byte] + // status flags [2 bytes] + // capability flags (upper 2 bytes) [2 bytes] + // length of auth-plugin-data [1 byte] + // reserved (all [00]) [10 bytes] + pos += 1 + 2 + 2 + 1 + 10 + + // second part of the password cipher [mininum 13 bytes], + // where len=MAX(13, length of auth-plugin-data - 8) + // + // The web documentation is ambiguous about the length. However, + // according to mysql-5.7/sql/auth/sql_authentication.cc line 538, + // the 13th byte is "\0 byte, terminating the second part of + // a scramble". So the second part of the password cipher is + // a NULL terminated string that's at least 13 bytes with the + // last byte being NULL. + // + // The official Python library uses the fixed length 12 + // which seems to work but technically could have a hidden bug. + cipher = append(cipher, data[pos:pos+12]...) + + // TODO: Verify string termination + // EOF if version (>= 5.5.7 and < 5.5.10) or (>= 5.6.0 and < 5.6.2) + // \NUL otherwise + // + //if data[len(data)-1] == 0 { + // return + //} + //return ErrMalformPkt + + // make a memory safe copy of the cipher slice + var b [20]byte + copy(b[:], cipher) + return b[:], nil + } + + // make a memory safe copy of the cipher slice + var b [8]byte + copy(b[:], cipher) + return b[:], nil +} + +// Client Authentication Packet +// http://dev.mysql.com/doc/internals/en/connection-phase-packets.html#packet-Protocol::HandshakeResponse +func (mc *mysqlConn) writeAuthPacket(cipher []byte) error { + // Adjust client flags based on server support + clientFlags := clientProtocol41 | + clientSecureConn | + clientLongPassword | + clientTransactions | + clientLocalFiles | + clientPluginAuth | + mc.flags&clientLongFlag + + if mc.cfg.clientFoundRows { + clientFlags |= clientFoundRows + } + + // To enable TLS / SSL + if mc.cfg.tls != nil { + clientFlags |= clientSSL + } + + // User Password + scrambleBuff := scramblePassword(cipher, []byte(mc.cfg.passwd)) + + pktLen := 4 + 4 + 1 + 23 + len(mc.cfg.user) + 1 + 1 + len(scrambleBuff) + 21 + 1 + + // To specify a db name + if n := len(mc.cfg.dbname); n > 0 { + clientFlags |= clientConnectWithDB + pktLen += n + 1 + } + + // Calculate packet length and get buffer with that size + data := mc.buf.takeSmallBuffer(pktLen + 4) + if data == nil { + // can not take the buffer. Something must be wrong with the connection + errLog.Print(ErrBusyBuffer) + return driver.ErrBadConn + } + + // ClientFlags [32 bit] + data[4] = byte(clientFlags) + data[5] = byte(clientFlags >> 8) + data[6] = byte(clientFlags >> 16) + data[7] = byte(clientFlags >> 24) + + // MaxPacketSize [32 bit] (none) + data[8] = 0x00 + data[9] = 0x00 + data[10] = 0x00 + data[11] = 0x00 + + // Charset [1 byte] + data[12] = mc.cfg.collation + + // SSL Connection Request Packet + // http://dev.mysql.com/doc/internals/en/connection-phase-packets.html#packet-Protocol::SSLRequest + if mc.cfg.tls != nil { + // Send TLS / SSL request packet + if err := mc.writePacket(data[:(4+4+1+23)+4]); err != nil { + return err + } + + // Switch to TLS + tlsConn := tls.Client(mc.netConn, mc.cfg.tls) + if err := tlsConn.Handshake(); err != nil { + return err + } + mc.netConn = tlsConn + mc.buf.rd = tlsConn + } + + // Filler [23 bytes] (all 0x00) + pos := 13 + 23 + + // User [null terminated string] + if len(mc.cfg.user) > 0 { + pos += copy(data[pos:], mc.cfg.user) + } + data[pos] = 0x00 + pos++ + + // ScrambleBuffer [length encoded integer] + data[pos] = byte(len(scrambleBuff)) + pos += 1 + copy(data[pos+1:], scrambleBuff) + + // Databasename [null terminated string] + if len(mc.cfg.dbname) > 0 { + pos += copy(data[pos:], mc.cfg.dbname) + data[pos] = 0x00 + pos++ + } + + // Assume native client during response + pos += copy(data[pos:], "mysql_native_password") + data[pos] = 0x00 + + // Send Auth packet + return mc.writePacket(data) +} + +// Client old authentication packet +// http://dev.mysql.com/doc/internals/en/connection-phase-packets.html#packet-Protocol::AuthSwitchResponse +func (mc *mysqlConn) writeOldAuthPacket(cipher []byte) error { + // User password + scrambleBuff := scrambleOldPassword(cipher, []byte(mc.cfg.passwd)) + + // Calculate the packet length and add a tailing 0 + pktLen := len(scrambleBuff) + 1 + data := mc.buf.takeSmallBuffer(4 + pktLen) + if data == nil { + // can not take the buffer. Something must be wrong with the connection + errLog.Print(ErrBusyBuffer) + return driver.ErrBadConn + } + + // Add the scrambled password [null terminated string] + copy(data[4:], scrambleBuff) + data[4+pktLen-1] = 0x00 + + return mc.writePacket(data) +} + +// Client clear text authentication packet +// http://dev.mysql.com/doc/internals/en/connection-phase-packets.html#packet-Protocol::AuthSwitchResponse +func (mc *mysqlConn) writeClearAuthPacket() error { + // Calculate the packet length and add a tailing 0 + pktLen := len(mc.cfg.passwd) + 1 + data := mc.buf.takeSmallBuffer(4 + pktLen) + if data == nil { + // can not take the buffer. Something must be wrong with the connection + errLog.Print(ErrBusyBuffer) + return driver.ErrBadConn + } + + // Add the clear password [null terminated string] + copy(data[4:], mc.cfg.passwd) + data[4+pktLen-1] = 0x00 + + return mc.writePacket(data) +} + +/****************************************************************************** +* Command Packets * +******************************************************************************/ + +func (mc *mysqlConn) writeCommandPacket(command byte) error { + // Reset Packet Sequence + mc.sequence = 0 + + data := mc.buf.takeSmallBuffer(4 + 1) + if data == nil { + // can not take the buffer. Something must be wrong with the connection + errLog.Print(ErrBusyBuffer) + return driver.ErrBadConn + } + + // Add command byte + data[4] = command + + // Send CMD packet + return mc.writePacket(data) +} + +func (mc *mysqlConn) writeCommandPacketStr(command byte, arg string) error { + // Reset Packet Sequence + mc.sequence = 0 + + pktLen := 1 + len(arg) + data := mc.buf.takeBuffer(pktLen + 4) + if data == nil { + // can not take the buffer. Something must be wrong with the connection + errLog.Print(ErrBusyBuffer) + return driver.ErrBadConn + } + + // Add command byte + data[4] = command + + // Add arg + copy(data[5:], arg) + + // Send CMD packet + return mc.writePacket(data) +} + +func (mc *mysqlConn) writeCommandPacketUint32(command byte, arg uint32) error { + // Reset Packet Sequence + mc.sequence = 0 + + data := mc.buf.takeSmallBuffer(4 + 1 + 4) + if data == nil { + // can not take the buffer. Something must be wrong with the connection + errLog.Print(ErrBusyBuffer) + return driver.ErrBadConn + } + + // Add command byte + data[4] = command + + // Add arg [32 bit] + data[5] = byte(arg) + data[6] = byte(arg >> 8) + data[7] = byte(arg >> 16) + data[8] = byte(arg >> 24) + + // Send CMD packet + return mc.writePacket(data) +} + +/****************************************************************************** +* Result Packets * +******************************************************************************/ + +// Returns error if Packet is not an 'Result OK'-Packet +func (mc *mysqlConn) readResultOK() error { + data, err := mc.readPacket() + if err == nil { + // packet indicator + switch data[0] { + + case iOK: + return mc.handleOkPacket(data) + + case iEOF: + if len(data) > 1 { + plugin := string(data[1:bytes.IndexByte(data, 0x00)]) + if plugin == "mysql_old_password" { + // using old_passwords + return ErrOldPassword + } else if plugin == "mysql_clear_password" { + // using clear text password + return ErrCleartextPassword + } else { + return ErrUnknownPlugin + } + } else { + return ErrOldPassword + } + + default: // Error otherwise + return mc.handleErrorPacket(data) + } + } + return err +} + +// Result Set Header Packet +// http://dev.mysql.com/doc/internals/en/com-query-response.html#packet-ProtocolText::Resultset +func (mc *mysqlConn) readResultSetHeaderPacket() (int, error) { + data, err := mc.readPacket() + if err == nil { + switch data[0] { + + case iOK: + return 0, mc.handleOkPacket(data) + + case iERR: + return 0, mc.handleErrorPacket(data) + + case iLocalInFile: + return 0, mc.handleInFileRequest(string(data[1:])) + } + + // column count + num, _, n := readLengthEncodedInteger(data) + if n-len(data) == 0 { + return int(num), nil + } + + return 0, ErrMalformPkt + } + return 0, err +} + +// Error Packet +// http://dev.mysql.com/doc/internals/en/generic-response-packets.html#packet-ERR_Packet +func (mc *mysqlConn) handleErrorPacket(data []byte) error { + if data[0] != iERR { + return ErrMalformPkt + } + + // 0xff [1 byte] + + // Error Number [16 bit uint] + errno := binary.LittleEndian.Uint16(data[1:3]) + + pos := 3 + + // SQL State [optional: # + 5bytes string] + if data[3] == 0x23 { + //sqlstate := string(data[4 : 4+5]) + pos = 9 + } + + // Error Message [string] + return &MySQLError{ + Number: errno, + Message: string(data[pos:]), + } +} + +// Ok Packet +// http://dev.mysql.com/doc/internals/en/generic-response-packets.html#packet-OK_Packet +func (mc *mysqlConn) handleOkPacket(data []byte) error { + var n, m int + + // 0x00 [1 byte] + + // Affected rows [Length Coded Binary] + mc.affectedRows, _, n = readLengthEncodedInteger(data[1:]) + + // Insert id [Length Coded Binary] + mc.insertId, _, m = readLengthEncodedInteger(data[1+n:]) + + // server_status [2 bytes] + mc.status = statusFlag(data[1+n+m]) | statusFlag(data[1+n+m+1])<<8 + + // warning count [2 bytes] + if !mc.strict { + return nil + } else { + pos := 1 + n + m + 2 + if binary.LittleEndian.Uint16(data[pos:pos+2]) > 0 { + return mc.getWarnings() + } + return nil + } +} + +// Read Packets as Field Packets until EOF-Packet or an Error appears +// http://dev.mysql.com/doc/internals/en/com-query-response.html#packet-Protocol::ColumnDefinition41 +func (mc *mysqlConn) readColumns(count int) ([]mysqlField, error) { + columns := make([]mysqlField, count) + + for i := 0; ; i++ { + data, err := mc.readPacket() + if err != nil { + return nil, err + } + + // EOF Packet + if data[0] == iEOF && (len(data) == 5 || len(data) == 1) { + if i == count { + return columns, nil + } + return nil, fmt.Errorf("ColumnsCount mismatch n:%d len:%d", count, len(columns)) + } + + // Catalog + pos, err := skipLengthEncodedString(data) + if err != nil { + return nil, err + } + + // Database [len coded string] + n, err := skipLengthEncodedString(data[pos:]) + if err != nil { + return nil, err + } + pos += n + + // Table [len coded string] + if mc.cfg.columnsWithAlias { + tableName, _, n, err := readLengthEncodedString(data[pos:]) + if err != nil { + return nil, err + } + pos += n + columns[i].tableName = string(tableName) + } else { + n, err = skipLengthEncodedString(data[pos:]) + if err != nil { + return nil, err + } + pos += n + } + + // Original table [len coded string] + n, err = skipLengthEncodedString(data[pos:]) + if err != nil { + return nil, err + } + pos += n + + // Name [len coded string] + name, _, n, err := readLengthEncodedString(data[pos:]) + if err != nil { + return nil, err + } + columns[i].name = string(name) + pos += n + + // Original name [len coded string] + n, err = skipLengthEncodedString(data[pos:]) + if err != nil { + return nil, err + } + + // Filler [uint8] + // Charset [charset, collation uint8] + // Length [uint32] + pos += n + 1 + 2 + 4 + + // Field type [uint8] + columns[i].fieldType = data[pos] + pos++ + + // Flags [uint16] + columns[i].flags = fieldFlag(binary.LittleEndian.Uint16(data[pos : pos+2])) + pos += 2 + + // Decimals [uint8] + columns[i].decimals = data[pos] + //pos++ + + // Default value [len coded binary] + //if pos < len(data) { + // defaultVal, _, err = bytesToLengthCodedBinary(data[pos:]) + //} + } +} + +// Read Packets as Field Packets until EOF-Packet or an Error appears +// http://dev.mysql.com/doc/internals/en/com-query-response.html#packet-ProtocolText::ResultsetRow +func (rows *textRows) readRow(dest []driver.Value) error { + mc := rows.mc + + data, err := mc.readPacket() + if err != nil { + return err + } + + // EOF Packet + if data[0] == iEOF && len(data) == 5 { + rows.mc = nil + return io.EOF + } + if data[0] == iERR { + rows.mc = nil + return mc.handleErrorPacket(data) + } + + // RowSet Packet + var n int + var isNull bool + pos := 0 + + for i := range dest { + // Read bytes and convert to string + dest[i], isNull, n, err = readLengthEncodedString(data[pos:]) + pos += n + if err == nil { + if !isNull { + if !mc.parseTime { + continue + } else { + switch rows.columns[i].fieldType { + case fieldTypeTimestamp, fieldTypeDateTime, + fieldTypeDate, fieldTypeNewDate: + dest[i], err = parseDateTime( + string(dest[i].([]byte)), + mc.cfg.loc, + ) + if err == nil { + continue + } + default: + continue + } + } + + } else { + dest[i] = nil + continue + } + } + return err // err != nil + } + + return nil +} + +// Reads Packets until EOF-Packet or an Error appears. Returns count of Packets read +func (mc *mysqlConn) readUntilEOF() error { + for { + data, err := mc.readPacket() + + // No Err and no EOF Packet + if err == nil && data[0] != iEOF { + continue + } + return err // Err or EOF + } +} + +/****************************************************************************** +* Prepared Statements * +******************************************************************************/ + +// Prepare Result Packets +// http://dev.mysql.com/doc/internals/en/com-stmt-prepare-response.html +func (stmt *mysqlStmt) readPrepareResultPacket() (uint16, error) { + data, err := stmt.mc.readPacket() + if err == nil { + // packet indicator [1 byte] + if data[0] != iOK { + return 0, stmt.mc.handleErrorPacket(data) + } + + // statement id [4 bytes] + stmt.id = binary.LittleEndian.Uint32(data[1:5]) + + // Column count [16 bit uint] + columnCount := binary.LittleEndian.Uint16(data[5:7]) + + // Param count [16 bit uint] + stmt.paramCount = int(binary.LittleEndian.Uint16(data[7:9])) + + // Reserved [8 bit] + + // Warning count [16 bit uint] + if !stmt.mc.strict { + return columnCount, nil + } else { + // Check for warnings count > 0, only available in MySQL > 4.1 + if len(data) >= 12 && binary.LittleEndian.Uint16(data[10:12]) > 0 { + return columnCount, stmt.mc.getWarnings() + } + return columnCount, nil + } + } + return 0, err +} + +// http://dev.mysql.com/doc/internals/en/com-stmt-send-long-data.html +func (stmt *mysqlStmt) writeCommandLongData(paramID int, arg []byte) error { + maxLen := stmt.mc.maxPacketAllowed - 1 + pktLen := maxLen + + // After the header (bytes 0-3) follows before the data: + // 1 byte command + // 4 bytes stmtID + // 2 bytes paramID + const dataOffset = 1 + 4 + 2 + + // Can not use the write buffer since + // a) the buffer is too small + // b) it is in use + data := make([]byte, 4+1+4+2+len(arg)) + + copy(data[4+dataOffset:], arg) + + for argLen := len(arg); argLen > 0; argLen -= pktLen - dataOffset { + if dataOffset+argLen < maxLen { + pktLen = dataOffset + argLen + } + + stmt.mc.sequence = 0 + // Add command byte [1 byte] + data[4] = comStmtSendLongData + + // Add stmtID [32 bit] + data[5] = byte(stmt.id) + data[6] = byte(stmt.id >> 8) + data[7] = byte(stmt.id >> 16) + data[8] = byte(stmt.id >> 24) + + // Add paramID [16 bit] + data[9] = byte(paramID) + data[10] = byte(paramID >> 8) + + // Send CMD packet + err := stmt.mc.writePacket(data[:4+pktLen]) + if err == nil { + data = data[pktLen-dataOffset:] + continue + } + return err + + } + + // Reset Packet Sequence + stmt.mc.sequence = 0 + return nil +} + +// Execute Prepared Statement +// http://dev.mysql.com/doc/internals/en/com-stmt-execute.html +func (stmt *mysqlStmt) writeExecutePacket(args []driver.Value) error { + if len(args) != stmt.paramCount { + return fmt.Errorf( + "Arguments count mismatch (Got: %d Has: %d)", + len(args), + stmt.paramCount, + ) + } + + const minPktLen = 4 + 1 + 4 + 1 + 4 + mc := stmt.mc + + // Reset packet-sequence + mc.sequence = 0 + + var data []byte + + if len(args) == 0 { + data = mc.buf.takeBuffer(minPktLen) + } else { + data = mc.buf.takeCompleteBuffer() + } + if data == nil { + // can not take the buffer. Something must be wrong with the connection + errLog.Print(ErrBusyBuffer) + return driver.ErrBadConn + } + + // command [1 byte] + data[4] = comStmtExecute + + // statement_id [4 bytes] + data[5] = byte(stmt.id) + data[6] = byte(stmt.id >> 8) + data[7] = byte(stmt.id >> 16) + data[8] = byte(stmt.id >> 24) + + // flags (0: CURSOR_TYPE_NO_CURSOR) [1 byte] + data[9] = 0x00 + + // iteration_count (uint32(1)) [4 bytes] + data[10] = 0x01 + data[11] = 0x00 + data[12] = 0x00 + data[13] = 0x00 + + if len(args) > 0 { + pos := minPktLen + + var nullMask []byte + if maskLen, typesLen := (len(args)+7)/8, 1+2*len(args); pos+maskLen+typesLen >= len(data) { + // buffer has to be extended but we don't know by how much so + // we depend on append after all data with known sizes fit. + // We stop at that because we deal with a lot of columns here + // which makes the required allocation size hard to guess. + tmp := make([]byte, pos+maskLen+typesLen) + copy(tmp[:pos], data[:pos]) + data = tmp + nullMask = data[pos : pos+maskLen] + pos += maskLen + } else { + nullMask = data[pos : pos+maskLen] + for i := 0; i < maskLen; i++ { + nullMask[i] = 0 + } + pos += maskLen + } + + // newParameterBoundFlag 1 [1 byte] + data[pos] = 0x01 + pos++ + + // type of each parameter [len(args)*2 bytes] + paramTypes := data[pos:] + pos += len(args) * 2 + + // value of each parameter [n bytes] + paramValues := data[pos:pos] + valuesCap := cap(paramValues) + + for i, arg := range args { + // build NULL-bitmap + if arg == nil { + nullMask[i/8] |= 1 << (uint(i) & 7) + paramTypes[i+i] = fieldTypeNULL + paramTypes[i+i+1] = 0x00 + continue + } + + // cache types and values + switch v := arg.(type) { + case int64: + paramTypes[i+i] = fieldTypeLongLong + paramTypes[i+i+1] = 0x00 + + if cap(paramValues)-len(paramValues)-8 >= 0 { + paramValues = paramValues[:len(paramValues)+8] + binary.LittleEndian.PutUint64( + paramValues[len(paramValues)-8:], + uint64(v), + ) + } else { + paramValues = append(paramValues, + uint64ToBytes(uint64(v))..., + ) + } + + case float64: + paramTypes[i+i] = fieldTypeDouble + paramTypes[i+i+1] = 0x00 + + if cap(paramValues)-len(paramValues)-8 >= 0 { + paramValues = paramValues[:len(paramValues)+8] + binary.LittleEndian.PutUint64( + paramValues[len(paramValues)-8:], + math.Float64bits(v), + ) + } else { + paramValues = append(paramValues, + uint64ToBytes(math.Float64bits(v))..., + ) + } + + case bool: + paramTypes[i+i] = fieldTypeTiny + paramTypes[i+i+1] = 0x00 + + if v { + paramValues = append(paramValues, 0x01) + } else { + paramValues = append(paramValues, 0x00) + } + + case []byte: + // Common case (non-nil value) first + if v != nil { + paramTypes[i+i] = fieldTypeString + paramTypes[i+i+1] = 0x00 + + if len(v) < mc.maxPacketAllowed-pos-len(paramValues)-(len(args)-(i+1))*64 { + paramValues = appendLengthEncodedInteger(paramValues, + uint64(len(v)), + ) + paramValues = append(paramValues, v...) + } else { + if err := stmt.writeCommandLongData(i, v); err != nil { + return err + } + } + continue + } + + // Handle []byte(nil) as a NULL value + nullMask[i/8] |= 1 << (uint(i) & 7) + paramTypes[i+i] = fieldTypeNULL + paramTypes[i+i+1] = 0x00 + + case string: + paramTypes[i+i] = fieldTypeString + paramTypes[i+i+1] = 0x00 + + if len(v) < mc.maxPacketAllowed-pos-len(paramValues)-(len(args)-(i+1))*64 { + paramValues = appendLengthEncodedInteger(paramValues, + uint64(len(v)), + ) + paramValues = append(paramValues, v...) + } else { + if err := stmt.writeCommandLongData(i, []byte(v)); err != nil { + return err + } + } + + case time.Time: + paramTypes[i+i] = fieldTypeString + paramTypes[i+i+1] = 0x00 + + var val []byte + if v.IsZero() { + val = []byte("0000-00-00") + } else { + val = []byte(v.In(mc.cfg.loc).Format(timeFormat)) + } + + paramValues = appendLengthEncodedInteger(paramValues, + uint64(len(val)), + ) + paramValues = append(paramValues, val...) + + default: + return fmt.Errorf("Can't convert type: %T", arg) + } + } + + // Check if param values exceeded the available buffer + // In that case we must build the data packet with the new values buffer + if valuesCap != cap(paramValues) { + data = append(data[:pos], paramValues...) + mc.buf.buf = data + } + + pos += len(paramValues) + data = data[:pos] + } + + return mc.writePacket(data) +} + +// http://dev.mysql.com/doc/internals/en/binary-protocol-resultset-row.html +func (rows *binaryRows) readRow(dest []driver.Value) error { + data, err := rows.mc.readPacket() + if err != nil { + return err + } + + // packet indicator [1 byte] + if data[0] != iOK { + rows.mc = nil + // EOF Packet + if data[0] == iEOF && len(data) == 5 { + return io.EOF + } + + // Error otherwise + return rows.mc.handleErrorPacket(data) + } + + // NULL-bitmap, [(column-count + 7 + 2) / 8 bytes] + pos := 1 + (len(dest)+7+2)>>3 + nullMask := data[1:pos] + + for i := range dest { + // Field is NULL + // (byte >> bit-pos) % 2 == 1 + if ((nullMask[(i+2)>>3] >> uint((i+2)&7)) & 1) == 1 { + dest[i] = nil + continue + } + + // Convert to byte-coded string + switch rows.columns[i].fieldType { + case fieldTypeNULL: + dest[i] = nil + continue + + // Numeric Types + case fieldTypeTiny: + if rows.columns[i].flags&flagUnsigned != 0 { + dest[i] = int64(data[pos]) + } else { + dest[i] = int64(int8(data[pos])) + } + pos++ + continue + + case fieldTypeShort, fieldTypeYear: + if rows.columns[i].flags&flagUnsigned != 0 { + dest[i] = int64(binary.LittleEndian.Uint16(data[pos : pos+2])) + } else { + dest[i] = int64(int16(binary.LittleEndian.Uint16(data[pos : pos+2]))) + } + pos += 2 + continue + + case fieldTypeInt24, fieldTypeLong: + if rows.columns[i].flags&flagUnsigned != 0 { + dest[i] = int64(binary.LittleEndian.Uint32(data[pos : pos+4])) + } else { + dest[i] = int64(int32(binary.LittleEndian.Uint32(data[pos : pos+4]))) + } + pos += 4 + continue + + case fieldTypeLongLong: + if rows.columns[i].flags&flagUnsigned != 0 { + val := binary.LittleEndian.Uint64(data[pos : pos+8]) + if val > math.MaxInt64 { + dest[i] = uint64ToString(val) + } else { + dest[i] = int64(val) + } + } else { + dest[i] = int64(binary.LittleEndian.Uint64(data[pos : pos+8])) + } + pos += 8 + continue + + case fieldTypeFloat: + dest[i] = float64(math.Float32frombits(binary.LittleEndian.Uint32(data[pos : pos+4]))) + pos += 4 + continue + + case fieldTypeDouble: + dest[i] = math.Float64frombits(binary.LittleEndian.Uint64(data[pos : pos+8])) + pos += 8 + continue + + // Length coded Binary Strings + case fieldTypeDecimal, fieldTypeNewDecimal, fieldTypeVarChar, + fieldTypeBit, fieldTypeEnum, fieldTypeSet, fieldTypeTinyBLOB, + fieldTypeMediumBLOB, fieldTypeLongBLOB, fieldTypeBLOB, + fieldTypeVarString, fieldTypeString, fieldTypeGeometry: + var isNull bool + var n int + dest[i], isNull, n, err = readLengthEncodedString(data[pos:]) + pos += n + if err == nil { + if !isNull { + continue + } else { + dest[i] = nil + continue + } + } + return err + + case + fieldTypeDate, fieldTypeNewDate, // Date YYYY-MM-DD + fieldTypeTime, // Time [-][H]HH:MM:SS[.fractal] + fieldTypeTimestamp, fieldTypeDateTime: // Timestamp YYYY-MM-DD HH:MM:SS[.fractal] + + num, isNull, n := readLengthEncodedInteger(data[pos:]) + pos += n + + switch { + case isNull: + dest[i] = nil + continue + case rows.columns[i].fieldType == fieldTypeTime: + // database/sql does not support an equivalent to TIME, return a string + var dstlen uint8 + switch decimals := rows.columns[i].decimals; decimals { + case 0x00, 0x1f: + dstlen = 8 + case 1, 2, 3, 4, 5, 6: + dstlen = 8 + 1 + decimals + default: + return fmt.Errorf( + "MySQL protocol error, illegal decimals value %d", + rows.columns[i].decimals, + ) + } + dest[i], err = formatBinaryDateTime(data[pos:pos+int(num)], dstlen, true) + case rows.mc.parseTime: + dest[i], err = parseBinaryDateTime(num, data[pos:], rows.mc.cfg.loc) + default: + var dstlen uint8 + if rows.columns[i].fieldType == fieldTypeDate { + dstlen = 10 + } else { + switch decimals := rows.columns[i].decimals; decimals { + case 0x00, 0x1f: + dstlen = 19 + case 1, 2, 3, 4, 5, 6: + dstlen = 19 + 1 + decimals + default: + return fmt.Errorf( + "MySQL protocol error, illegal decimals value %d", + rows.columns[i].decimals, + ) + } + } + dest[i], err = formatBinaryDateTime(data[pos:pos+int(num)], dstlen, false) + } + + if err == nil { + pos += int(num) + continue + } else { + return err + } + + // Please report if this happens! + default: + return fmt.Errorf("Unknown FieldType %d", rows.columns[i].fieldType) + } + } + + return nil +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/result.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/result.go new file mode 100644 index 0000000000000..c6438d0347db5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/result.go @@ -0,0 +1,22 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2012 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +type mysqlResult struct { + affectedRows int64 + insertId int64 +} + +func (res *mysqlResult) LastInsertId() (int64, error) { + return res.insertId, nil +} + +func (res *mysqlResult) RowsAffected() (int64, error) { + return res.affectedRows, nil +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/rows.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/rows.go new file mode 100644 index 0000000000000..ba606e146201b --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/rows.go @@ -0,0 +1,106 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2012 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +import ( + "database/sql/driver" + "io" +) + +type mysqlField struct { + tableName string + name string + flags fieldFlag + fieldType byte + decimals byte +} + +type mysqlRows struct { + mc *mysqlConn + columns []mysqlField +} + +type binaryRows struct { + mysqlRows +} + +type textRows struct { + mysqlRows +} + +type emptyRows struct{} + +func (rows *mysqlRows) Columns() []string { + columns := make([]string, len(rows.columns)) + if rows.mc.cfg.columnsWithAlias { + for i := range columns { + if tableName := rows.columns[i].tableName; len(tableName) > 0 { + columns[i] = tableName + "." + rows.columns[i].name + } else { + columns[i] = rows.columns[i].name + } + } + } else { + for i := range columns { + columns[i] = rows.columns[i].name + } + } + return columns +} + +func (rows *mysqlRows) Close() error { + mc := rows.mc + if mc == nil { + return nil + } + if mc.netConn == nil { + return ErrInvalidConn + } + + // Remove unread packets from stream + err := mc.readUntilEOF() + rows.mc = nil + return err +} + +func (rows *binaryRows) Next(dest []driver.Value) error { + if mc := rows.mc; mc != nil { + if mc.netConn == nil { + return ErrInvalidConn + } + + // Fetch next row from stream + return rows.readRow(dest) + } + return io.EOF +} + +func (rows *textRows) Next(dest []driver.Value) error { + if mc := rows.mc; mc != nil { + if mc.netConn == nil { + return ErrInvalidConn + } + + // Fetch next row from stream + return rows.readRow(dest) + } + return io.EOF +} + +func (rows emptyRows) Columns() []string { + return nil +} + +func (rows emptyRows) Close() error { + return nil +} + +func (rows emptyRows) Next(dest []driver.Value) error { + return io.EOF +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/statement.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/statement.go new file mode 100644 index 0000000000000..6e869b3405d1d --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/statement.go @@ -0,0 +1,150 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2012 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +import ( + "database/sql/driver" + "fmt" + "reflect" + "strconv" +) + +type mysqlStmt struct { + mc *mysqlConn + id uint32 + paramCount int + columns []mysqlField // cached from the first query +} + +func (stmt *mysqlStmt) Close() error { + if stmt.mc == nil || stmt.mc.netConn == nil { + errLog.Print(ErrInvalidConn) + return driver.ErrBadConn + } + + err := stmt.mc.writeCommandPacketUint32(comStmtClose, stmt.id) + stmt.mc = nil + return err +} + +func (stmt *mysqlStmt) NumInput() int { + return stmt.paramCount +} + +func (stmt *mysqlStmt) ColumnConverter(idx int) driver.ValueConverter { + return converter{} +} + +func (stmt *mysqlStmt) Exec(args []driver.Value) (driver.Result, error) { + if stmt.mc.netConn == nil { + errLog.Print(ErrInvalidConn) + return nil, driver.ErrBadConn + } + // Send command + err := stmt.writeExecutePacket(args) + if err != nil { + return nil, err + } + + mc := stmt.mc + + mc.affectedRows = 0 + mc.insertId = 0 + + // Read Result + resLen, err := mc.readResultSetHeaderPacket() + if err == nil { + if resLen > 0 { + // Columns + err = mc.readUntilEOF() + if err != nil { + return nil, err + } + + // Rows + err = mc.readUntilEOF() + } + if err == nil { + return &mysqlResult{ + affectedRows: int64(mc.affectedRows), + insertId: int64(mc.insertId), + }, nil + } + } + + return nil, err +} + +func (stmt *mysqlStmt) Query(args []driver.Value) (driver.Rows, error) { + if stmt.mc.netConn == nil { + errLog.Print(ErrInvalidConn) + return nil, driver.ErrBadConn + } + // Send command + err := stmt.writeExecutePacket(args) + if err != nil { + return nil, err + } + + mc := stmt.mc + + // Read Result + resLen, err := mc.readResultSetHeaderPacket() + if err != nil { + return nil, err + } + + rows := new(binaryRows) + rows.mc = mc + + if resLen > 0 { + // Columns + // If not cached, read them and cache them + if stmt.columns == nil { + rows.columns, err = mc.readColumns(resLen) + stmt.columns = rows.columns + } else { + rows.columns = stmt.columns + err = mc.readUntilEOF() + } + } + + return rows, err +} + +type converter struct{} + +func (c converter) ConvertValue(v interface{}) (driver.Value, error) { + if driver.IsValue(v) { + return v, nil + } + + rv := reflect.ValueOf(v) + switch rv.Kind() { + case reflect.Ptr: + // indirect pointers + if rv.IsNil() { + return nil, nil + } + return c.ConvertValue(rv.Elem().Interface()) + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return rv.Int(), nil + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32: + return int64(rv.Uint()), nil + case reflect.Uint64: + u64 := rv.Uint() + if u64 >= 1<<63 { + return strconv.FormatUint(u64, 10), nil + } + return int64(u64), nil + case reflect.Float32, reflect.Float64: + return rv.Float(), nil + } + return nil, fmt.Errorf("unsupported type %T, a %s", v, rv.Kind()) +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/transaction.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/transaction.go new file mode 100644 index 0000000000000..33c749b35c492 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/transaction.go @@ -0,0 +1,31 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2012 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +type mysqlTx struct { + mc *mysqlConn +} + +func (tx *mysqlTx) Commit() (err error) { + if tx.mc == nil || tx.mc.netConn == nil { + return ErrInvalidConn + } + err = tx.mc.exec("COMMIT") + tx.mc = nil + return +} + +func (tx *mysqlTx) Rollback() (err error) { + if tx.mc == nil || tx.mc.netConn == nil { + return ErrInvalidConn + } + err = tx.mc.exec("ROLLBACK") + tx.mc = nil + return +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/utils.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/utils.go new file mode 100644 index 0000000000000..6a26ad129ca96 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/utils.go @@ -0,0 +1,973 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2012 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +import ( + "crypto/sha1" + "crypto/tls" + "database/sql/driver" + "encoding/binary" + "errors" + "fmt" + "io" + "net" + "net/url" + "strings" + "time" +) + +var ( + tlsConfigRegister map[string]*tls.Config // Register for custom tls.Configs + + errInvalidDSNUnescaped = errors.New("Invalid DSN: Did you forget to escape a param value?") + errInvalidDSNAddr = errors.New("Invalid DSN: Network Address not terminated (missing closing brace)") + errInvalidDSNNoSlash = errors.New("Invalid DSN: Missing the slash separating the database name") + errInvalidDSNUnsafeCollation = errors.New("Invalid DSN: interpolateParams can be used with ascii, latin1, utf8 and utf8mb4 charset") +) + +func init() { + tlsConfigRegister = make(map[string]*tls.Config) +} + +// RegisterTLSConfig registers a custom tls.Config to be used with sql.Open. +// Use the key as a value in the DSN where tls=value. +// +// rootCertPool := x509.NewCertPool() +// pem, err := ioutil.ReadFile("/path/ca-cert.pem") +// if err != nil { +// log.Fatal(err) +// } +// if ok := rootCertPool.AppendCertsFromPEM(pem); !ok { +// log.Fatal("Failed to append PEM.") +// } +// clientCert := make([]tls.Certificate, 0, 1) +// certs, err := tls.LoadX509KeyPair("/path/client-cert.pem", "/path/client-key.pem") +// if err != nil { +// log.Fatal(err) +// } +// clientCert = append(clientCert, certs) +// mysql.RegisterTLSConfig("custom", &tls.Config{ +// RootCAs: rootCertPool, +// Certificates: clientCert, +// }) +// db, err := sql.Open("mysql", "user@tcp(localhost:3306)/test?tls=custom") +// +func RegisterTLSConfig(key string, config *tls.Config) error { + if _, isBool := readBool(key); isBool || strings.ToLower(key) == "skip-verify" { + return fmt.Errorf("Key '%s' is reserved", key) + } + + tlsConfigRegister[key] = config + return nil +} + +// DeregisterTLSConfig removes the tls.Config associated with key. +func DeregisterTLSConfig(key string) { + delete(tlsConfigRegister, key) +} + +// parseDSN parses the DSN string to a config +func parseDSN(dsn string) (cfg *config, err error) { + // New config with some default values + cfg = &config{ + loc: time.UTC, + collation: defaultCollation, + } + + // [user[:password]@][net[(addr)]]/dbname[?param1=value1¶mN=valueN] + // Find the last '/' (since the password or the net addr might contain a '/') + foundSlash := false + for i := len(dsn) - 1; i >= 0; i-- { + if dsn[i] == '/' { + foundSlash = true + var j, k int + + // left part is empty if i <= 0 + if i > 0 { + // [username[:password]@][protocol[(address)]] + // Find the last '@' in dsn[:i] + for j = i; j >= 0; j-- { + if dsn[j] == '@' { + // username[:password] + // Find the first ':' in dsn[:j] + for k = 0; k < j; k++ { + if dsn[k] == ':' { + cfg.passwd = dsn[k+1 : j] + break + } + } + cfg.user = dsn[:k] + + break + } + } + + // [protocol[(address)]] + // Find the first '(' in dsn[j+1:i] + for k = j + 1; k < i; k++ { + if dsn[k] == '(' { + // dsn[i-1] must be == ')' if an address is specified + if dsn[i-1] != ')' { + if strings.ContainsRune(dsn[k+1:i], ')') { + return nil, errInvalidDSNUnescaped + } + return nil, errInvalidDSNAddr + } + cfg.addr = dsn[k+1 : i-1] + break + } + } + cfg.net = dsn[j+1 : k] + } + + // dbname[?param1=value1&...¶mN=valueN] + // Find the first '?' in dsn[i+1:] + for j = i + 1; j < len(dsn); j++ { + if dsn[j] == '?' { + if err = parseDSNParams(cfg, dsn[j+1:]); err != nil { + return + } + break + } + } + cfg.dbname = dsn[i+1 : j] + + break + } + } + + if !foundSlash && len(dsn) > 0 { + return nil, errInvalidDSNNoSlash + } + + if cfg.interpolateParams && unsafeCollations[cfg.collation] { + return nil, errInvalidDSNUnsafeCollation + } + + // Set default network if empty + if cfg.net == "" { + cfg.net = "tcp" + } + + // Set default address if empty + if cfg.addr == "" { + switch cfg.net { + case "tcp": + cfg.addr = "127.0.0.1:3306" + case "unix": + cfg.addr = "/tmp/mysql.sock" + default: + return nil, errors.New("Default addr for network '" + cfg.net + "' unknown") + } + + } + + return +} + +// parseDSNParams parses the DSN "query string" +// Values must be url.QueryEscape'ed +func parseDSNParams(cfg *config, params string) (err error) { + for _, v := range strings.Split(params, "&") { + param := strings.SplitN(v, "=", 2) + if len(param) != 2 { + continue + } + + // cfg params + switch value := param[1]; param[0] { + + // Enable client side placeholder substitution + case "interpolateParams": + var isBool bool + cfg.interpolateParams, isBool = readBool(value) + if !isBool { + return fmt.Errorf("Invalid Bool value: %s", value) + } + + // Disable INFILE whitelist / enable all files + case "allowAllFiles": + var isBool bool + cfg.allowAllFiles, isBool = readBool(value) + if !isBool { + return fmt.Errorf("Invalid Bool value: %s", value) + } + + // Use cleartext authentication mode (MySQL 5.5.10+) + case "allowCleartextPasswords": + var isBool bool + cfg.allowCleartextPasswords, isBool = readBool(value) + if !isBool { + return fmt.Errorf("Invalid Bool value: %s", value) + } + + // Use old authentication mode (pre MySQL 4.1) + case "allowOldPasswords": + var isBool bool + cfg.allowOldPasswords, isBool = readBool(value) + if !isBool { + return fmt.Errorf("Invalid Bool value: %s", value) + } + + // Switch "rowsAffected" mode + case "clientFoundRows": + var isBool bool + cfg.clientFoundRows, isBool = readBool(value) + if !isBool { + return fmt.Errorf("Invalid Bool value: %s", value) + } + + // Collation + case "collation": + collation, ok := collations[value] + if !ok { + // Note possibility for false negatives: + // could be triggered although the collation is valid if the + // collations map does not contain entries the server supports. + err = errors.New("unknown collation") + return + } + cfg.collation = collation + break + + case "columnsWithAlias": + var isBool bool + cfg.columnsWithAlias, isBool = readBool(value) + if !isBool { + return fmt.Errorf("Invalid Bool value: %s", value) + } + + // Time Location + case "loc": + if value, err = url.QueryUnescape(value); err != nil { + return + } + cfg.loc, err = time.LoadLocation(value) + if err != nil { + return + } + + // Dial Timeout + case "timeout": + cfg.timeout, err = time.ParseDuration(value) + if err != nil { + return + } + + // TLS-Encryption + case "tls": + boolValue, isBool := readBool(value) + if isBool { + if boolValue { + cfg.tls = &tls.Config{} + } + } else { + if strings.ToLower(value) == "skip-verify" { + cfg.tls = &tls.Config{InsecureSkipVerify: true} + } else if tlsConfig, ok := tlsConfigRegister[value]; ok { + if len(tlsConfig.ServerName) == 0 && !tlsConfig.InsecureSkipVerify { + host, _, err := net.SplitHostPort(cfg.addr) + if err == nil { + tlsConfig.ServerName = host + } + } + + cfg.tls = tlsConfig + } else { + return fmt.Errorf("Invalid value / unknown config name: %s", value) + } + } + + default: + // lazy init + if cfg.params == nil { + cfg.params = make(map[string]string) + } + + if cfg.params[param[0]], err = url.QueryUnescape(value); err != nil { + return + } + } + } + + return +} + +// Returns the bool value of the input. +// The 2nd return value indicates if the input was a valid bool value +func readBool(input string) (value bool, valid bool) { + switch input { + case "1", "true", "TRUE", "True": + return true, true + case "0", "false", "FALSE", "False": + return false, true + } + + // Not a valid bool value + return +} + +/****************************************************************************** +* Authentication * +******************************************************************************/ + +// Encrypt password using 4.1+ method +func scramblePassword(scramble, password []byte) []byte { + if len(password) == 0 { + return nil + } + + // stage1Hash = SHA1(password) + crypt := sha1.New() + crypt.Write(password) + stage1 := crypt.Sum(nil) + + // scrambleHash = SHA1(scramble + SHA1(stage1Hash)) + // inner Hash + crypt.Reset() + crypt.Write(stage1) + hash := crypt.Sum(nil) + + // outer Hash + crypt.Reset() + crypt.Write(scramble) + crypt.Write(hash) + scramble = crypt.Sum(nil) + + // token = scrambleHash XOR stage1Hash + for i := range scramble { + scramble[i] ^= stage1[i] + } + return scramble +} + +// Encrypt password using pre 4.1 (old password) method +// https://github.com/atcurtis/mariadb/blob/master/mysys/my_rnd.c +type myRnd struct { + seed1, seed2 uint32 +} + +const myRndMaxVal = 0x3FFFFFFF + +// Pseudo random number generator +func newMyRnd(seed1, seed2 uint32) *myRnd { + return &myRnd{ + seed1: seed1 % myRndMaxVal, + seed2: seed2 % myRndMaxVal, + } +} + +// Tested to be equivalent to MariaDB's floating point variant +// http://play.golang.org/p/QHvhd4qved +// http://play.golang.org/p/RG0q4ElWDx +func (r *myRnd) NextByte() byte { + r.seed1 = (r.seed1*3 + r.seed2) % myRndMaxVal + r.seed2 = (r.seed1 + r.seed2 + 33) % myRndMaxVal + + return byte(uint64(r.seed1) * 31 / myRndMaxVal) +} + +// Generate binary hash from byte string using insecure pre 4.1 method +func pwHash(password []byte) (result [2]uint32) { + var add uint32 = 7 + var tmp uint32 + + result[0] = 1345345333 + result[1] = 0x12345671 + + for _, c := range password { + // skip spaces and tabs in password + if c == ' ' || c == '\t' { + continue + } + + tmp = uint32(c) + result[0] ^= (((result[0] & 63) + add) * tmp) + (result[0] << 8) + result[1] += (result[1] << 8) ^ result[0] + add += tmp + } + + // Remove sign bit (1<<31)-1) + result[0] &= 0x7FFFFFFF + result[1] &= 0x7FFFFFFF + + return +} + +// Encrypt password using insecure pre 4.1 method +func scrambleOldPassword(scramble, password []byte) []byte { + if len(password) == 0 { + return nil + } + + scramble = scramble[:8] + + hashPw := pwHash(password) + hashSc := pwHash(scramble) + + r := newMyRnd(hashPw[0]^hashSc[0], hashPw[1]^hashSc[1]) + + var out [8]byte + for i := range out { + out[i] = r.NextByte() + 64 + } + + mask := r.NextByte() + for i := range out { + out[i] ^= mask + } + + return out[:] +} + +/****************************************************************************** +* Time related utils * +******************************************************************************/ + +// NullTime represents a time.Time that may be NULL. +// NullTime implements the Scanner interface so +// it can be used as a scan destination: +// +// var nt NullTime +// err := db.QueryRow("SELECT time FROM foo WHERE id=?", id).Scan(&nt) +// ... +// if nt.Valid { +// // use nt.Time +// } else { +// // NULL value +// } +// +// This NullTime implementation is not driver-specific +type NullTime struct { + Time time.Time + Valid bool // Valid is true if Time is not NULL +} + +// Scan implements the Scanner interface. +// The value type must be time.Time or string / []byte (formatted time-string), +// otherwise Scan fails. +func (nt *NullTime) Scan(value interface{}) (err error) { + if value == nil { + nt.Time, nt.Valid = time.Time{}, false + return + } + + switch v := value.(type) { + case time.Time: + nt.Time, nt.Valid = v, true + return + case []byte: + nt.Time, err = parseDateTime(string(v), time.UTC) + nt.Valid = (err == nil) + return + case string: + nt.Time, err = parseDateTime(v, time.UTC) + nt.Valid = (err == nil) + return + } + + nt.Valid = false + return fmt.Errorf("Can't convert %T to time.Time", value) +} + +// Value implements the driver Valuer interface. +func (nt NullTime) Value() (driver.Value, error) { + if !nt.Valid { + return nil, nil + } + return nt.Time, nil +} + +func parseDateTime(str string, loc *time.Location) (t time.Time, err error) { + base := "0000-00-00 00:00:00.0000000" + switch len(str) { + case 10, 19, 21, 22, 23, 24, 25, 26: // up to "YYYY-MM-DD HH:MM:SS.MMMMMM" + if str == base[:len(str)] { + return + } + t, err = time.Parse(timeFormat[:len(str)], str) + default: + err = fmt.Errorf("Invalid Time-String: %s", str) + return + } + + // Adjust location + if err == nil && loc != time.UTC { + y, mo, d := t.Date() + h, mi, s := t.Clock() + t, err = time.Date(y, mo, d, h, mi, s, t.Nanosecond(), loc), nil + } + + return +} + +func parseBinaryDateTime(num uint64, data []byte, loc *time.Location) (driver.Value, error) { + switch num { + case 0: + return time.Time{}, nil + case 4: + return time.Date( + int(binary.LittleEndian.Uint16(data[:2])), // year + time.Month(data[2]), // month + int(data[3]), // day + 0, 0, 0, 0, + loc, + ), nil + case 7: + return time.Date( + int(binary.LittleEndian.Uint16(data[:2])), // year + time.Month(data[2]), // month + int(data[3]), // day + int(data[4]), // hour + int(data[5]), // minutes + int(data[6]), // seconds + 0, + loc, + ), nil + case 11: + return time.Date( + int(binary.LittleEndian.Uint16(data[:2])), // year + time.Month(data[2]), // month + int(data[3]), // day + int(data[4]), // hour + int(data[5]), // minutes + int(data[6]), // seconds + int(binary.LittleEndian.Uint32(data[7:11]))*1000, // nanoseconds + loc, + ), nil + } + return nil, fmt.Errorf("Invalid DATETIME-packet length %d", num) +} + +// zeroDateTime is used in formatBinaryDateTime to avoid an allocation +// if the DATE or DATETIME has the zero value. +// It must never be changed. +// The current behavior depends on database/sql copying the result. +var zeroDateTime = []byte("0000-00-00 00:00:00.000000") + +const digits01 = "0123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789" +const digits10 = "0000000000111111111122222222223333333333444444444455555555556666666666777777777788888888889999999999" + +func formatBinaryDateTime(src []byte, length uint8, justTime bool) (driver.Value, error) { + // length expects the deterministic length of the zero value, + // negative time and 100+ hours are automatically added if needed + if len(src) == 0 { + if justTime { + return zeroDateTime[11 : 11+length], nil + } + return zeroDateTime[:length], nil + } + var dst []byte // return value + var pt, p1, p2, p3 byte // current digit pair + var zOffs byte // offset of value in zeroDateTime + if justTime { + switch length { + case + 8, // time (can be up to 10 when negative and 100+ hours) + 10, 11, 12, 13, 14, 15: // time with fractional seconds + default: + return nil, fmt.Errorf("illegal TIME length %d", length) + } + switch len(src) { + case 8, 12: + default: + return nil, fmt.Errorf("Invalid TIME-packet length %d", len(src)) + } + // +2 to enable negative time and 100+ hours + dst = make([]byte, 0, length+2) + if src[0] == 1 { + dst = append(dst, '-') + } + if src[1] != 0 { + hour := uint16(src[1])*24 + uint16(src[5]) + pt = byte(hour / 100) + p1 = byte(hour - 100*uint16(pt)) + dst = append(dst, digits01[pt]) + } else { + p1 = src[5] + } + zOffs = 11 + src = src[6:] + } else { + switch length { + case 10, 19, 21, 22, 23, 24, 25, 26: + default: + t := "DATE" + if length > 10 { + t += "TIME" + } + return nil, fmt.Errorf("illegal %s length %d", t, length) + } + switch len(src) { + case 4, 7, 11: + default: + t := "DATE" + if length > 10 { + t += "TIME" + } + return nil, fmt.Errorf("illegal %s-packet length %d", t, len(src)) + } + dst = make([]byte, 0, length) + // start with the date + year := binary.LittleEndian.Uint16(src[:2]) + pt = byte(year / 100) + p1 = byte(year - 100*uint16(pt)) + p2, p3 = src[2], src[3] + dst = append(dst, + digits10[pt], digits01[pt], + digits10[p1], digits01[p1], '-', + digits10[p2], digits01[p2], '-', + digits10[p3], digits01[p3], + ) + if length == 10 { + return dst, nil + } + if len(src) == 4 { + return append(dst, zeroDateTime[10:length]...), nil + } + dst = append(dst, ' ') + p1 = src[4] // hour + src = src[5:] + } + // p1 is 2-digit hour, src is after hour + p2, p3 = src[0], src[1] + dst = append(dst, + digits10[p1], digits01[p1], ':', + digits10[p2], digits01[p2], ':', + digits10[p3], digits01[p3], + ) + if length <= byte(len(dst)) { + return dst, nil + } + src = src[2:] + if len(src) == 0 { + return append(dst, zeroDateTime[19:zOffs+length]...), nil + } + microsecs := binary.LittleEndian.Uint32(src[:4]) + p1 = byte(microsecs / 10000) + microsecs -= 10000 * uint32(p1) + p2 = byte(microsecs / 100) + microsecs -= 100 * uint32(p2) + p3 = byte(microsecs) + switch decimals := zOffs + length - 20; decimals { + default: + return append(dst, '.', + digits10[p1], digits01[p1], + digits10[p2], digits01[p2], + digits10[p3], digits01[p3], + ), nil + case 1: + return append(dst, '.', + digits10[p1], + ), nil + case 2: + return append(dst, '.', + digits10[p1], digits01[p1], + ), nil + case 3: + return append(dst, '.', + digits10[p1], digits01[p1], + digits10[p2], + ), nil + case 4: + return append(dst, '.', + digits10[p1], digits01[p1], + digits10[p2], digits01[p2], + ), nil + case 5: + return append(dst, '.', + digits10[p1], digits01[p1], + digits10[p2], digits01[p2], + digits10[p3], + ), nil + } +} + +/****************************************************************************** +* Convert from and to bytes * +******************************************************************************/ + +func uint64ToBytes(n uint64) []byte { + return []byte{ + byte(n), + byte(n >> 8), + byte(n >> 16), + byte(n >> 24), + byte(n >> 32), + byte(n >> 40), + byte(n >> 48), + byte(n >> 56), + } +} + +func uint64ToString(n uint64) []byte { + var a [20]byte + i := 20 + + // U+0030 = 0 + // ... + // U+0039 = 9 + + var q uint64 + for n >= 10 { + i-- + q = n / 10 + a[i] = uint8(n-q*10) + 0x30 + n = q + } + + i-- + a[i] = uint8(n) + 0x30 + + return a[i:] +} + +// treats string value as unsigned integer representation +func stringToInt(b []byte) int { + val := 0 + for i := range b { + val *= 10 + val += int(b[i] - 0x30) + } + return val +} + +// returns the string read as a bytes slice, wheter the value is NULL, +// the number of bytes read and an error, in case the string is longer than +// the input slice +func readLengthEncodedString(b []byte) ([]byte, bool, int, error) { + // Get length + num, isNull, n := readLengthEncodedInteger(b) + if num < 1 { + return b[n:n], isNull, n, nil + } + + n += int(num) + + // Check data length + if len(b) >= n { + return b[n-int(num) : n], false, n, nil + } + return nil, false, n, io.EOF +} + +// returns the number of bytes skipped and an error, in case the string is +// longer than the input slice +func skipLengthEncodedString(b []byte) (int, error) { + // Get length + num, _, n := readLengthEncodedInteger(b) + if num < 1 { + return n, nil + } + + n += int(num) + + // Check data length + if len(b) >= n { + return n, nil + } + return n, io.EOF +} + +// returns the number read, whether the value is NULL and the number of bytes read +func readLengthEncodedInteger(b []byte) (uint64, bool, int) { + // See issue #349 + if len(b) == 0 { + return 0, true, 1 + } + switch b[0] { + + // 251: NULL + case 0xfb: + return 0, true, 1 + + // 252: value of following 2 + case 0xfc: + return uint64(b[1]) | uint64(b[2])<<8, false, 3 + + // 253: value of following 3 + case 0xfd: + return uint64(b[1]) | uint64(b[2])<<8 | uint64(b[3])<<16, false, 4 + + // 254: value of following 8 + case 0xfe: + return uint64(b[1]) | uint64(b[2])<<8 | uint64(b[3])<<16 | + uint64(b[4])<<24 | uint64(b[5])<<32 | uint64(b[6])<<40 | + uint64(b[7])<<48 | uint64(b[8])<<56, + false, 9 + } + + // 0-250: value of first byte + return uint64(b[0]), false, 1 +} + +// encodes a uint64 value and appends it to the given bytes slice +func appendLengthEncodedInteger(b []byte, n uint64) []byte { + switch { + case n <= 250: + return append(b, byte(n)) + + case n <= 0xffff: + return append(b, 0xfc, byte(n), byte(n>>8)) + + case n <= 0xffffff: + return append(b, 0xfd, byte(n), byte(n>>8), byte(n>>16)) + } + return append(b, 0xfe, byte(n), byte(n>>8), byte(n>>16), byte(n>>24), + byte(n>>32), byte(n>>40), byte(n>>48), byte(n>>56)) +} + +// reserveBuffer checks cap(buf) and expand buffer to len(buf) + appendSize. +// If cap(buf) is not enough, reallocate new buffer. +func reserveBuffer(buf []byte, appendSize int) []byte { + newSize := len(buf) + appendSize + if cap(buf) < newSize { + // Grow buffer exponentially + newBuf := make([]byte, len(buf)*2+appendSize) + copy(newBuf, buf) + buf = newBuf + } + return buf[:newSize] +} + +// escapeBytesBackslash escapes []byte with backslashes (\) +// This escapes the contents of a string (provided as []byte) by adding backslashes before special +// characters, and turning others into specific escape sequences, such as +// turning newlines into \n and null bytes into \0. +// https://github.com/mysql/mysql-server/blob/mysql-5.7.5/mysys/charset.c#L823-L932 +func escapeBytesBackslash(buf, v []byte) []byte { + pos := len(buf) + buf = reserveBuffer(buf, len(v)*2) + + for _, c := range v { + switch c { + case '\x00': + buf[pos] = '\\' + buf[pos+1] = '0' + pos += 2 + case '\n': + buf[pos] = '\\' + buf[pos+1] = 'n' + pos += 2 + case '\r': + buf[pos] = '\\' + buf[pos+1] = 'r' + pos += 2 + case '\x1a': + buf[pos] = '\\' + buf[pos+1] = 'Z' + pos += 2 + case '\'': + buf[pos] = '\\' + buf[pos+1] = '\'' + pos += 2 + case '"': + buf[pos] = '\\' + buf[pos+1] = '"' + pos += 2 + case '\\': + buf[pos] = '\\' + buf[pos+1] = '\\' + pos += 2 + default: + buf[pos] = c + pos += 1 + } + } + + return buf[:pos] +} + +// escapeStringBackslash is similar to escapeBytesBackslash but for string. +func escapeStringBackslash(buf []byte, v string) []byte { + pos := len(buf) + buf = reserveBuffer(buf, len(v)*2) + + for i := 0; i < len(v); i++ { + c := v[i] + switch c { + case '\x00': + buf[pos] = '\\' + buf[pos+1] = '0' + pos += 2 + case '\n': + buf[pos] = '\\' + buf[pos+1] = 'n' + pos += 2 + case '\r': + buf[pos] = '\\' + buf[pos+1] = 'r' + pos += 2 + case '\x1a': + buf[pos] = '\\' + buf[pos+1] = 'Z' + pos += 2 + case '\'': + buf[pos] = '\\' + buf[pos+1] = '\'' + pos += 2 + case '"': + buf[pos] = '\\' + buf[pos+1] = '"' + pos += 2 + case '\\': + buf[pos] = '\\' + buf[pos+1] = '\\' + pos += 2 + default: + buf[pos] = c + pos += 1 + } + } + + return buf[:pos] +} + +// escapeBytesQuotes escapes apostrophes in []byte by doubling them up. +// This escapes the contents of a string by doubling up any apostrophes that +// it contains. This is used when the NO_BACKSLASH_ESCAPES SQL_MODE is in +// effect on the server. +// https://github.com/mysql/mysql-server/blob/mysql-5.7.5/mysys/charset.c#L963-L1038 +func escapeBytesQuotes(buf, v []byte) []byte { + pos := len(buf) + buf = reserveBuffer(buf, len(v)*2) + + for _, c := range v { + if c == '\'' { + buf[pos] = '\'' + buf[pos+1] = '\'' + pos += 2 + } else { + buf[pos] = c + pos++ + } + } + + return buf[:pos] +} + +// escapeStringQuotes is similar to escapeBytesQuotes but for string. +func escapeStringQuotes(buf []byte, v string) []byte { + pos := len(buf) + buf = reserveBuffer(buf, len(v)*2) + + for i := 0; i < len(v); i++ { + c := v[i] + if c == '\'' { + buf[pos] = '\'' + buf[pos+1] = '\'' + pos += 2 + } else { + buf[pos] = c + pos++ + } + } + + return buf[:pos] +} diff --git a/Godeps/_workspace/src/github.com/go-sql-driver/mysql/utils_test.go b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/utils_test.go new file mode 100644 index 0000000000000..79fbdd1eb35f9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/go-sql-driver/mysql/utils_test.go @@ -0,0 +1,346 @@ +// Go MySQL Driver - A MySQL-Driver for Go's database/sql package +// +// Copyright 2013 The Go-MySQL-Driver Authors. All rights reserved. +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this file, +// You can obtain one at http://mozilla.org/MPL/2.0/. + +package mysql + +import ( + "bytes" + "crypto/tls" + "encoding/binary" + "fmt" + "testing" + "time" +) + +var testDSNs = []struct { + in string + out string + loc *time.Location +}{ + {"username:password@protocol(address)/dbname?param=value", "&{user:username passwd:password net:protocol addr:address dbname:dbname params:map[param:value] loc:%p tls: timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false allowCleartextPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC}, + {"username:password@protocol(address)/dbname?param=value&columnsWithAlias=true", "&{user:username passwd:password net:protocol addr:address dbname:dbname params:map[param:value] loc:%p tls: timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false allowCleartextPasswords:false clientFoundRows:false columnsWithAlias:true interpolateParams:false}", time.UTC}, + {"user@unix(/path/to/socket)/dbname?charset=utf8", "&{user:user passwd: net:unix addr:/path/to/socket dbname:dbname params:map[charset:utf8] loc:%p tls: timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false allowCleartextPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC}, + {"user:password@tcp(localhost:5555)/dbname?charset=utf8&tls=true", "&{user:user passwd:password net:tcp addr:localhost:5555 dbname:dbname params:map[charset:utf8] loc:%p tls: timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false allowCleartextPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC}, + {"user:password@tcp(localhost:5555)/dbname?charset=utf8mb4,utf8&tls=skip-verify", "&{user:user passwd:password net:tcp addr:localhost:5555 dbname:dbname params:map[charset:utf8mb4,utf8] loc:%p tls: timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false allowCleartextPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC}, + {"user:password@/dbname?loc=UTC&timeout=30s&allowAllFiles=1&clientFoundRows=true&allowOldPasswords=TRUE&collation=utf8mb4_unicode_ci", "&{user:user passwd:password net:tcp addr:127.0.0.1:3306 dbname:dbname params:map[] loc:%p tls: timeout:30000000000 collation:224 allowAllFiles:true allowOldPasswords:true allowCleartextPasswords:false clientFoundRows:true columnsWithAlias:false interpolateParams:false}", time.UTC}, + {"user:p@ss(word)@tcp([de:ad:be:ef::ca:fe]:80)/dbname?loc=Local", "&{user:user passwd:p@ss(word) net:tcp addr:[de:ad:be:ef::ca:fe]:80 dbname:dbname params:map[] loc:%p tls: timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false allowCleartextPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.Local}, + {"/dbname", "&{user: passwd: net:tcp addr:127.0.0.1:3306 dbname:dbname params:map[] loc:%p tls: timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false allowCleartextPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC}, + {"@/", "&{user: passwd: net:tcp addr:127.0.0.1:3306 dbname: params:map[] loc:%p tls: timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false allowCleartextPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC}, + {"/", "&{user: passwd: net:tcp addr:127.0.0.1:3306 dbname: params:map[] loc:%p tls: timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false allowCleartextPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC}, + {"", "&{user: passwd: net:tcp addr:127.0.0.1:3306 dbname: params:map[] loc:%p tls: timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false allowCleartextPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC}, + {"user:p@/ssword@/", "&{user:user passwd:p@/ssword net:tcp addr:127.0.0.1:3306 dbname: params:map[] loc:%p tls: timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false allowCleartextPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC}, + {"unix/?arg=%2Fsome%2Fpath.ext", "&{user: passwd: net:unix addr:/tmp/mysql.sock dbname: params:map[arg:/some/path.ext] loc:%p tls: timeout:0 collation:33 allowAllFiles:false allowOldPasswords:false allowCleartextPasswords:false clientFoundRows:false columnsWithAlias:false interpolateParams:false}", time.UTC}, +} + +func TestDSNParser(t *testing.T) { + var cfg *config + var err error + var res string + + for i, tst := range testDSNs { + cfg, err = parseDSN(tst.in) + if err != nil { + t.Error(err.Error()) + } + + // pointer not static + cfg.tls = nil + + res = fmt.Sprintf("%+v", cfg) + if res != fmt.Sprintf(tst.out, tst.loc) { + t.Errorf("%d. parseDSN(%q) => %q, want %q", i, tst.in, res, fmt.Sprintf(tst.out, tst.loc)) + } + } +} + +func TestDSNParserInvalid(t *testing.T) { + var invalidDSNs = []string{ + "@net(addr/", // no closing brace + "@tcp(/", // no closing brace + "tcp(/", // no closing brace + "(/", // no closing brace + "net(addr)//", // unescaped + "user:pass@tcp(1.2.3.4:3306)", // no trailing slash + //"/dbname?arg=/some/unescaped/path", + } + + for i, tst := range invalidDSNs { + if _, err := parseDSN(tst); err == nil { + t.Errorf("invalid DSN #%d. (%s) didn't error!", i, tst) + } + } +} + +func TestDSNWithCustomTLS(t *testing.T) { + baseDSN := "user:password@tcp(localhost:5555)/dbname?tls=" + tlsCfg := tls.Config{} + + RegisterTLSConfig("utils_test", &tlsCfg) + + // Custom TLS is missing + tst := baseDSN + "invalid_tls" + cfg, err := parseDSN(tst) + if err == nil { + t.Errorf("Invalid custom TLS in DSN (%s) but did not error. Got config: %#v", tst, cfg) + } + + tst = baseDSN + "utils_test" + + // Custom TLS with a server name + name := "foohost" + tlsCfg.ServerName = name + cfg, err = parseDSN(tst) + + if err != nil { + t.Error(err.Error()) + } else if cfg.tls.ServerName != name { + t.Errorf("Did not get the correct TLS ServerName (%s) parsing DSN (%s).", name, tst) + } + + // Custom TLS without a server name + name = "localhost" + tlsCfg.ServerName = "" + cfg, err = parseDSN(tst) + + if err != nil { + t.Error(err.Error()) + } else if cfg.tls.ServerName != name { + t.Errorf("Did not get the correct ServerName (%s) parsing DSN (%s).", name, tst) + } + + DeregisterTLSConfig("utils_test") +} + +func TestDSNUnsafeCollation(t *testing.T) { + _, err := parseDSN("/dbname?collation=gbk_chinese_ci&interpolateParams=true") + if err != errInvalidDSNUnsafeCollation { + t.Error("Expected %v, Got %v", errInvalidDSNUnsafeCollation, err) + } + + _, err = parseDSN("/dbname?collation=gbk_chinese_ci&interpolateParams=false") + if err != nil { + t.Error("Expected %v, Got %v", nil, err) + } + + _, err = parseDSN("/dbname?collation=gbk_chinese_ci") + if err != nil { + t.Error("Expected %v, Got %v", nil, err) + } + + _, err = parseDSN("/dbname?collation=ascii_bin&interpolateParams=true") + if err != nil { + t.Error("Expected %v, Got %v", nil, err) + } + + _, err = parseDSN("/dbname?collation=latin1_german1_ci&interpolateParams=true") + if err != nil { + t.Error("Expected %v, Got %v", nil, err) + } + + _, err = parseDSN("/dbname?collation=utf8_general_ci&interpolateParams=true") + if err != nil { + t.Error("Expected %v, Got %v", nil, err) + } + + _, err = parseDSN("/dbname?collation=utf8mb4_general_ci&interpolateParams=true") + if err != nil { + t.Error("Expected %v, Got %v", nil, err) + } +} + +func BenchmarkParseDSN(b *testing.B) { + b.ReportAllocs() + + for i := 0; i < b.N; i++ { + for _, tst := range testDSNs { + if _, err := parseDSN(tst.in); err != nil { + b.Error(err.Error()) + } + } + } +} + +func TestScanNullTime(t *testing.T) { + var scanTests = []struct { + in interface{} + error bool + valid bool + time time.Time + }{ + {tDate, false, true, tDate}, + {sDate, false, true, tDate}, + {[]byte(sDate), false, true, tDate}, + {tDateTime, false, true, tDateTime}, + {sDateTime, false, true, tDateTime}, + {[]byte(sDateTime), false, true, tDateTime}, + {tDate0, false, true, tDate0}, + {sDate0, false, true, tDate0}, + {[]byte(sDate0), false, true, tDate0}, + {sDateTime0, false, true, tDate0}, + {[]byte(sDateTime0), false, true, tDate0}, + {"", true, false, tDate0}, + {"1234", true, false, tDate0}, + {0, true, false, tDate0}, + } + + var nt = NullTime{} + var err error + + for _, tst := range scanTests { + err = nt.Scan(tst.in) + if (err != nil) != tst.error { + t.Errorf("%v: expected error status %t, got %t", tst.in, tst.error, (err != nil)) + } + if nt.Valid != tst.valid { + t.Errorf("%v: expected valid status %t, got %t", tst.in, tst.valid, nt.Valid) + } + if nt.Time != tst.time { + t.Errorf("%v: expected time %v, got %v", tst.in, tst.time, nt.Time) + } + } +} + +func TestLengthEncodedInteger(t *testing.T) { + var integerTests = []struct { + num uint64 + encoded []byte + }{ + {0x0000000000000000, []byte{0x00}}, + {0x0000000000000012, []byte{0x12}}, + {0x00000000000000fa, []byte{0xfa}}, + {0x0000000000000100, []byte{0xfc, 0x00, 0x01}}, + {0x0000000000001234, []byte{0xfc, 0x34, 0x12}}, + {0x000000000000ffff, []byte{0xfc, 0xff, 0xff}}, + {0x0000000000010000, []byte{0xfd, 0x00, 0x00, 0x01}}, + {0x0000000000123456, []byte{0xfd, 0x56, 0x34, 0x12}}, + {0x0000000000ffffff, []byte{0xfd, 0xff, 0xff, 0xff}}, + {0x0000000001000000, []byte{0xfe, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00}}, + {0x123456789abcdef0, []byte{0xfe, 0xf0, 0xde, 0xbc, 0x9a, 0x78, 0x56, 0x34, 0x12}}, + {0xffffffffffffffff, []byte{0xfe, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}}, + } + + for _, tst := range integerTests { + num, isNull, numLen := readLengthEncodedInteger(tst.encoded) + if isNull { + t.Errorf("%x: expected %d, got NULL", tst.encoded, tst.num) + } + if num != tst.num { + t.Errorf("%x: expected %d, got %d", tst.encoded, tst.num, num) + } + if numLen != len(tst.encoded) { + t.Errorf("%x: expected size %d, got %d", tst.encoded, len(tst.encoded), numLen) + } + encoded := appendLengthEncodedInteger(nil, num) + if !bytes.Equal(encoded, tst.encoded) { + t.Errorf("%v: expected %x, got %x", num, tst.encoded, encoded) + } + } +} + +func TestOldPass(t *testing.T) { + scramble := []byte{9, 8, 7, 6, 5, 4, 3, 2} + vectors := []struct { + pass string + out string + }{ + {" pass", "47575c5a435b4251"}, + {"pass ", "47575c5a435b4251"}, + {"123\t456", "575c47505b5b5559"}, + {"C0mpl!ca ted#PASS123", "5d5d554849584a45"}, + } + for _, tuple := range vectors { + ours := scrambleOldPassword(scramble, []byte(tuple.pass)) + if tuple.out != fmt.Sprintf("%x", ours) { + t.Errorf("Failed old password %q", tuple.pass) + } + } +} + +func TestFormatBinaryDateTime(t *testing.T) { + rawDate := [11]byte{} + binary.LittleEndian.PutUint16(rawDate[:2], 1978) // years + rawDate[2] = 12 // months + rawDate[3] = 30 // days + rawDate[4] = 15 // hours + rawDate[5] = 46 // minutes + rawDate[6] = 23 // seconds + binary.LittleEndian.PutUint32(rawDate[7:], 987654) // microseconds + expect := func(expected string, inlen, outlen uint8) { + actual, _ := formatBinaryDateTime(rawDate[:inlen], outlen, false) + bytes, ok := actual.([]byte) + if !ok { + t.Errorf("formatBinaryDateTime must return []byte, was %T", actual) + } + if string(bytes) != expected { + t.Errorf( + "expected %q, got %q for length in %d, out %d", + bytes, actual, inlen, outlen, + ) + } + } + expect("0000-00-00", 0, 10) + expect("0000-00-00 00:00:00", 0, 19) + expect("1978-12-30", 4, 10) + expect("1978-12-30 15:46:23", 7, 19) + expect("1978-12-30 15:46:23.987654", 11, 26) +} + +func TestEscapeBackslash(t *testing.T) { + expect := func(expected, value string) { + actual := string(escapeBytesBackslash([]byte{}, []byte(value))) + if actual != expected { + t.Errorf( + "expected %s, got %s", + expected, actual, + ) + } + + actual = string(escapeStringBackslash([]byte{}, value)) + if actual != expected { + t.Errorf( + "expected %s, got %s", + expected, actual, + ) + } + } + + expect("foo\\0bar", "foo\x00bar") + expect("foo\\nbar", "foo\nbar") + expect("foo\\rbar", "foo\rbar") + expect("foo\\Zbar", "foo\x1abar") + expect("foo\\\"bar", "foo\"bar") + expect("foo\\\\bar", "foo\\bar") + expect("foo\\'bar", "foo'bar") +} + +func TestEscapeQuotes(t *testing.T) { + expect := func(expected, value string) { + actual := string(escapeBytesQuotes([]byte{}, []byte(value))) + if actual != expected { + t.Errorf( + "expected %s, got %s", + expected, actual, + ) + } + + actual = string(escapeStringQuotes([]byte{}, value)) + if actual != expected { + t.Errorf( + "expected %s, got %s", + expected, actual, + ) + } + } + + expect("foo\x00bar", "foo\x00bar") // not affected + expect("foo\nbar", "foo\nbar") // not affected + expect("foo\rbar", "foo\rbar") // not affected + expect("foo\x1abar", "foo\x1abar") // not affected + expect("foo''bar", "foo'bar") // affected + expect("foo\"bar", "foo\"bar") // not affected +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/Makefile b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/Makefile new file mode 100644 index 0000000000000..23a6b17344088 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/Makefile @@ -0,0 +1,43 @@ +# Go support for Protocol Buffers - Google's data interchange format +# +# Copyright 2010 The Go Authors. All rights reserved. +# https://github.com/golang/protobuf +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are +# met: +# +# * Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +# * Redistributions in binary form must reproduce the above +# copyright notice, this list of conditions and the following disclaimer +# in the documentation and/or other materials provided with the +# distribution. +# * Neither the name of Google Inc. nor the names of its +# contributors may be used to endorse or promote products derived from +# this software without specific prior written permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +install: + go install + +test: install generate-test-pbs + go test + + +generate-test-pbs: + make install + make -C testdata + protoc-min-version --version="3.0.0" --proto_path=.:../../../../ --gogo_out=. proto3_proto/proto3.proto + make diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/all_test.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/all_test.go new file mode 100644 index 0000000000000..88c506cf0afc8 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/all_test.go @@ -0,0 +1,2104 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "bytes" + "encoding/json" + "errors" + "fmt" + "math" + "math/rand" + "reflect" + "runtime/debug" + "strings" + "testing" + "time" + + . "github.com/gogo/protobuf/proto" + . "github.com/gogo/protobuf/proto/testdata" +) + +var globalO *Buffer + +func old() *Buffer { + if globalO == nil { + globalO = NewBuffer(nil) + } + globalO.Reset() + return globalO +} + +func equalbytes(b1, b2 []byte, t *testing.T) { + if len(b1) != len(b2) { + t.Errorf("wrong lengths: 2*%d != %d", len(b1), len(b2)) + return + } + for i := 0; i < len(b1); i++ { + if b1[i] != b2[i] { + t.Errorf("bad byte[%d]:%x %x: %s %s", i, b1[i], b2[i], b1, b2) + } + } +} + +func initGoTestField() *GoTestField { + f := new(GoTestField) + f.Label = String("label") + f.Type = String("type") + return f +} + +// These are all structurally equivalent but the tag numbers differ. +// (It's remarkable that required, optional, and repeated all have +// 8 letters.) +func initGoTest_RequiredGroup() *GoTest_RequiredGroup { + return &GoTest_RequiredGroup{ + RequiredField: String("required"), + } +} + +func initGoTest_OptionalGroup() *GoTest_OptionalGroup { + return &GoTest_OptionalGroup{ + RequiredField: String("optional"), + } +} + +func initGoTest_RepeatedGroup() *GoTest_RepeatedGroup { + return &GoTest_RepeatedGroup{ + RequiredField: String("repeated"), + } +} + +func initGoTest(setdefaults bool) *GoTest { + pb := new(GoTest) + if setdefaults { + pb.F_BoolDefaulted = Bool(Default_GoTest_F_BoolDefaulted) + pb.F_Int32Defaulted = Int32(Default_GoTest_F_Int32Defaulted) + pb.F_Int64Defaulted = Int64(Default_GoTest_F_Int64Defaulted) + pb.F_Fixed32Defaulted = Uint32(Default_GoTest_F_Fixed32Defaulted) + pb.F_Fixed64Defaulted = Uint64(Default_GoTest_F_Fixed64Defaulted) + pb.F_Uint32Defaulted = Uint32(Default_GoTest_F_Uint32Defaulted) + pb.F_Uint64Defaulted = Uint64(Default_GoTest_F_Uint64Defaulted) + pb.F_FloatDefaulted = Float32(Default_GoTest_F_FloatDefaulted) + pb.F_DoubleDefaulted = Float64(Default_GoTest_F_DoubleDefaulted) + pb.F_StringDefaulted = String(Default_GoTest_F_StringDefaulted) + pb.F_BytesDefaulted = Default_GoTest_F_BytesDefaulted + pb.F_Sint32Defaulted = Int32(Default_GoTest_F_Sint32Defaulted) + pb.F_Sint64Defaulted = Int64(Default_GoTest_F_Sint64Defaulted) + } + + pb.Kind = GoTest_TIME.Enum() + pb.RequiredField = initGoTestField() + pb.F_BoolRequired = Bool(true) + pb.F_Int32Required = Int32(3) + pb.F_Int64Required = Int64(6) + pb.F_Fixed32Required = Uint32(32) + pb.F_Fixed64Required = Uint64(64) + pb.F_Uint32Required = Uint32(3232) + pb.F_Uint64Required = Uint64(6464) + pb.F_FloatRequired = Float32(3232) + pb.F_DoubleRequired = Float64(6464) + pb.F_StringRequired = String("string") + pb.F_BytesRequired = []byte("bytes") + pb.F_Sint32Required = Int32(-32) + pb.F_Sint64Required = Int64(-64) + pb.Requiredgroup = initGoTest_RequiredGroup() + + return pb +} + +func fail(msg string, b *bytes.Buffer, s string, t *testing.T) { + data := b.Bytes() + ld := len(data) + ls := len(s) / 2 + + fmt.Printf("fail %s ld=%d ls=%d\n", msg, ld, ls) + + // find the interesting spot - n + n := ls + if ld < ls { + n = ld + } + j := 0 + for i := 0; i < n; i++ { + bs := hex(s[j])*16 + hex(s[j+1]) + j += 2 + if data[i] == bs { + continue + } + n = i + break + } + l := n - 10 + if l < 0 { + l = 0 + } + h := n + 10 + + // find the interesting spot - n + fmt.Printf("is[%d]:", l) + for i := l; i < h; i++ { + if i >= ld { + fmt.Printf(" --") + continue + } + fmt.Printf(" %.2x", data[i]) + } + fmt.Printf("\n") + + fmt.Printf("sb[%d]:", l) + for i := l; i < h; i++ { + if i >= ls { + fmt.Printf(" --") + continue + } + bs := hex(s[j])*16 + hex(s[j+1]) + j += 2 + fmt.Printf(" %.2x", bs) + } + fmt.Printf("\n") + + t.Fail() + + // t.Errorf("%s: \ngood: %s\nbad: %x", msg, s, b.Bytes()) + // Print the output in a partially-decoded format; can + // be helpful when updating the test. It produces the output + // that is pasted, with minor edits, into the argument to verify(). + // data := b.Bytes() + // nesting := 0 + // for b.Len() > 0 { + // start := len(data) - b.Len() + // var u uint64 + // u, err := DecodeVarint(b) + // if err != nil { + // fmt.Printf("decode error on varint:", err) + // return + // } + // wire := u & 0x7 + // tag := u >> 3 + // switch wire { + // case WireVarint: + // v, err := DecodeVarint(b) + // if err != nil { + // fmt.Printf("decode error on varint:", err) + // return + // } + // fmt.Printf("\t\t\"%x\" // field %d, encoding %d, value %d\n", + // data[start:len(data)-b.Len()], tag, wire, v) + // case WireFixed32: + // v, err := DecodeFixed32(b) + // if err != nil { + // fmt.Printf("decode error on fixed32:", err) + // return + // } + // fmt.Printf("\t\t\"%x\" // field %d, encoding %d, value %d\n", + // data[start:len(data)-b.Len()], tag, wire, v) + // case WireFixed64: + // v, err := DecodeFixed64(b) + // if err != nil { + // fmt.Printf("decode error on fixed64:", err) + // return + // } + // fmt.Printf("\t\t\"%x\" // field %d, encoding %d, value %d\n", + // data[start:len(data)-b.Len()], tag, wire, v) + // case WireBytes: + // nb, err := DecodeVarint(b) + // if err != nil { + // fmt.Printf("decode error on bytes:", err) + // return + // } + // after_tag := len(data) - b.Len() + // str := make([]byte, nb) + // _, err = b.Read(str) + // if err != nil { + // fmt.Printf("decode error on bytes:", err) + // return + // } + // fmt.Printf("\t\t\"%x\" \"%x\" // field %d, encoding %d (FIELD)\n", + // data[start:after_tag], str, tag, wire) + // case WireStartGroup: + // nesting++ + // fmt.Printf("\t\t\"%x\"\t\t// start group field %d level %d\n", + // data[start:len(data)-b.Len()], tag, nesting) + // case WireEndGroup: + // fmt.Printf("\t\t\"%x\"\t\t// end group field %d level %d\n", + // data[start:len(data)-b.Len()], tag, nesting) + // nesting-- + // default: + // fmt.Printf("unrecognized wire type %d\n", wire) + // return + // } + // } +} + +func hex(c uint8) uint8 { + if '0' <= c && c <= '9' { + return c - '0' + } + if 'a' <= c && c <= 'f' { + return 10 + c - 'a' + } + if 'A' <= c && c <= 'F' { + return 10 + c - 'A' + } + return 0 +} + +func equal(b []byte, s string, t *testing.T) bool { + if 2*len(b) != len(s) { + // fail(fmt.Sprintf("wrong lengths: 2*%d != %d", len(b), len(s)), b, s, t) + fmt.Printf("wrong lengths: 2*%d != %d\n", len(b), len(s)) + return false + } + for i, j := 0, 0; i < len(b); i, j = i+1, j+2 { + x := hex(s[j])*16 + hex(s[j+1]) + if b[i] != x { + // fail(fmt.Sprintf("bad byte[%d]:%x %x", i, b[i], x), b, s, t) + fmt.Printf("bad byte[%d]:%x %x", i, b[i], x) + return false + } + } + return true +} + +func overify(t *testing.T, pb *GoTest, expected string) { + o := old() + err := o.Marshal(pb) + if err != nil { + fmt.Printf("overify marshal-1 err = %v", err) + o.DebugPrint("", o.Bytes()) + t.Fatalf("expected = %s", expected) + } + if !equal(o.Bytes(), expected, t) { + o.DebugPrint("overify neq 1", o.Bytes()) + t.Fatalf("expected = %s", expected) + } + + // Now test Unmarshal by recreating the original buffer. + pbd := new(GoTest) + err = o.Unmarshal(pbd) + if err != nil { + t.Fatalf("overify unmarshal err = %v", err) + o.DebugPrint("", o.Bytes()) + t.Fatalf("string = %s", expected) + } + o.Reset() + err = o.Marshal(pbd) + if err != nil { + t.Errorf("overify marshal-2 err = %v", err) + o.DebugPrint("", o.Bytes()) + t.Fatalf("string = %s", expected) + } + if !equal(o.Bytes(), expected, t) { + o.DebugPrint("overify neq 2", o.Bytes()) + t.Fatalf("string = %s", expected) + } +} + +// Simple tests for numeric encode/decode primitives (varint, etc.) +func TestNumericPrimitives(t *testing.T) { + for i := uint64(0); i < 1e6; i += 111 { + o := old() + if o.EncodeVarint(i) != nil { + t.Error("EncodeVarint") + break + } + x, e := o.DecodeVarint() + if e != nil { + t.Fatal("DecodeVarint") + } + if x != i { + t.Fatal("varint decode fail:", i, x) + } + + o = old() + if o.EncodeFixed32(i) != nil { + t.Fatal("encFixed32") + } + x, e = o.DecodeFixed32() + if e != nil { + t.Fatal("decFixed32") + } + if x != i { + t.Fatal("fixed32 decode fail:", i, x) + } + + o = old() + if o.EncodeFixed64(i*1234567) != nil { + t.Error("encFixed64") + break + } + x, e = o.DecodeFixed64() + if e != nil { + t.Error("decFixed64") + break + } + if x != i*1234567 { + t.Error("fixed64 decode fail:", i*1234567, x) + break + } + + o = old() + i32 := int32(i - 12345) + if o.EncodeZigzag32(uint64(i32)) != nil { + t.Fatal("EncodeZigzag32") + } + x, e = o.DecodeZigzag32() + if e != nil { + t.Fatal("DecodeZigzag32") + } + if x != uint64(uint32(i32)) { + t.Fatal("zigzag32 decode fail:", i32, x) + } + + o = old() + i64 := int64(i - 12345) + if o.EncodeZigzag64(uint64(i64)) != nil { + t.Fatal("EncodeZigzag64") + } + x, e = o.DecodeZigzag64() + if e != nil { + t.Fatal("DecodeZigzag64") + } + if x != uint64(i64) { + t.Fatal("zigzag64 decode fail:", i64, x) + } + } +} + +// fakeMarshaler is a simple struct implementing Marshaler and Message interfaces. +type fakeMarshaler struct { + b []byte + err error +} + +func (f *fakeMarshaler) Marshal() ([]byte, error) { return f.b, f.err } +func (f *fakeMarshaler) String() string { return fmt.Sprintf("Bytes: %v Error: %v", f.b, f.err) } +func (f *fakeMarshaler) ProtoMessage() {} +func (f *fakeMarshaler) Reset() {} + +type msgWithFakeMarshaler struct { + M *fakeMarshaler `protobuf:"bytes,1,opt,name=fake"` +} + +func (m *msgWithFakeMarshaler) String() string { return CompactTextString(m) } +func (m *msgWithFakeMarshaler) ProtoMessage() {} +func (m *msgWithFakeMarshaler) Reset() {} + +// Simple tests for proto messages that implement the Marshaler interface. +func TestMarshalerEncoding(t *testing.T) { + tests := []struct { + name string + m Message + want []byte + wantErr error + }{ + { + name: "Marshaler that fails", + m: &fakeMarshaler{ + err: errors.New("some marshal err"), + b: []byte{5, 6, 7}, + }, + // Since there's an error, nothing should be written to buffer. + want: nil, + wantErr: errors.New("some marshal err"), + }, + { + name: "Marshaler that fails with RequiredNotSetError", + m: &msgWithFakeMarshaler{ + M: &fakeMarshaler{ + err: &RequiredNotSetError{}, + b: []byte{5, 6, 7}, + }, + }, + // Since there's an error that can be continued after, + // the buffer should be written. + want: []byte{ + 10, 3, // for &msgWithFakeMarshaler + 5, 6, 7, // for &fakeMarshaler + }, + wantErr: &RequiredNotSetError{}, + }, + { + name: "Marshaler that succeeds", + m: &fakeMarshaler{ + b: []byte{0, 1, 2, 3, 4, 127, 255}, + }, + want: []byte{0, 1, 2, 3, 4, 127, 255}, + wantErr: nil, + }, + } + for _, test := range tests { + b := NewBuffer(nil) + err := b.Marshal(test.m) + if _, ok := err.(*RequiredNotSetError); ok { + // We're not in package proto, so we can only assert the type in this case. + err = &RequiredNotSetError{} + } + if !reflect.DeepEqual(test.wantErr, err) { + t.Errorf("%s: got err %v wanted %v", test.name, err, test.wantErr) + } + if !reflect.DeepEqual(test.want, b.Bytes()) { + t.Errorf("%s: got bytes %v wanted %v", test.name, b.Bytes(), test.want) + } + } +} + +// Simple tests for bytes +func TestBytesPrimitives(t *testing.T) { + o := old() + bytes := []byte{'n', 'o', 'w', ' ', 'i', 's', ' ', 't', 'h', 'e', ' ', 't', 'i', 'm', 'e'} + if o.EncodeRawBytes(bytes) != nil { + t.Error("EncodeRawBytes") + } + decb, e := o.DecodeRawBytes(false) + if e != nil { + t.Error("DecodeRawBytes") + } + equalbytes(bytes, decb, t) +} + +// Simple tests for strings +func TestStringPrimitives(t *testing.T) { + o := old() + s := "now is the time" + if o.EncodeStringBytes(s) != nil { + t.Error("enc_string") + } + decs, e := o.DecodeStringBytes() + if e != nil { + t.Error("dec_string") + } + if s != decs { + t.Error("string encode/decode fail:", s, decs) + } +} + +// Do we catch the "required bit not set" case? +func TestRequiredBit(t *testing.T) { + o := old() + pb := new(GoTest) + err := o.Marshal(pb) + if err == nil { + t.Error("did not catch missing required fields") + } else if strings.Index(err.Error(), "Kind") < 0 { + t.Error("wrong error type:", err) + } +} + +// Check that all fields are nil. +// Clearly silly, and a residue from a more interesting test with an earlier, +// different initialization property, but it once caught a compiler bug so +// it lives. +func checkInitialized(pb *GoTest, t *testing.T) { + if pb.F_BoolDefaulted != nil { + t.Error("New or Reset did not set boolean:", *pb.F_BoolDefaulted) + } + if pb.F_Int32Defaulted != nil { + t.Error("New or Reset did not set int32:", *pb.F_Int32Defaulted) + } + if pb.F_Int64Defaulted != nil { + t.Error("New or Reset did not set int64:", *pb.F_Int64Defaulted) + } + if pb.F_Fixed32Defaulted != nil { + t.Error("New or Reset did not set fixed32:", *pb.F_Fixed32Defaulted) + } + if pb.F_Fixed64Defaulted != nil { + t.Error("New or Reset did not set fixed64:", *pb.F_Fixed64Defaulted) + } + if pb.F_Uint32Defaulted != nil { + t.Error("New or Reset did not set uint32:", *pb.F_Uint32Defaulted) + } + if pb.F_Uint64Defaulted != nil { + t.Error("New or Reset did not set uint64:", *pb.F_Uint64Defaulted) + } + if pb.F_FloatDefaulted != nil { + t.Error("New or Reset did not set float:", *pb.F_FloatDefaulted) + } + if pb.F_DoubleDefaulted != nil { + t.Error("New or Reset did not set double:", *pb.F_DoubleDefaulted) + } + if pb.F_StringDefaulted != nil { + t.Error("New or Reset did not set string:", *pb.F_StringDefaulted) + } + if pb.F_BytesDefaulted != nil { + t.Error("New or Reset did not set bytes:", string(pb.F_BytesDefaulted)) + } + if pb.F_Sint32Defaulted != nil { + t.Error("New or Reset did not set int32:", *pb.F_Sint32Defaulted) + } + if pb.F_Sint64Defaulted != nil { + t.Error("New or Reset did not set int64:", *pb.F_Sint64Defaulted) + } +} + +// Does Reset() reset? +func TestReset(t *testing.T) { + pb := initGoTest(true) + // muck with some values + pb.F_BoolDefaulted = Bool(false) + pb.F_Int32Defaulted = Int32(237) + pb.F_Int64Defaulted = Int64(12346) + pb.F_Fixed32Defaulted = Uint32(32000) + pb.F_Fixed64Defaulted = Uint64(666) + pb.F_Uint32Defaulted = Uint32(323232) + pb.F_Uint64Defaulted = nil + pb.F_FloatDefaulted = nil + pb.F_DoubleDefaulted = Float64(0) + pb.F_StringDefaulted = String("gotcha") + pb.F_BytesDefaulted = []byte("asdfasdf") + pb.F_Sint32Defaulted = Int32(123) + pb.F_Sint64Defaulted = Int64(789) + pb.Reset() + checkInitialized(pb, t) +} + +// All required fields set, no defaults provided. +func TestEncodeDecode1(t *testing.T) { + pb := initGoTest(false) + overify(t, pb, + "0807"+ // field 1, encoding 0, value 7 + "220d"+"0a056c6162656c120474797065"+ // field 4, encoding 2 (GoTestField) + "5001"+ // field 10, encoding 0, value 1 + "5803"+ // field 11, encoding 0, value 3 + "6006"+ // field 12, encoding 0, value 6 + "6d20000000"+ // field 13, encoding 5, value 0x20 + "714000000000000000"+ // field 14, encoding 1, value 0x40 + "78a019"+ // field 15, encoding 0, value 0xca0 = 3232 + "8001c032"+ // field 16, encoding 0, value 0x1940 = 6464 + "8d0100004a45"+ // field 17, encoding 5, value 3232.0 + "9101000000000040b940"+ // field 18, encoding 1, value 6464.0 + "9a0106"+"737472696e67"+ // field 19, encoding 2, string "string" + "b304"+ // field 70, encoding 3, start group + "ba0408"+"7265717569726564"+ // field 71, encoding 2, string "required" + "b404"+ // field 70, encoding 4, end group + "aa0605"+"6279746573"+ // field 101, encoding 2, string "bytes" + "b0063f"+ // field 102, encoding 0, 0x3f zigzag32 + "b8067f") // field 103, encoding 0, 0x7f zigzag64 +} + +// All required fields set, defaults provided. +func TestEncodeDecode2(t *testing.T) { + pb := initGoTest(true) + overify(t, pb, + "0807"+ // field 1, encoding 0, value 7 + "220d"+"0a056c6162656c120474797065"+ // field 4, encoding 2 (GoTestField) + "5001"+ // field 10, encoding 0, value 1 + "5803"+ // field 11, encoding 0, value 3 + "6006"+ // field 12, encoding 0, value 6 + "6d20000000"+ // field 13, encoding 5, value 32 + "714000000000000000"+ // field 14, encoding 1, value 64 + "78a019"+ // field 15, encoding 0, value 3232 + "8001c032"+ // field 16, encoding 0, value 6464 + "8d0100004a45"+ // field 17, encoding 5, value 3232.0 + "9101000000000040b940"+ // field 18, encoding 1, value 6464.0 + "9a0106"+"737472696e67"+ // field 19, encoding 2 string "string" + "c00201"+ // field 40, encoding 0, value 1 + "c80220"+ // field 41, encoding 0, value 32 + "d00240"+ // field 42, encoding 0, value 64 + "dd0240010000"+ // field 43, encoding 5, value 320 + "e1028002000000000000"+ // field 44, encoding 1, value 640 + "e8028019"+ // field 45, encoding 0, value 3200 + "f0028032"+ // field 46, encoding 0, value 6400 + "fd02e0659948"+ // field 47, encoding 5, value 314159.0 + "81030000000050971041"+ // field 48, encoding 1, value 271828.0 + "8a0310"+"68656c6c6f2c2022776f726c6421220a"+ // field 49, encoding 2 string "hello, \"world!\"\n" + "b304"+ // start group field 70 level 1 + "ba0408"+"7265717569726564"+ // field 71, encoding 2, string "required" + "b404"+ // end group field 70 level 1 + "aa0605"+"6279746573"+ // field 101, encoding 2 string "bytes" + "b0063f"+ // field 102, encoding 0, 0x3f zigzag32 + "b8067f"+ // field 103, encoding 0, 0x7f zigzag64 + "8a1907"+"4269676e6f7365"+ // field 401, encoding 2, string "Bignose" + "90193f"+ // field 402, encoding 0, value 63 + "98197f") // field 403, encoding 0, value 127 + +} + +// All default fields set to their default value by hand +func TestEncodeDecode3(t *testing.T) { + pb := initGoTest(false) + pb.F_BoolDefaulted = Bool(true) + pb.F_Int32Defaulted = Int32(32) + pb.F_Int64Defaulted = Int64(64) + pb.F_Fixed32Defaulted = Uint32(320) + pb.F_Fixed64Defaulted = Uint64(640) + pb.F_Uint32Defaulted = Uint32(3200) + pb.F_Uint64Defaulted = Uint64(6400) + pb.F_FloatDefaulted = Float32(314159) + pb.F_DoubleDefaulted = Float64(271828) + pb.F_StringDefaulted = String("hello, \"world!\"\n") + pb.F_BytesDefaulted = []byte("Bignose") + pb.F_Sint32Defaulted = Int32(-32) + pb.F_Sint64Defaulted = Int64(-64) + + overify(t, pb, + "0807"+ // field 1, encoding 0, value 7 + "220d"+"0a056c6162656c120474797065"+ // field 4, encoding 2 (GoTestField) + "5001"+ // field 10, encoding 0, value 1 + "5803"+ // field 11, encoding 0, value 3 + "6006"+ // field 12, encoding 0, value 6 + "6d20000000"+ // field 13, encoding 5, value 32 + "714000000000000000"+ // field 14, encoding 1, value 64 + "78a019"+ // field 15, encoding 0, value 3232 + "8001c032"+ // field 16, encoding 0, value 6464 + "8d0100004a45"+ // field 17, encoding 5, value 3232.0 + "9101000000000040b940"+ // field 18, encoding 1, value 6464.0 + "9a0106"+"737472696e67"+ // field 19, encoding 2 string "string" + "c00201"+ // field 40, encoding 0, value 1 + "c80220"+ // field 41, encoding 0, value 32 + "d00240"+ // field 42, encoding 0, value 64 + "dd0240010000"+ // field 43, encoding 5, value 320 + "e1028002000000000000"+ // field 44, encoding 1, value 640 + "e8028019"+ // field 45, encoding 0, value 3200 + "f0028032"+ // field 46, encoding 0, value 6400 + "fd02e0659948"+ // field 47, encoding 5, value 314159.0 + "81030000000050971041"+ // field 48, encoding 1, value 271828.0 + "8a0310"+"68656c6c6f2c2022776f726c6421220a"+ // field 49, encoding 2 string "hello, \"world!\"\n" + "b304"+ // start group field 70 level 1 + "ba0408"+"7265717569726564"+ // field 71, encoding 2, string "required" + "b404"+ // end group field 70 level 1 + "aa0605"+"6279746573"+ // field 101, encoding 2 string "bytes" + "b0063f"+ // field 102, encoding 0, 0x3f zigzag32 + "b8067f"+ // field 103, encoding 0, 0x7f zigzag64 + "8a1907"+"4269676e6f7365"+ // field 401, encoding 2, string "Bignose" + "90193f"+ // field 402, encoding 0, value 63 + "98197f") // field 403, encoding 0, value 127 + +} + +// All required fields set, defaults provided, all non-defaulted optional fields have values. +func TestEncodeDecode4(t *testing.T) { + pb := initGoTest(true) + pb.Table = String("hello") + pb.Param = Int32(7) + pb.OptionalField = initGoTestField() + pb.F_BoolOptional = Bool(true) + pb.F_Int32Optional = Int32(32) + pb.F_Int64Optional = Int64(64) + pb.F_Fixed32Optional = Uint32(3232) + pb.F_Fixed64Optional = Uint64(6464) + pb.F_Uint32Optional = Uint32(323232) + pb.F_Uint64Optional = Uint64(646464) + pb.F_FloatOptional = Float32(32.) + pb.F_DoubleOptional = Float64(64.) + pb.F_StringOptional = String("hello") + pb.F_BytesOptional = []byte("Bignose") + pb.F_Sint32Optional = Int32(-32) + pb.F_Sint64Optional = Int64(-64) + pb.Optionalgroup = initGoTest_OptionalGroup() + + overify(t, pb, + "0807"+ // field 1, encoding 0, value 7 + "1205"+"68656c6c6f"+ // field 2, encoding 2, string "hello" + "1807"+ // field 3, encoding 0, value 7 + "220d"+"0a056c6162656c120474797065"+ // field 4, encoding 2 (GoTestField) + "320d"+"0a056c6162656c120474797065"+ // field 6, encoding 2 (GoTestField) + "5001"+ // field 10, encoding 0, value 1 + "5803"+ // field 11, encoding 0, value 3 + "6006"+ // field 12, encoding 0, value 6 + "6d20000000"+ // field 13, encoding 5, value 32 + "714000000000000000"+ // field 14, encoding 1, value 64 + "78a019"+ // field 15, encoding 0, value 3232 + "8001c032"+ // field 16, encoding 0, value 6464 + "8d0100004a45"+ // field 17, encoding 5, value 3232.0 + "9101000000000040b940"+ // field 18, encoding 1, value 6464.0 + "9a0106"+"737472696e67"+ // field 19, encoding 2 string "string" + "f00101"+ // field 30, encoding 0, value 1 + "f80120"+ // field 31, encoding 0, value 32 + "800240"+ // field 32, encoding 0, value 64 + "8d02a00c0000"+ // field 33, encoding 5, value 3232 + "91024019000000000000"+ // field 34, encoding 1, value 6464 + "9802a0dd13"+ // field 35, encoding 0, value 323232 + "a002c0ba27"+ // field 36, encoding 0, value 646464 + "ad0200000042"+ // field 37, encoding 5, value 32.0 + "b1020000000000005040"+ // field 38, encoding 1, value 64.0 + "ba0205"+"68656c6c6f"+ // field 39, encoding 2, string "hello" + "c00201"+ // field 40, encoding 0, value 1 + "c80220"+ // field 41, encoding 0, value 32 + "d00240"+ // field 42, encoding 0, value 64 + "dd0240010000"+ // field 43, encoding 5, value 320 + "e1028002000000000000"+ // field 44, encoding 1, value 640 + "e8028019"+ // field 45, encoding 0, value 3200 + "f0028032"+ // field 46, encoding 0, value 6400 + "fd02e0659948"+ // field 47, encoding 5, value 314159.0 + "81030000000050971041"+ // field 48, encoding 1, value 271828.0 + "8a0310"+"68656c6c6f2c2022776f726c6421220a"+ // field 49, encoding 2 string "hello, \"world!\"\n" + "b304"+ // start group field 70 level 1 + "ba0408"+"7265717569726564"+ // field 71, encoding 2, string "required" + "b404"+ // end group field 70 level 1 + "d305"+ // start group field 90 level 1 + "da0508"+"6f7074696f6e616c"+ // field 91, encoding 2, string "optional" + "d405"+ // end group field 90 level 1 + "aa0605"+"6279746573"+ // field 101, encoding 2 string "bytes" + "b0063f"+ // field 102, encoding 0, 0x3f zigzag32 + "b8067f"+ // field 103, encoding 0, 0x7f zigzag64 + "ea1207"+"4269676e6f7365"+ // field 301, encoding 2, string "Bignose" + "f0123f"+ // field 302, encoding 0, value 63 + "f8127f"+ // field 303, encoding 0, value 127 + "8a1907"+"4269676e6f7365"+ // field 401, encoding 2, string "Bignose" + "90193f"+ // field 402, encoding 0, value 63 + "98197f") // field 403, encoding 0, value 127 + +} + +// All required fields set, defaults provided, all repeated fields given two values. +func TestEncodeDecode5(t *testing.T) { + pb := initGoTest(true) + pb.RepeatedField = []*GoTestField{initGoTestField(), initGoTestField()} + pb.F_BoolRepeated = []bool{false, true} + pb.F_Int32Repeated = []int32{32, 33} + pb.F_Int64Repeated = []int64{64, 65} + pb.F_Fixed32Repeated = []uint32{3232, 3333} + pb.F_Fixed64Repeated = []uint64{6464, 6565} + pb.F_Uint32Repeated = []uint32{323232, 333333} + pb.F_Uint64Repeated = []uint64{646464, 656565} + pb.F_FloatRepeated = []float32{32., 33.} + pb.F_DoubleRepeated = []float64{64., 65.} + pb.F_StringRepeated = []string{"hello", "sailor"} + pb.F_BytesRepeated = [][]byte{[]byte("big"), []byte("nose")} + pb.F_Sint32Repeated = []int32{32, -32} + pb.F_Sint64Repeated = []int64{64, -64} + pb.Repeatedgroup = []*GoTest_RepeatedGroup{initGoTest_RepeatedGroup(), initGoTest_RepeatedGroup()} + + overify(t, pb, + "0807"+ // field 1, encoding 0, value 7 + "220d"+"0a056c6162656c120474797065"+ // field 4, encoding 2 (GoTestField) + "2a0d"+"0a056c6162656c120474797065"+ // field 5, encoding 2 (GoTestField) + "2a0d"+"0a056c6162656c120474797065"+ // field 5, encoding 2 (GoTestField) + "5001"+ // field 10, encoding 0, value 1 + "5803"+ // field 11, encoding 0, value 3 + "6006"+ // field 12, encoding 0, value 6 + "6d20000000"+ // field 13, encoding 5, value 32 + "714000000000000000"+ // field 14, encoding 1, value 64 + "78a019"+ // field 15, encoding 0, value 3232 + "8001c032"+ // field 16, encoding 0, value 6464 + "8d0100004a45"+ // field 17, encoding 5, value 3232.0 + "9101000000000040b940"+ // field 18, encoding 1, value 6464.0 + "9a0106"+"737472696e67"+ // field 19, encoding 2 string "string" + "a00100"+ // field 20, encoding 0, value 0 + "a00101"+ // field 20, encoding 0, value 1 + "a80120"+ // field 21, encoding 0, value 32 + "a80121"+ // field 21, encoding 0, value 33 + "b00140"+ // field 22, encoding 0, value 64 + "b00141"+ // field 22, encoding 0, value 65 + "bd01a00c0000"+ // field 23, encoding 5, value 3232 + "bd01050d0000"+ // field 23, encoding 5, value 3333 + "c1014019000000000000"+ // field 24, encoding 1, value 6464 + "c101a519000000000000"+ // field 24, encoding 1, value 6565 + "c801a0dd13"+ // field 25, encoding 0, value 323232 + "c80195ac14"+ // field 25, encoding 0, value 333333 + "d001c0ba27"+ // field 26, encoding 0, value 646464 + "d001b58928"+ // field 26, encoding 0, value 656565 + "dd0100000042"+ // field 27, encoding 5, value 32.0 + "dd0100000442"+ // field 27, encoding 5, value 33.0 + "e1010000000000005040"+ // field 28, encoding 1, value 64.0 + "e1010000000000405040"+ // field 28, encoding 1, value 65.0 + "ea0105"+"68656c6c6f"+ // field 29, encoding 2, string "hello" + "ea0106"+"7361696c6f72"+ // field 29, encoding 2, string "sailor" + "c00201"+ // field 40, encoding 0, value 1 + "c80220"+ // field 41, encoding 0, value 32 + "d00240"+ // field 42, encoding 0, value 64 + "dd0240010000"+ // field 43, encoding 5, value 320 + "e1028002000000000000"+ // field 44, encoding 1, value 640 + "e8028019"+ // field 45, encoding 0, value 3200 + "f0028032"+ // field 46, encoding 0, value 6400 + "fd02e0659948"+ // field 47, encoding 5, value 314159.0 + "81030000000050971041"+ // field 48, encoding 1, value 271828.0 + "8a0310"+"68656c6c6f2c2022776f726c6421220a"+ // field 49, encoding 2 string "hello, \"world!\"\n" + "b304"+ // start group field 70 level 1 + "ba0408"+"7265717569726564"+ // field 71, encoding 2, string "required" + "b404"+ // end group field 70 level 1 + "8305"+ // start group field 80 level 1 + "8a0508"+"7265706561746564"+ // field 81, encoding 2, string "repeated" + "8405"+ // end group field 80 level 1 + "8305"+ // start group field 80 level 1 + "8a0508"+"7265706561746564"+ // field 81, encoding 2, string "repeated" + "8405"+ // end group field 80 level 1 + "aa0605"+"6279746573"+ // field 101, encoding 2 string "bytes" + "b0063f"+ // field 102, encoding 0, 0x3f zigzag32 + "b8067f"+ // field 103, encoding 0, 0x7f zigzag64 + "ca0c03"+"626967"+ // field 201, encoding 2, string "big" + "ca0c04"+"6e6f7365"+ // field 201, encoding 2, string "nose" + "d00c40"+ // field 202, encoding 0, value 32 + "d00c3f"+ // field 202, encoding 0, value -32 + "d80c8001"+ // field 203, encoding 0, value 64 + "d80c7f"+ // field 203, encoding 0, value -64 + "8a1907"+"4269676e6f7365"+ // field 401, encoding 2, string "Bignose" + "90193f"+ // field 402, encoding 0, value 63 + "98197f") // field 403, encoding 0, value 127 + +} + +// All required fields set, all packed repeated fields given two values. +func TestEncodeDecode6(t *testing.T) { + pb := initGoTest(false) + pb.F_BoolRepeatedPacked = []bool{false, true} + pb.F_Int32RepeatedPacked = []int32{32, 33} + pb.F_Int64RepeatedPacked = []int64{64, 65} + pb.F_Fixed32RepeatedPacked = []uint32{3232, 3333} + pb.F_Fixed64RepeatedPacked = []uint64{6464, 6565} + pb.F_Uint32RepeatedPacked = []uint32{323232, 333333} + pb.F_Uint64RepeatedPacked = []uint64{646464, 656565} + pb.F_FloatRepeatedPacked = []float32{32., 33.} + pb.F_DoubleRepeatedPacked = []float64{64., 65.} + pb.F_Sint32RepeatedPacked = []int32{32, -32} + pb.F_Sint64RepeatedPacked = []int64{64, -64} + + overify(t, pb, + "0807"+ // field 1, encoding 0, value 7 + "220d"+"0a056c6162656c120474797065"+ // field 4, encoding 2 (GoTestField) + "5001"+ // field 10, encoding 0, value 1 + "5803"+ // field 11, encoding 0, value 3 + "6006"+ // field 12, encoding 0, value 6 + "6d20000000"+ // field 13, encoding 5, value 32 + "714000000000000000"+ // field 14, encoding 1, value 64 + "78a019"+ // field 15, encoding 0, value 3232 + "8001c032"+ // field 16, encoding 0, value 6464 + "8d0100004a45"+ // field 17, encoding 5, value 3232.0 + "9101000000000040b940"+ // field 18, encoding 1, value 6464.0 + "9a0106"+"737472696e67"+ // field 19, encoding 2 string "string" + "9203020001"+ // field 50, encoding 2, 2 bytes, value 0, value 1 + "9a03022021"+ // field 51, encoding 2, 2 bytes, value 32, value 33 + "a203024041"+ // field 52, encoding 2, 2 bytes, value 64, value 65 + "aa0308"+ // field 53, encoding 2, 8 bytes + "a00c0000050d0000"+ // value 3232, value 3333 + "b20310"+ // field 54, encoding 2, 16 bytes + "4019000000000000a519000000000000"+ // value 6464, value 6565 + "ba0306"+ // field 55, encoding 2, 6 bytes + "a0dd1395ac14"+ // value 323232, value 333333 + "c20306"+ // field 56, encoding 2, 6 bytes + "c0ba27b58928"+ // value 646464, value 656565 + "ca0308"+ // field 57, encoding 2, 8 bytes + "0000004200000442"+ // value 32.0, value 33.0 + "d20310"+ // field 58, encoding 2, 16 bytes + "00000000000050400000000000405040"+ // value 64.0, value 65.0 + "b304"+ // start group field 70 level 1 + "ba0408"+"7265717569726564"+ // field 71, encoding 2, string "required" + "b404"+ // end group field 70 level 1 + "aa0605"+"6279746573"+ // field 101, encoding 2 string "bytes" + "b0063f"+ // field 102, encoding 0, 0x3f zigzag32 + "b8067f"+ // field 103, encoding 0, 0x7f zigzag64 + "b21f02"+ // field 502, encoding 2, 2 bytes + "403f"+ // value 32, value -32 + "ba1f03"+ // field 503, encoding 2, 3 bytes + "80017f") // value 64, value -64 +} + +// Test that we can encode empty bytes fields. +func TestEncodeDecodeBytes1(t *testing.T) { + pb := initGoTest(false) + + // Create our bytes + pb.F_BytesRequired = []byte{} + pb.F_BytesRepeated = [][]byte{{}} + pb.F_BytesOptional = []byte{} + + d, err := Marshal(pb) + if err != nil { + t.Error(err) + } + + pbd := new(GoTest) + if err := Unmarshal(d, pbd); err != nil { + t.Error(err) + } + + if pbd.F_BytesRequired == nil || len(pbd.F_BytesRequired) != 0 { + t.Error("required empty bytes field is incorrect") + } + if pbd.F_BytesRepeated == nil || len(pbd.F_BytesRepeated) == 1 && pbd.F_BytesRepeated[0] == nil { + t.Error("repeated empty bytes field is incorrect") + } + if pbd.F_BytesOptional == nil || len(pbd.F_BytesOptional) != 0 { + t.Error("optional empty bytes field is incorrect") + } +} + +// Test that we encode nil-valued fields of a repeated bytes field correctly. +// Since entries in a repeated field cannot be nil, nil must mean empty value. +func TestEncodeDecodeBytes2(t *testing.T) { + pb := initGoTest(false) + + // Create our bytes + pb.F_BytesRepeated = [][]byte{nil} + + d, err := Marshal(pb) + if err != nil { + t.Error(err) + } + + pbd := new(GoTest) + if err := Unmarshal(d, pbd); err != nil { + t.Error(err) + } + + if len(pbd.F_BytesRepeated) != 1 || pbd.F_BytesRepeated[0] == nil { + t.Error("Unexpected value for repeated bytes field") + } +} + +// All required fields set, defaults provided, all repeated fields given two values. +func TestSkippingUnrecognizedFields(t *testing.T) { + o := old() + pb := initGoTestField() + + // Marshal it normally. + o.Marshal(pb) + + // Now new a GoSkipTest record. + skip := &GoSkipTest{ + SkipInt32: Int32(32), + SkipFixed32: Uint32(3232), + SkipFixed64: Uint64(6464), + SkipString: String("skipper"), + Skipgroup: &GoSkipTest_SkipGroup{ + GroupInt32: Int32(75), + GroupString: String("wxyz"), + }, + } + + // Marshal it into same buffer. + o.Marshal(skip) + + pbd := new(GoTestField) + o.Unmarshal(pbd) + + // The __unrecognized field should be a marshaling of GoSkipTest + skipd := new(GoSkipTest) + + o.SetBuf(pbd.XXX_unrecognized) + o.Unmarshal(skipd) + + if *skipd.SkipInt32 != *skip.SkipInt32 { + t.Error("skip int32", skipd.SkipInt32) + } + if *skipd.SkipFixed32 != *skip.SkipFixed32 { + t.Error("skip fixed32", skipd.SkipFixed32) + } + if *skipd.SkipFixed64 != *skip.SkipFixed64 { + t.Error("skip fixed64", skipd.SkipFixed64) + } + if *skipd.SkipString != *skip.SkipString { + t.Error("skip string", *skipd.SkipString) + } + if *skipd.Skipgroup.GroupInt32 != *skip.Skipgroup.GroupInt32 { + t.Error("skip group int32", skipd.Skipgroup.GroupInt32) + } + if *skipd.Skipgroup.GroupString != *skip.Skipgroup.GroupString { + t.Error("skip group string", *skipd.Skipgroup.GroupString) + } +} + +// Check that unrecognized fields of a submessage are preserved. +func TestSubmessageUnrecognizedFields(t *testing.T) { + nm := &NewMessage{ + Nested: &NewMessage_Nested{ + Name: String("Nigel"), + FoodGroup: String("carbs"), + }, + } + b, err := Marshal(nm) + if err != nil { + t.Fatalf("Marshal of NewMessage: %v", err) + } + + // Unmarshal into an OldMessage. + om := new(OldMessage) + if err := Unmarshal(b, om); err != nil { + t.Fatalf("Unmarshal to OldMessage: %v", err) + } + exp := &OldMessage{ + Nested: &OldMessage_Nested{ + Name: String("Nigel"), + // normal protocol buffer users should not do this + XXX_unrecognized: []byte("\x12\x05carbs"), + }, + } + if !Equal(om, exp) { + t.Errorf("om = %v, want %v", om, exp) + } + + // Clone the OldMessage. + om = Clone(om).(*OldMessage) + if !Equal(om, exp) { + t.Errorf("Clone(om) = %v, want %v", om, exp) + } + + // Marshal the OldMessage, then unmarshal it into an empty NewMessage. + if b, err = Marshal(om); err != nil { + t.Fatalf("Marshal of OldMessage: %v", err) + } + t.Logf("Marshal(%v) -> %q", om, b) + nm2 := new(NewMessage) + if err := Unmarshal(b, nm2); err != nil { + t.Fatalf("Unmarshal to NewMessage: %v", err) + } + if !Equal(nm, nm2) { + t.Errorf("NewMessage round-trip: %v => %v", nm, nm2) + } +} + +// Check that an int32 field can be upgraded to an int64 field. +func TestNegativeInt32(t *testing.T) { + om := &OldMessage{ + Num: Int32(-1), + } + b, err := Marshal(om) + if err != nil { + t.Fatalf("Marshal of OldMessage: %v", err) + } + + // Check the size. It should be 11 bytes; + // 1 for the field/wire type, and 10 for the negative number. + if len(b) != 11 { + t.Errorf("%v marshaled as %q, wanted 11 bytes", om, b) + } + + // Unmarshal into a NewMessage. + nm := new(NewMessage) + if err := Unmarshal(b, nm); err != nil { + t.Fatalf("Unmarshal to NewMessage: %v", err) + } + want := &NewMessage{ + Num: Int64(-1), + } + if !Equal(nm, want) { + t.Errorf("nm = %v, want %v", nm, want) + } +} + +// Check that we can grow an array (repeated field) to have many elements. +// This test doesn't depend only on our encoding; for variety, it makes sure +// we create, encode, and decode the correct contents explicitly. It's therefore +// a bit messier. +// This test also uses (and hence tests) the Marshal/Unmarshal functions +// instead of the methods. +func TestBigRepeated(t *testing.T) { + pb := initGoTest(true) + + // Create the arrays + const N = 50 // Internally the library starts much smaller. + pb.Repeatedgroup = make([]*GoTest_RepeatedGroup, N) + pb.F_Sint64Repeated = make([]int64, N) + pb.F_Sint32Repeated = make([]int32, N) + pb.F_BytesRepeated = make([][]byte, N) + pb.F_StringRepeated = make([]string, N) + pb.F_DoubleRepeated = make([]float64, N) + pb.F_FloatRepeated = make([]float32, N) + pb.F_Uint64Repeated = make([]uint64, N) + pb.F_Uint32Repeated = make([]uint32, N) + pb.F_Fixed64Repeated = make([]uint64, N) + pb.F_Fixed32Repeated = make([]uint32, N) + pb.F_Int64Repeated = make([]int64, N) + pb.F_Int32Repeated = make([]int32, N) + pb.F_BoolRepeated = make([]bool, N) + pb.RepeatedField = make([]*GoTestField, N) + + // Fill in the arrays with checkable values. + igtf := initGoTestField() + igtrg := initGoTest_RepeatedGroup() + for i := 0; i < N; i++ { + pb.Repeatedgroup[i] = igtrg + pb.F_Sint64Repeated[i] = int64(i) + pb.F_Sint32Repeated[i] = int32(i) + s := fmt.Sprint(i) + pb.F_BytesRepeated[i] = []byte(s) + pb.F_StringRepeated[i] = s + pb.F_DoubleRepeated[i] = float64(i) + pb.F_FloatRepeated[i] = float32(i) + pb.F_Uint64Repeated[i] = uint64(i) + pb.F_Uint32Repeated[i] = uint32(i) + pb.F_Fixed64Repeated[i] = uint64(i) + pb.F_Fixed32Repeated[i] = uint32(i) + pb.F_Int64Repeated[i] = int64(i) + pb.F_Int32Repeated[i] = int32(i) + pb.F_BoolRepeated[i] = i%2 == 0 + pb.RepeatedField[i] = igtf + } + + // Marshal. + buf, _ := Marshal(pb) + + // Now test Unmarshal by recreating the original buffer. + pbd := new(GoTest) + Unmarshal(buf, pbd) + + // Check the checkable values + for i := uint64(0); i < N; i++ { + if pbd.Repeatedgroup[i] == nil { // TODO: more checking? + t.Error("pbd.Repeatedgroup bad") + } + var x uint64 + x = uint64(pbd.F_Sint64Repeated[i]) + if x != i { + t.Error("pbd.F_Sint64Repeated bad", x, i) + } + x = uint64(pbd.F_Sint32Repeated[i]) + if x != i { + t.Error("pbd.F_Sint32Repeated bad", x, i) + } + s := fmt.Sprint(i) + equalbytes(pbd.F_BytesRepeated[i], []byte(s), t) + if pbd.F_StringRepeated[i] != s { + t.Error("pbd.F_Sint32Repeated bad", pbd.F_StringRepeated[i], i) + } + x = uint64(pbd.F_DoubleRepeated[i]) + if x != i { + t.Error("pbd.F_DoubleRepeated bad", x, i) + } + x = uint64(pbd.F_FloatRepeated[i]) + if x != i { + t.Error("pbd.F_FloatRepeated bad", x, i) + } + x = pbd.F_Uint64Repeated[i] + if x != i { + t.Error("pbd.F_Uint64Repeated bad", x, i) + } + x = uint64(pbd.F_Uint32Repeated[i]) + if x != i { + t.Error("pbd.F_Uint32Repeated bad", x, i) + } + x = pbd.F_Fixed64Repeated[i] + if x != i { + t.Error("pbd.F_Fixed64Repeated bad", x, i) + } + x = uint64(pbd.F_Fixed32Repeated[i]) + if x != i { + t.Error("pbd.F_Fixed32Repeated bad", x, i) + } + x = uint64(pbd.F_Int64Repeated[i]) + if x != i { + t.Error("pbd.F_Int64Repeated bad", x, i) + } + x = uint64(pbd.F_Int32Repeated[i]) + if x != i { + t.Error("pbd.F_Int32Repeated bad", x, i) + } + if pbd.F_BoolRepeated[i] != (i%2 == 0) { + t.Error("pbd.F_BoolRepeated bad", x, i) + } + if pbd.RepeatedField[i] == nil { // TODO: more checking? + t.Error("pbd.RepeatedField bad") + } + } +} + +// Verify we give a useful message when decoding to the wrong structure type. +func TestTypeMismatch(t *testing.T) { + pb1 := initGoTest(true) + + // Marshal + o := old() + o.Marshal(pb1) + + // Now Unmarshal it to the wrong type. + pb2 := initGoTestField() + err := o.Unmarshal(pb2) + if err == nil { + t.Error("expected error, got no error") + } else if !strings.Contains(err.Error(), "bad wiretype") { + t.Error("expected bad wiretype error, got", err) + } +} + +func encodeDecode(t *testing.T, in, out Message, msg string) { + buf, err := Marshal(in) + if err != nil { + t.Fatalf("failed marshaling %v: %v", msg, err) + } + if err := Unmarshal(buf, out); err != nil { + t.Fatalf("failed unmarshaling %v: %v", msg, err) + } +} + +func TestPackedNonPackedDecoderSwitching(t *testing.T) { + np, p := new(NonPackedTest), new(PackedTest) + + // non-packed -> packed + np.A = []int32{0, 1, 1, 2, 3, 5} + encodeDecode(t, np, p, "non-packed -> packed") + if !reflect.DeepEqual(np.A, p.B) { + t.Errorf("failed non-packed -> packed; np.A=%+v, p.B=%+v", np.A, p.B) + } + + // packed -> non-packed + np.Reset() + p.B = []int32{3, 1, 4, 1, 5, 9} + encodeDecode(t, p, np, "packed -> non-packed") + if !reflect.DeepEqual(p.B, np.A) { + t.Errorf("failed packed -> non-packed; p.B=%+v, np.A=%+v", p.B, np.A) + } +} + +func TestProto1RepeatedGroup(t *testing.T) { + pb := &MessageList{ + Message: []*MessageList_Message{ + { + Name: String("blah"), + Count: Int32(7), + }, + // NOTE: pb.Message[1] is a nil + nil, + }, + } + + o := old() + err := o.Marshal(pb) + if err == nil || !strings.Contains(err.Error(), "repeated field Message has nil") { + t.Fatalf("unexpected or no error when marshaling: %v", err) + } +} + +// Test that enums work. Checks for a bug introduced by making enums +// named types instead of int32: newInt32FromUint64 would crash with +// a type mismatch in reflect.PointTo. +func TestEnum(t *testing.T) { + pb := new(GoEnum) + pb.Foo = FOO_FOO1.Enum() + o := old() + if err := o.Marshal(pb); err != nil { + t.Fatal("error encoding enum:", err) + } + pb1 := new(GoEnum) + if err := o.Unmarshal(pb1); err != nil { + t.Fatal("error decoding enum:", err) + } + if *pb1.Foo != FOO_FOO1 { + t.Error("expected 7 but got ", *pb1.Foo) + } +} + +// Enum types have String methods. Check that enum fields can be printed. +// We don't care what the value actually is, just as long as it doesn't crash. +func TestPrintingNilEnumFields(t *testing.T) { + pb := new(GoEnum) + _ = fmt.Sprintf("%+v", pb) +} + +// Verify that absent required fields cause Marshal/Unmarshal to return errors. +func TestRequiredFieldEnforcement(t *testing.T) { + pb := new(GoTestField) + _, err := Marshal(pb) + if err == nil { + t.Error("marshal: expected error, got nil") + } else if strings.Index(err.Error(), "Label") < 0 { + t.Errorf("marshal: bad error type: %v", err) + } + + // A slightly sneaky, yet valid, proto. It encodes the same required field twice, + // so simply counting the required fields is insufficient. + // field 1, encoding 2, value "hi" + buf := []byte("\x0A\x02hi\x0A\x02hi") + err = Unmarshal(buf, pb) + if err == nil { + t.Error("unmarshal: expected error, got nil") + } else if strings.Index(err.Error(), "{Unknown}") < 0 { + t.Errorf("unmarshal: bad error type: %v", err) + } +} + +func TestTypedNilMarshal(t *testing.T) { + // A typed nil should return ErrNil and not crash. + _, err := Marshal((*GoEnum)(nil)) + if err != ErrNil { + t.Errorf("Marshal: got err %v, want ErrNil", err) + } +} + +// A type that implements the Marshaler interface, but is not nillable. +type nonNillableInt uint64 + +func (nni nonNillableInt) Marshal() ([]byte, error) { + return EncodeVarint(uint64(nni)), nil +} + +type NNIMessage struct { + nni nonNillableInt +} + +func (*NNIMessage) Reset() {} +func (*NNIMessage) String() string { return "" } +func (*NNIMessage) ProtoMessage() {} + +// A type that implements the Marshaler interface and is nillable. +type nillableMessage struct { + x uint64 +} + +func (nm *nillableMessage) Marshal() ([]byte, error) { + return EncodeVarint(nm.x), nil +} + +type NMMessage struct { + nm *nillableMessage +} + +func (*NMMessage) Reset() {} +func (*NMMessage) String() string { return "" } +func (*NMMessage) ProtoMessage() {} + +// Verify a type that uses the Marshaler interface, but has a nil pointer. +func TestNilMarshaler(t *testing.T) { + // Try a struct with a Marshaler field that is nil. + // It should be directly marshable. + nmm := new(NMMessage) + if _, err := Marshal(nmm); err != nil { + t.Error("unexpected error marshaling nmm: ", err) + } + + // Try a struct with a Marshaler field that is not nillable. + nnim := new(NNIMessage) + nnim.nni = 7 + var _ Marshaler = nnim.nni // verify it is truly a Marshaler + if _, err := Marshal(nnim); err != nil { + t.Error("unexpected error marshaling nnim: ", err) + } +} + +func TestAllSetDefaults(t *testing.T) { + // Exercise SetDefaults with all scalar field types. + m := &Defaults{ + // NaN != NaN, so override that here. + F_Nan: Float32(1.7), + } + expected := &Defaults{ + F_Bool: Bool(true), + F_Int32: Int32(32), + F_Int64: Int64(64), + F_Fixed32: Uint32(320), + F_Fixed64: Uint64(640), + F_Uint32: Uint32(3200), + F_Uint64: Uint64(6400), + F_Float: Float32(314159), + F_Double: Float64(271828), + F_String: String(`hello, "world!"` + "\n"), + F_Bytes: []byte("Bignose"), + F_Sint32: Int32(-32), + F_Sint64: Int64(-64), + F_Enum: Defaults_GREEN.Enum(), + F_Pinf: Float32(float32(math.Inf(1))), + F_Ninf: Float32(float32(math.Inf(-1))), + F_Nan: Float32(1.7), + StrZero: String(""), + } + SetDefaults(m) + if !Equal(m, expected) { + t.Errorf("SetDefaults failed\n got %v\nwant %v", m, expected) + } +} + +func TestSetDefaultsWithSetField(t *testing.T) { + // Check that a set value is not overridden. + m := &Defaults{ + F_Int32: Int32(12), + } + SetDefaults(m) + if v := m.GetF_Int32(); v != 12 { + t.Errorf("m.FInt32 = %v, want 12", v) + } +} + +func TestSetDefaultsWithSubMessage(t *testing.T) { + m := &OtherMessage{ + Key: Int64(123), + Inner: &InnerMessage{ + Host: String("gopher"), + }, + } + expected := &OtherMessage{ + Key: Int64(123), + Inner: &InnerMessage{ + Host: String("gopher"), + Port: Int32(4000), + }, + } + SetDefaults(m) + if !Equal(m, expected) { + t.Errorf("\n got %v\nwant %v", m, expected) + } +} + +func TestSetDefaultsWithRepeatedSubMessage(t *testing.T) { + m := &MyMessage{ + RepInner: []*InnerMessage{{}}, + } + expected := &MyMessage{ + RepInner: []*InnerMessage{{ + Port: Int32(4000), + }}, + } + SetDefaults(m) + if !Equal(m, expected) { + t.Errorf("\n got %v\nwant %v", m, expected) + } +} + +func TestSetDefaultWithRepeatedNonMessage(t *testing.T) { + m := &MyMessage{ + Pet: []string{"turtle", "wombat"}, + } + expected := Clone(m) + SetDefaults(m) + if !Equal(m, expected) { + t.Errorf("\n got %v\nwant %v", m, expected) + } +} + +func TestMaximumTagNumber(t *testing.T) { + m := &MaxTag{ + LastField: String("natural goat essence"), + } + buf, err := Marshal(m) + if err != nil { + t.Fatalf("proto.Marshal failed: %v", err) + } + m2 := new(MaxTag) + if err := Unmarshal(buf, m2); err != nil { + t.Fatalf("proto.Unmarshal failed: %v", err) + } + if got, want := m2.GetLastField(), *m.LastField; got != want { + t.Errorf("got %q, want %q", got, want) + } +} + +func TestJSON(t *testing.T) { + m := &MyMessage{ + Count: Int32(4), + Pet: []string{"bunny", "kitty"}, + Inner: &InnerMessage{ + Host: String("cauchy"), + }, + Bikeshed: MyMessage_GREEN.Enum(), + } + const expected = `{"count":4,"pet":["bunny","kitty"],"inner":{"host":"cauchy"},"bikeshed":1}` + + b, err := json.Marshal(m) + if err != nil { + t.Fatalf("json.Marshal failed: %v", err) + } + s := string(b) + if s != expected { + t.Errorf("got %s\nwant %s", s, expected) + } + + received := new(MyMessage) + if err := json.Unmarshal(b, received); err != nil { + t.Fatalf("json.Unmarshal failed: %v", err) + } + if !Equal(received, m) { + t.Fatalf("got %s, want %s", received, m) + } + + // Test unmarshalling of JSON with symbolic enum name. + const old = `{"count":4,"pet":["bunny","kitty"],"inner":{"host":"cauchy"},"bikeshed":"GREEN"}` + received.Reset() + if err := json.Unmarshal([]byte(old), received); err != nil { + t.Fatalf("json.Unmarshal failed: %v", err) + } + if !Equal(received, m) { + t.Fatalf("got %s, want %s", received, m) + } +} + +func TestBadWireType(t *testing.T) { + b := []byte{7<<3 | 6} // field 7, wire type 6 + pb := new(OtherMessage) + if err := Unmarshal(b, pb); err == nil { + t.Errorf("Unmarshal did not fail") + } else if !strings.Contains(err.Error(), "unknown wire type") { + t.Errorf("wrong error: %v", err) + } +} + +func TestBytesWithInvalidLength(t *testing.T) { + // If a byte sequence has an invalid (negative) length, Unmarshal should not panic. + b := []byte{2<<3 | WireBytes, 0xff, 0xff, 0xff, 0xff, 0xff, 0} + Unmarshal(b, new(MyMessage)) +} + +func TestLengthOverflow(t *testing.T) { + // Overflowing a length should not panic. + b := []byte{2<<3 | WireBytes, 1, 1, 3<<3 | WireBytes, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x7f, 0x01} + Unmarshal(b, new(MyMessage)) +} + +func TestVarintOverflow(t *testing.T) { + // Overflowing a 64-bit length should not be allowed. + b := []byte{1<<3 | WireVarint, 0x01, 3<<3 | WireBytes, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80, 0x01} + if err := Unmarshal(b, new(MyMessage)); err == nil { + t.Fatalf("Overflowed uint64 length without error") + } +} + +func TestUnmarshalFuzz(t *testing.T) { + const N = 1000 + seed := time.Now().UnixNano() + t.Logf("RNG seed is %d", seed) + rng := rand.New(rand.NewSource(seed)) + buf := make([]byte, 20) + for i := 0; i < N; i++ { + for j := range buf { + buf[j] = byte(rng.Intn(256)) + } + fuzzUnmarshal(t, buf) + } +} + +func TestMergeMessages(t *testing.T) { + pb := &MessageList{Message: []*MessageList_Message{{Name: String("x"), Count: Int32(1)}}} + data, err := Marshal(pb) + if err != nil { + t.Fatalf("Marshal: %v", err) + } + + pb1 := new(MessageList) + if err := Unmarshal(data, pb1); err != nil { + t.Fatalf("first Unmarshal: %v", err) + } + if err := Unmarshal(data, pb1); err != nil { + t.Fatalf("second Unmarshal: %v", err) + } + if len(pb1.Message) != 1 { + t.Errorf("two Unmarshals produced %d Messages, want 1", len(pb1.Message)) + } + + pb2 := new(MessageList) + if err := UnmarshalMerge(data, pb2); err != nil { + t.Fatalf("first UnmarshalMerge: %v", err) + } + if err := UnmarshalMerge(data, pb2); err != nil { + t.Fatalf("second UnmarshalMerge: %v", err) + } + if len(pb2.Message) != 2 { + t.Errorf("two UnmarshalMerges produced %d Messages, want 2", len(pb2.Message)) + } +} + +func TestExtensionMarshalOrder(t *testing.T) { + m := &MyMessage{Count: Int(123)} + if err := SetExtension(m, E_Ext_More, &Ext{Data: String("alpha")}); err != nil { + t.Fatalf("SetExtension: %v", err) + } + if err := SetExtension(m, E_Ext_Text, String("aleph")); err != nil { + t.Fatalf("SetExtension: %v", err) + } + if err := SetExtension(m, E_Ext_Number, Int32(1)); err != nil { + t.Fatalf("SetExtension: %v", err) + } + + // Serialize m several times, and check we get the same bytes each time. + var orig []byte + for i := 0; i < 100; i++ { + b, err := Marshal(m) + if err != nil { + t.Fatalf("Marshal: %v", err) + } + if i == 0 { + orig = b + continue + } + if !bytes.Equal(b, orig) { + t.Errorf("Bytes differ on attempt #%d", i) + } + } +} + +// Many extensions, because small maps might not iterate differently on each iteration. +var exts = []*ExtensionDesc{ + E_X201, + E_X202, + E_X203, + E_X204, + E_X205, + E_X206, + E_X207, + E_X208, + E_X209, + E_X210, + E_X211, + E_X212, + E_X213, + E_X214, + E_X215, + E_X216, + E_X217, + E_X218, + E_X219, + E_X220, + E_X221, + E_X222, + E_X223, + E_X224, + E_X225, + E_X226, + E_X227, + E_X228, + E_X229, + E_X230, + E_X231, + E_X232, + E_X233, + E_X234, + E_X235, + E_X236, + E_X237, + E_X238, + E_X239, + E_X240, + E_X241, + E_X242, + E_X243, + E_X244, + E_X245, + E_X246, + E_X247, + E_X248, + E_X249, + E_X250, +} + +func TestMessageSetMarshalOrder(t *testing.T) { + m := &MyMessageSet{} + for _, x := range exts { + if err := SetExtension(m, x, &Empty{}); err != nil { + t.Fatalf("SetExtension: %v", err) + } + } + + buf, err := Marshal(m) + if err != nil { + t.Fatalf("Marshal: %v", err) + } + + // Serialize m several times, and check we get the same bytes each time. + for i := 0; i < 10; i++ { + b1, err := Marshal(m) + if err != nil { + t.Fatalf("Marshal: %v", err) + } + if !bytes.Equal(b1, buf) { + t.Errorf("Bytes differ on re-Marshal #%d", i) + } + + m2 := &MyMessageSet{} + if err := Unmarshal(buf, m2); err != nil { + t.Errorf("Unmarshal: %v", err) + } + b2, err := Marshal(m2) + if err != nil { + t.Errorf("re-Marshal: %v", err) + } + if !bytes.Equal(b2, buf) { + t.Errorf("Bytes differ on round-trip #%d", i) + } + } +} + +func TestUnmarshalMergesMessages(t *testing.T) { + // If a nested message occurs twice in the input, + // the fields should be merged when decoding. + a := &OtherMessage{ + Key: Int64(123), + Inner: &InnerMessage{ + Host: String("polhode"), + Port: Int32(1234), + }, + } + aData, err := Marshal(a) + if err != nil { + t.Fatalf("Marshal(a): %v", err) + } + b := &OtherMessage{ + Weight: Float32(1.2), + Inner: &InnerMessage{ + Host: String("herpolhode"), + Connected: Bool(true), + }, + } + bData, err := Marshal(b) + if err != nil { + t.Fatalf("Marshal(b): %v", err) + } + want := &OtherMessage{ + Key: Int64(123), + Weight: Float32(1.2), + Inner: &InnerMessage{ + Host: String("herpolhode"), + Port: Int32(1234), + Connected: Bool(true), + }, + } + got := new(OtherMessage) + if err := Unmarshal(append(aData, bData...), got); err != nil { + t.Fatalf("Unmarshal: %v", err) + } + if !Equal(got, want) { + t.Errorf("\n got %v\nwant %v", got, want) + } +} + +func TestEncodingSizes(t *testing.T) { + tests := []struct { + m Message + n int + }{ + {&Defaults{F_Int32: Int32(math.MaxInt32)}, 6}, + {&Defaults{F_Int32: Int32(math.MinInt32)}, 11}, + {&Defaults{F_Uint32: Uint32(uint32(math.MaxInt32) + 1)}, 6}, + {&Defaults{F_Uint32: Uint32(math.MaxUint32)}, 6}, + } + for _, test := range tests { + b, err := Marshal(test.m) + if err != nil { + t.Errorf("Marshal(%v): %v", test.m, err) + continue + } + if len(b) != test.n { + t.Errorf("Marshal(%v) yielded %d bytes, want %d bytes", test.m, len(b), test.n) + } + } +} + +func TestRequiredNotSetError(t *testing.T) { + pb := initGoTest(false) + pb.RequiredField.Label = nil + pb.F_Int32Required = nil + pb.F_Int64Required = nil + + expected := "0807" + // field 1, encoding 0, value 7 + "2206" + "120474797065" + // field 4, encoding 2 (GoTestField) + "5001" + // field 10, encoding 0, value 1 + "6d20000000" + // field 13, encoding 5, value 0x20 + "714000000000000000" + // field 14, encoding 1, value 0x40 + "78a019" + // field 15, encoding 0, value 0xca0 = 3232 + "8001c032" + // field 16, encoding 0, value 0x1940 = 6464 + "8d0100004a45" + // field 17, encoding 5, value 3232.0 + "9101000000000040b940" + // field 18, encoding 1, value 6464.0 + "9a0106" + "737472696e67" + // field 19, encoding 2, string "string" + "b304" + // field 70, encoding 3, start group + "ba0408" + "7265717569726564" + // field 71, encoding 2, string "required" + "b404" + // field 70, encoding 4, end group + "aa0605" + "6279746573" + // field 101, encoding 2, string "bytes" + "b0063f" + // field 102, encoding 0, 0x3f zigzag32 + "b8067f" // field 103, encoding 0, 0x7f zigzag64 + + o := old() + bytes, err := Marshal(pb) + if _, ok := err.(*RequiredNotSetError); !ok { + fmt.Printf("marshal-1 err = %v, want *RequiredNotSetError", err) + o.DebugPrint("", bytes) + t.Fatalf("expected = %s", expected) + } + if strings.Index(err.Error(), "RequiredField.Label") < 0 { + t.Errorf("marshal-1 wrong err msg: %v", err) + } + if !equal(bytes, expected, t) { + o.DebugPrint("neq 1", bytes) + t.Fatalf("expected = %s", expected) + } + + // Now test Unmarshal by recreating the original buffer. + pbd := new(GoTest) + err = Unmarshal(bytes, pbd) + if _, ok := err.(*RequiredNotSetError); !ok { + t.Fatalf("unmarshal err = %v, want *RequiredNotSetError", err) + o.DebugPrint("", bytes) + t.Fatalf("string = %s", expected) + } + if strings.Index(err.Error(), "RequiredField.{Unknown}") < 0 { + t.Errorf("unmarshal wrong err msg: %v", err) + } + bytes, err = Marshal(pbd) + if _, ok := err.(*RequiredNotSetError); !ok { + t.Errorf("marshal-2 err = %v, want *RequiredNotSetError", err) + o.DebugPrint("", bytes) + t.Fatalf("string = %s", expected) + } + if strings.Index(err.Error(), "RequiredField.Label") < 0 { + t.Errorf("marshal-2 wrong err msg: %v", err) + } + if !equal(bytes, expected, t) { + o.DebugPrint("neq 2", bytes) + t.Fatalf("string = %s", expected) + } +} + +func fuzzUnmarshal(t *testing.T, data []byte) { + defer func() { + if e := recover(); e != nil { + t.Errorf("These bytes caused a panic: %+v", data) + t.Logf("Stack:\n%s", debug.Stack()) + t.FailNow() + } + }() + + pb := new(MyMessage) + Unmarshal(data, pb) +} + +func TestMapFieldMarshal(t *testing.T) { + m := &MessageWithMap{ + NameMapping: map[int32]string{ + 1: "Rob", + 4: "Ian", + 8: "Dave", + }, + } + b, err := Marshal(m) + if err != nil { + t.Fatalf("Marshal: %v", err) + } + + // b should be the concatenation of these three byte sequences in some order. + parts := []string{ + "\n\a\b\x01\x12\x03Rob", + "\n\a\b\x04\x12\x03Ian", + "\n\b\b\x08\x12\x04Dave", + } + ok := false + for i := range parts { + for j := range parts { + if j == i { + continue + } + for k := range parts { + if k == i || k == j { + continue + } + try := parts[i] + parts[j] + parts[k] + if bytes.Equal(b, []byte(try)) { + ok = true + break + } + } + } + } + if !ok { + t.Fatalf("Incorrect Marshal output.\n got %q\nwant %q (or a permutation of that)", b, parts[0]+parts[1]+parts[2]) + } + t.Logf("FYI b: %q", b) + + (new(Buffer)).DebugPrint("Dump of b", b) +} + +func TestMapFieldRoundTrips(t *testing.T) { + m := &MessageWithMap{ + NameMapping: map[int32]string{ + 1: "Rob", + 4: "Ian", + 8: "Dave", + }, + MsgMapping: map[int64]*FloatingPoint{ + 0x7001: {F: Float64(2.0)}, + }, + ByteMapping: map[bool][]byte{ + false: []byte("that's not right!"), + true: []byte("aye, 'tis true!"), + }, + } + b, err := Marshal(m) + if err != nil { + t.Fatalf("Marshal: %v", err) + } + t.Logf("FYI b: %q", b) + m2 := new(MessageWithMap) + if err := Unmarshal(b, m2); err != nil { + t.Fatalf("Unmarshal: %v", err) + } + for _, pair := range [][2]interface{}{ + {m.NameMapping, m2.NameMapping}, + {m.MsgMapping, m2.MsgMapping}, + {m.ByteMapping, m2.ByteMapping}, + } { + if !reflect.DeepEqual(pair[0], pair[1]) { + t.Errorf("Map did not survive a round trip.\ninitial: %v\n final: %v", pair[0], pair[1]) + } + } +} + +func TestMapFieldWithNil(t *testing.T) { + m := &MessageWithMap{ + MsgMapping: map[int64]*FloatingPoint{ + 1: nil, + }, + } + b, err := Marshal(m) + if err == nil { + t.Fatalf("Marshal of bad map should have failed, got these bytes: %v", b) + } +} + +// Benchmarks + +func testMsg() *GoTest { + pb := initGoTest(true) + const N = 1000 // Internally the library starts much smaller. + pb.F_Int32Repeated = make([]int32, N) + pb.F_DoubleRepeated = make([]float64, N) + for i := 0; i < N; i++ { + pb.F_Int32Repeated[i] = int32(i) + pb.F_DoubleRepeated[i] = float64(i) + } + return pb +} + +func bytesMsg() *GoTest { + pb := initGoTest(true) + buf := make([]byte, 4000) + for i := range buf { + buf[i] = byte(i) + } + pb.F_BytesDefaulted = buf + return pb +} + +func benchmarkMarshal(b *testing.B, pb Message, marshal func(Message) ([]byte, error)) { + d, _ := marshal(pb) + b.SetBytes(int64(len(d))) + b.ResetTimer() + for i := 0; i < b.N; i++ { + marshal(pb) + } +} + +func benchmarkBufferMarshal(b *testing.B, pb Message) { + p := NewBuffer(nil) + benchmarkMarshal(b, pb, func(pb0 Message) ([]byte, error) { + p.Reset() + err := p.Marshal(pb0) + return p.Bytes(), err + }) +} + +func benchmarkSize(b *testing.B, pb Message) { + benchmarkMarshal(b, pb, func(pb0 Message) ([]byte, error) { + Size(pb) + return nil, nil + }) +} + +func newOf(pb Message) Message { + in := reflect.ValueOf(pb) + if in.IsNil() { + return pb + } + return reflect.New(in.Type().Elem()).Interface().(Message) +} + +func benchmarkUnmarshal(b *testing.B, pb Message, unmarshal func([]byte, Message) error) { + d, _ := Marshal(pb) + b.SetBytes(int64(len(d))) + pbd := newOf(pb) + + b.ResetTimer() + for i := 0; i < b.N; i++ { + unmarshal(d, pbd) + } +} + +func benchmarkBufferUnmarshal(b *testing.B, pb Message) { + p := NewBuffer(nil) + benchmarkUnmarshal(b, pb, func(d []byte, pb0 Message) error { + p.SetBuf(d) + return p.Unmarshal(pb0) + }) +} + +// Benchmark{Marshal,BufferMarshal,Size,Unmarshal,BufferUnmarshal}{,Bytes} + +func BenchmarkMarshal(b *testing.B) { + benchmarkMarshal(b, testMsg(), Marshal) +} + +func BenchmarkBufferMarshal(b *testing.B) { + benchmarkBufferMarshal(b, testMsg()) +} + +func BenchmarkSize(b *testing.B) { + benchmarkSize(b, testMsg()) +} + +func BenchmarkUnmarshal(b *testing.B) { + benchmarkUnmarshal(b, testMsg(), Unmarshal) +} + +func BenchmarkBufferUnmarshal(b *testing.B) { + benchmarkBufferUnmarshal(b, testMsg()) +} + +func BenchmarkMarshalBytes(b *testing.B) { + benchmarkMarshal(b, bytesMsg(), Marshal) +} + +func BenchmarkBufferMarshalBytes(b *testing.B) { + benchmarkBufferMarshal(b, bytesMsg()) +} + +func BenchmarkSizeBytes(b *testing.B) { + benchmarkSize(b, bytesMsg()) +} + +func BenchmarkUnmarshalBytes(b *testing.B) { + benchmarkUnmarshal(b, bytesMsg(), Unmarshal) +} + +func BenchmarkBufferUnmarshalBytes(b *testing.B) { + benchmarkBufferUnmarshal(b, bytesMsg()) +} + +func BenchmarkUnmarshalUnrecognizedFields(b *testing.B) { + b.StopTimer() + pb := initGoTestField() + skip := &GoSkipTest{ + SkipInt32: Int32(32), + SkipFixed32: Uint32(3232), + SkipFixed64: Uint64(6464), + SkipString: String("skipper"), + Skipgroup: &GoSkipTest_SkipGroup{ + GroupInt32: Int32(75), + GroupString: String("wxyz"), + }, + } + + pbd := new(GoTestField) + p := NewBuffer(nil) + p.Marshal(pb) + p.Marshal(skip) + p2 := NewBuffer(nil) + + b.StartTimer() + for i := 0; i < b.N; i++ { + p2.SetBuf(p.Bytes()) + p2.Unmarshal(pbd) + } +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/clone.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/clone.go new file mode 100644 index 0000000000000..57297947beb25 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/clone.go @@ -0,0 +1,217 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2011 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// Protocol buffer deep copy and merge. +// TODO: MessageSet and RawMessage. + +package proto + +import ( + "log" + "reflect" + "strings" +) + +// Clone returns a deep copy of a protocol buffer. +func Clone(pb Message) Message { + in := reflect.ValueOf(pb) + if in.IsNil() { + return pb + } + + out := reflect.New(in.Type().Elem()) + // out is empty so a merge is a deep copy. + mergeStruct(out.Elem(), in.Elem()) + return out.Interface().(Message) +} + +// Merge merges src into dst. +// Required and optional fields that are set in src will be set to that value in dst. +// Elements of repeated fields will be appended. +// Merge panics if src and dst are not the same type, or if dst is nil. +func Merge(dst, src Message) { + in := reflect.ValueOf(src) + out := reflect.ValueOf(dst) + if out.IsNil() { + panic("proto: nil destination") + } + if in.Type() != out.Type() { + // Explicit test prior to mergeStruct so that mistyped nils will fail + panic("proto: type mismatch") + } + if in.IsNil() { + // Merging nil into non-nil is a quiet no-op + return + } + mergeStruct(out.Elem(), in.Elem()) +} + +func mergeStruct(out, in reflect.Value) { + sprop := GetProperties(in.Type()) + for i := 0; i < in.NumField(); i++ { + f := in.Type().Field(i) + if strings.HasPrefix(f.Name, "XXX_") { + continue + } + mergeAny(out.Field(i), in.Field(i), false, sprop.Prop[i]) + } + + if emIn, ok := in.Addr().Interface().(extensionsMap); ok { + emOut := out.Addr().Interface().(extensionsMap) + mergeExtension(emOut.ExtensionMap(), emIn.ExtensionMap()) + } else if emIn, ok := in.Addr().Interface().(extensionsBytes); ok { + emOut := out.Addr().Interface().(extensionsBytes) + bIn := emIn.GetExtensions() + bOut := emOut.GetExtensions() + *bOut = append(*bOut, *bIn...) + } + + uf := in.FieldByName("XXX_unrecognized") + if !uf.IsValid() { + return + } + uin := uf.Bytes() + if len(uin) > 0 { + out.FieldByName("XXX_unrecognized").SetBytes(append([]byte(nil), uin...)) + } +} + +// mergeAny performs a merge between two values of the same type. +// viaPtr indicates whether the values were indirected through a pointer (implying proto2). +// prop is set if this is a struct field (it may be nil). +func mergeAny(out, in reflect.Value, viaPtr bool, prop *Properties) { + if in.Type() == protoMessageType { + if !in.IsNil() { + if out.IsNil() { + out.Set(reflect.ValueOf(Clone(in.Interface().(Message)))) + } else { + Merge(out.Interface().(Message), in.Interface().(Message)) + } + } + return + } + switch in.Kind() { + case reflect.Bool, reflect.Float32, reflect.Float64, reflect.Int32, reflect.Int64, + reflect.String, reflect.Uint32, reflect.Uint64: + if !viaPtr && isProto3Zero(in) { + return + } + out.Set(in) + case reflect.Map: + if in.Len() == 0 { + return + } + if out.IsNil() { + out.Set(reflect.MakeMap(in.Type())) + } + // For maps with value types of *T or []byte we need to deep copy each value. + elemKind := in.Type().Elem().Kind() + for _, key := range in.MapKeys() { + var val reflect.Value + switch elemKind { + case reflect.Ptr: + val = reflect.New(in.Type().Elem().Elem()) + mergeAny(val, in.MapIndex(key), false, nil) + case reflect.Slice: + val = in.MapIndex(key) + val = reflect.ValueOf(append([]byte{}, val.Bytes()...)) + default: + val = in.MapIndex(key) + } + out.SetMapIndex(key, val) + } + case reflect.Ptr: + if in.IsNil() { + return + } + if out.IsNil() { + out.Set(reflect.New(in.Elem().Type())) + } + mergeAny(out.Elem(), in.Elem(), true, nil) + case reflect.Slice: + if in.IsNil() { + return + } + if in.Type().Elem().Kind() == reflect.Uint8 { + // []byte is a scalar bytes field, not a repeated field. + + // Edge case: if this is in a proto3 message, a zero length + // bytes field is considered the zero value, and should not + // be merged. + if prop != nil && prop.proto3 && in.Len() == 0 { + return + } + + // Make a deep copy. + // Append to []byte{} instead of []byte(nil) so that we never end up + // with a nil result. + out.SetBytes(append([]byte{}, in.Bytes()...)) + return + } + n := in.Len() + if out.IsNil() { + out.Set(reflect.MakeSlice(in.Type(), 0, n)) + } + switch in.Type().Elem().Kind() { + case reflect.Bool, reflect.Float32, reflect.Float64, reflect.Int32, reflect.Int64, + reflect.String, reflect.Uint32, reflect.Uint64: + out.Set(reflect.AppendSlice(out, in)) + default: + for i := 0; i < n; i++ { + x := reflect.Indirect(reflect.New(in.Type().Elem())) + mergeAny(x, in.Index(i), false, nil) + out.Set(reflect.Append(out, x)) + } + } + case reflect.Struct: + mergeStruct(out, in) + default: + // unknown type, so not a protocol buffer + log.Printf("proto: don't know how to copy %v", in) + } +} + +func mergeExtension(out, in map[int32]Extension) { + for extNum, eIn := range in { + eOut := Extension{desc: eIn.desc} + if eIn.value != nil { + v := reflect.New(reflect.TypeOf(eIn.value)).Elem() + mergeAny(v, reflect.ValueOf(eIn.value), false, nil) + eOut.value = v.Interface() + } + if eIn.enc != nil { + eOut.enc = make([]byte, len(eIn.enc)) + copy(eOut.enc, eIn.enc) + } + + out[extNum] = eOut + } +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/clone_test.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/clone_test.go new file mode 100644 index 0000000000000..7eef89ee09e16 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/clone_test.go @@ -0,0 +1,245 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2011 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "testing" + + "github.com/gogo/protobuf/proto" + + proto3pb "github.com/gogo/protobuf/proto/proto3_proto" + pb "github.com/gogo/protobuf/proto/testdata" +) + +var cloneTestMessage = &pb.MyMessage{ + Count: proto.Int32(42), + Name: proto.String("Dave"), + Pet: []string{"bunny", "kitty", "horsey"}, + Inner: &pb.InnerMessage{ + Host: proto.String("niles"), + Port: proto.Int32(9099), + Connected: proto.Bool(true), + }, + Others: []*pb.OtherMessage{ + { + Value: []byte("some bytes"), + }, + }, + Somegroup: &pb.MyMessage_SomeGroup{ + GroupField: proto.Int32(6), + }, + RepBytes: [][]byte{[]byte("sham"), []byte("wow")}, +} + +func init() { + ext := &pb.Ext{ + Data: proto.String("extension"), + } + if err := proto.SetExtension(cloneTestMessage, pb.E_Ext_More, ext); err != nil { + panic("SetExtension: " + err.Error()) + } +} + +func TestClone(t *testing.T) { + m := proto.Clone(cloneTestMessage).(*pb.MyMessage) + if !proto.Equal(m, cloneTestMessage) { + t.Errorf("Clone(%v) = %v", cloneTestMessage, m) + } + + // Verify it was a deep copy. + *m.Inner.Port++ + if proto.Equal(m, cloneTestMessage) { + t.Error("Mutating clone changed the original") + } + // Byte fields and repeated fields should be copied. + if &m.Pet[0] == &cloneTestMessage.Pet[0] { + t.Error("Pet: repeated field not copied") + } + if &m.Others[0] == &cloneTestMessage.Others[0] { + t.Error("Others: repeated field not copied") + } + if &m.Others[0].Value[0] == &cloneTestMessage.Others[0].Value[0] { + t.Error("Others[0].Value: bytes field not copied") + } + if &m.RepBytes[0] == &cloneTestMessage.RepBytes[0] { + t.Error("RepBytes: repeated field not copied") + } + if &m.RepBytes[0][0] == &cloneTestMessage.RepBytes[0][0] { + t.Error("RepBytes[0]: bytes field not copied") + } +} + +func TestCloneNil(t *testing.T) { + var m *pb.MyMessage + if c := proto.Clone(m); !proto.Equal(m, c) { + t.Errorf("Clone(%v) = %v", m, c) + } +} + +var mergeTests = []struct { + src, dst, want proto.Message +}{ + { + src: &pb.MyMessage{ + Count: proto.Int32(42), + }, + dst: &pb.MyMessage{ + Name: proto.String("Dave"), + }, + want: &pb.MyMessage{ + Count: proto.Int32(42), + Name: proto.String("Dave"), + }, + }, + { + src: &pb.MyMessage{ + Inner: &pb.InnerMessage{ + Host: proto.String("hey"), + Connected: proto.Bool(true), + }, + Pet: []string{"horsey"}, + Others: []*pb.OtherMessage{ + { + Value: []byte("some bytes"), + }, + }, + }, + dst: &pb.MyMessage{ + Inner: &pb.InnerMessage{ + Host: proto.String("niles"), + Port: proto.Int32(9099), + }, + Pet: []string{"bunny", "kitty"}, + Others: []*pb.OtherMessage{ + { + Key: proto.Int64(31415926535), + }, + { + // Explicitly test a src=nil field + Inner: nil, + }, + }, + }, + want: &pb.MyMessage{ + Inner: &pb.InnerMessage{ + Host: proto.String("hey"), + Connected: proto.Bool(true), + Port: proto.Int32(9099), + }, + Pet: []string{"bunny", "kitty", "horsey"}, + Others: []*pb.OtherMessage{ + { + Key: proto.Int64(31415926535), + }, + {}, + { + Value: []byte("some bytes"), + }, + }, + }, + }, + { + src: &pb.MyMessage{ + RepBytes: [][]byte{[]byte("wow")}, + }, + dst: &pb.MyMessage{ + Somegroup: &pb.MyMessage_SomeGroup{ + GroupField: proto.Int32(6), + }, + RepBytes: [][]byte{[]byte("sham")}, + }, + want: &pb.MyMessage{ + Somegroup: &pb.MyMessage_SomeGroup{ + GroupField: proto.Int32(6), + }, + RepBytes: [][]byte{[]byte("sham"), []byte("wow")}, + }, + }, + // Check that a scalar bytes field replaces rather than appends. + { + src: &pb.OtherMessage{Value: []byte("foo")}, + dst: &pb.OtherMessage{Value: []byte("bar")}, + want: &pb.OtherMessage{Value: []byte("foo")}, + }, + { + src: &pb.MessageWithMap{ + NameMapping: map[int32]string{6: "Nigel"}, + MsgMapping: map[int64]*pb.FloatingPoint{ + 0x4001: {F: proto.Float64(2.0)}, + }, + ByteMapping: map[bool][]byte{true: []byte("wowsa")}, + }, + dst: &pb.MessageWithMap{ + NameMapping: map[int32]string{ + 6: "Bruce", // should be overwritten + 7: "Andrew", + }, + }, + want: &pb.MessageWithMap{ + NameMapping: map[int32]string{ + 6: "Nigel", + 7: "Andrew", + }, + MsgMapping: map[int64]*pb.FloatingPoint{ + 0x4001: {F: proto.Float64(2.0)}, + }, + ByteMapping: map[bool][]byte{true: []byte("wowsa")}, + }, + }, + // proto3 shouldn't merge zero values, + // in the same way that proto2 shouldn't merge nils. + { + src: &proto3pb.Message{ + Name: "Aaron", + Data: []byte(""), // zero value, but not nil + }, + dst: &proto3pb.Message{ + HeightInCm: 176, + Data: []byte("texas!"), + }, + want: &proto3pb.Message{ + Name: "Aaron", + HeightInCm: 176, + Data: []byte("texas!"), + }, + }, +} + +func TestMerge(t *testing.T) { + for _, m := range mergeTests { + got := proto.Clone(m.dst) + proto.Merge(got, m.src) + if !proto.Equal(got, m.want) { + t.Errorf("Merge(%v, %v)\n got %v\nwant %v\n", m.dst, m.src, got, m.want) + } + } +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/decode.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/decode.go new file mode 100644 index 0000000000000..f7b1884b3cb34 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/decode.go @@ -0,0 +1,832 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +/* + * Routines for decoding protocol buffer data to construct in-memory representations. + */ + +import ( + "errors" + "fmt" + "io" + "os" + "reflect" +) + +// errOverflow is returned when an integer is too large to be represented. +var errOverflow = errors.New("proto: integer overflow") + +// The fundamental decoders that interpret bytes on the wire. +// Those that take integer types all return uint64 and are +// therefore of type valueDecoder. + +// DecodeVarint reads a varint-encoded integer from the slice. +// It returns the integer and the number of bytes consumed, or +// zero if there is not enough. +// This is the format for the +// int32, int64, uint32, uint64, bool, and enum +// protocol buffer types. +func DecodeVarint(buf []byte) (x uint64, n int) { + // x, n already 0 + for shift := uint(0); shift < 64; shift += 7 { + if n >= len(buf) { + return 0, 0 + } + b := uint64(buf[n]) + n++ + x |= (b & 0x7F) << shift + if (b & 0x80) == 0 { + return x, n + } + } + + // The number is too large to represent in a 64-bit value. + return 0, 0 +} + +// DecodeVarint reads a varint-encoded integer from the Buffer. +// This is the format for the +// int32, int64, uint32, uint64, bool, and enum +// protocol buffer types. +func (p *Buffer) DecodeVarint() (x uint64, err error) { + // x, err already 0 + + i := p.index + l := len(p.buf) + + for shift := uint(0); shift < 64; shift += 7 { + if i >= l { + err = io.ErrUnexpectedEOF + return + } + b := p.buf[i] + i++ + x |= (uint64(b) & 0x7F) << shift + if b < 0x80 { + p.index = i + return + } + } + + // The number is too large to represent in a 64-bit value. + err = errOverflow + return +} + +// DecodeFixed64 reads a 64-bit integer from the Buffer. +// This is the format for the +// fixed64, sfixed64, and double protocol buffer types. +func (p *Buffer) DecodeFixed64() (x uint64, err error) { + // x, err already 0 + i := p.index + 8 + if i < 0 || i > len(p.buf) { + err = io.ErrUnexpectedEOF + return + } + p.index = i + + x = uint64(p.buf[i-8]) + x |= uint64(p.buf[i-7]) << 8 + x |= uint64(p.buf[i-6]) << 16 + x |= uint64(p.buf[i-5]) << 24 + x |= uint64(p.buf[i-4]) << 32 + x |= uint64(p.buf[i-3]) << 40 + x |= uint64(p.buf[i-2]) << 48 + x |= uint64(p.buf[i-1]) << 56 + return +} + +// DecodeFixed32 reads a 32-bit integer from the Buffer. +// This is the format for the +// fixed32, sfixed32, and float protocol buffer types. +func (p *Buffer) DecodeFixed32() (x uint64, err error) { + // x, err already 0 + i := p.index + 4 + if i < 0 || i > len(p.buf) { + err = io.ErrUnexpectedEOF + return + } + p.index = i + + x = uint64(p.buf[i-4]) + x |= uint64(p.buf[i-3]) << 8 + x |= uint64(p.buf[i-2]) << 16 + x |= uint64(p.buf[i-1]) << 24 + return +} + +// DecodeZigzag64 reads a zigzag-encoded 64-bit integer +// from the Buffer. +// This is the format used for the sint64 protocol buffer type. +func (p *Buffer) DecodeZigzag64() (x uint64, err error) { + x, err = p.DecodeVarint() + if err != nil { + return + } + x = (x >> 1) ^ uint64((int64(x&1)<<63)>>63) + return +} + +// DecodeZigzag32 reads a zigzag-encoded 32-bit integer +// from the Buffer. +// This is the format used for the sint32 protocol buffer type. +func (p *Buffer) DecodeZigzag32() (x uint64, err error) { + x, err = p.DecodeVarint() + if err != nil { + return + } + x = uint64((uint32(x) >> 1) ^ uint32((int32(x&1)<<31)>>31)) + return +} + +// These are not ValueDecoders: they produce an array of bytes or a string. +// bytes, embedded messages + +// DecodeRawBytes reads a count-delimited byte buffer from the Buffer. +// This is the format used for the bytes protocol buffer +// type and for embedded messages. +func (p *Buffer) DecodeRawBytes(alloc bool) (buf []byte, err error) { + n, err := p.DecodeVarint() + if err != nil { + return nil, err + } + + nb := int(n) + if nb < 0 { + return nil, fmt.Errorf("proto: bad byte length %d", nb) + } + end := p.index + nb + if end < p.index || end > len(p.buf) { + return nil, io.ErrUnexpectedEOF + } + + if !alloc { + // todo: check if can get more uses of alloc=false + buf = p.buf[p.index:end] + p.index += nb + return + } + + buf = make([]byte, nb) + copy(buf, p.buf[p.index:]) + p.index += nb + return +} + +// DecodeStringBytes reads an encoded string from the Buffer. +// This is the format used for the proto2 string type. +func (p *Buffer) DecodeStringBytes() (s string, err error) { + buf, err := p.DecodeRawBytes(false) + if err != nil { + return + } + return string(buf), nil +} + +// Skip the next item in the buffer. Its wire type is decoded and presented as an argument. +// If the protocol buffer has extensions, and the field matches, add it as an extension. +// Otherwise, if the XXX_unrecognized field exists, append the skipped data there. +func (o *Buffer) skipAndSave(t reflect.Type, tag, wire int, base structPointer, unrecField field) error { + oi := o.index + + err := o.skip(t, tag, wire) + if err != nil { + return err + } + + if !unrecField.IsValid() { + return nil + } + + ptr := structPointer_Bytes(base, unrecField) + + // Add the skipped field to struct field + obuf := o.buf + + o.buf = *ptr + o.EncodeVarint(uint64(tag<<3 | wire)) + *ptr = append(o.buf, obuf[oi:o.index]...) + + o.buf = obuf + + return nil +} + +// Skip the next item in the buffer. Its wire type is decoded and presented as an argument. +func (o *Buffer) skip(t reflect.Type, tag, wire int) error { + + var u uint64 + var err error + + switch wire { + case WireVarint: + _, err = o.DecodeVarint() + case WireFixed64: + _, err = o.DecodeFixed64() + case WireBytes: + _, err = o.DecodeRawBytes(false) + case WireFixed32: + _, err = o.DecodeFixed32() + case WireStartGroup: + for { + u, err = o.DecodeVarint() + if err != nil { + break + } + fwire := int(u & 0x7) + if fwire == WireEndGroup { + break + } + ftag := int(u >> 3) + err = o.skip(t, ftag, fwire) + if err != nil { + break + } + } + default: + err = fmt.Errorf("proto: can't skip unknown wire type %d for %s", wire, t) + } + return err +} + +// Unmarshaler is the interface representing objects that can +// unmarshal themselves. The method should reset the receiver before +// decoding starts. The argument points to data that may be +// overwritten, so implementations should not keep references to the +// buffer. +type Unmarshaler interface { + Unmarshal([]byte) error +} + +// Unmarshal parses the protocol buffer representation in buf and places the +// decoded result in pb. If the struct underlying pb does not match +// the data in buf, the results can be unpredictable. +// +// Unmarshal resets pb before starting to unmarshal, so any +// existing data in pb is always removed. Use UnmarshalMerge +// to preserve and append to existing data. +func Unmarshal(buf []byte, pb Message) error { + pb.Reset() + return UnmarshalMerge(buf, pb) +} + +// UnmarshalMerge parses the protocol buffer representation in buf and +// writes the decoded result to pb. If the struct underlying pb does not match +// the data in buf, the results can be unpredictable. +// +// UnmarshalMerge merges into existing data in pb. +// Most code should use Unmarshal instead. +func UnmarshalMerge(buf []byte, pb Message) error { + // If the object can unmarshal itself, let it. + if u, ok := pb.(Unmarshaler); ok { + return u.Unmarshal(buf) + } + return NewBuffer(buf).Unmarshal(pb) +} + +// Unmarshal parses the protocol buffer representation in the +// Buffer and places the decoded result in pb. If the struct +// underlying pb does not match the data in the buffer, the results can be +// unpredictable. +func (p *Buffer) Unmarshal(pb Message) error { + // If the object can unmarshal itself, let it. + if u, ok := pb.(Unmarshaler); ok { + err := u.Unmarshal(p.buf[p.index:]) + p.index = len(p.buf) + return err + } + + typ, base, err := getbase(pb) + if err != nil { + return err + } + + err = p.unmarshalType(typ.Elem(), GetProperties(typ.Elem()), false, base) + + if collectStats { + stats.Decode++ + } + + return err +} + +// unmarshalType does the work of unmarshaling a structure. +func (o *Buffer) unmarshalType(st reflect.Type, prop *StructProperties, is_group bool, base structPointer) error { + var state errorState + required, reqFields := prop.reqCount, uint64(0) + + var err error + for err == nil && o.index < len(o.buf) { + oi := o.index + var u uint64 + u, err = o.DecodeVarint() + if err != nil { + break + } + wire := int(u & 0x7) + if wire == WireEndGroup { + if is_group { + return nil // input is satisfied + } + return fmt.Errorf("proto: %s: wiretype end group for non-group", st) + } + tag := int(u >> 3) + if tag <= 0 { + return fmt.Errorf("proto: %s: illegal tag %d (wire type %d)", st, tag, wire) + } + fieldnum, ok := prop.decoderTags.get(tag) + if !ok { + // Maybe it's an extension? + if prop.extendable { + if e := structPointer_Interface(base, st).(extendableProto); isExtensionField(e, int32(tag)) { + if err = o.skip(st, tag, wire); err == nil { + if ee, ok := e.(extensionsMap); ok { + ext := ee.ExtensionMap()[int32(tag)] // may be missing + ext.enc = append(ext.enc, o.buf[oi:o.index]...) + ee.ExtensionMap()[int32(tag)] = ext + } else if ee, ok := e.(extensionsBytes); ok { + ext := ee.GetExtensions() + *ext = append(*ext, o.buf[oi:o.index]...) + } + } + continue + } + } + err = o.skipAndSave(st, tag, wire, base, prop.unrecField) + continue + } + p := prop.Prop[fieldnum] + + if p.dec == nil { + fmt.Fprintf(os.Stderr, "proto: no protobuf decoder for %s.%s\n", st, st.Field(fieldnum).Name) + continue + } + dec := p.dec + if wire != WireStartGroup && wire != p.WireType { + if wire == WireBytes && p.packedDec != nil { + // a packable field + dec = p.packedDec + } else { + err = fmt.Errorf("proto: bad wiretype for field %s.%s: got wiretype %d, want %d", st, st.Field(fieldnum).Name, wire, p.WireType) + continue + } + } + decErr := dec(o, p, base) + if decErr != nil && !state.shouldContinue(decErr, p) { + err = decErr + } + if err == nil && p.Required { + // Successfully decoded a required field. + if tag <= 64 { + // use bitmap for fields 1-64 to catch field reuse. + var mask uint64 = 1 << uint64(tag-1) + if reqFields&mask == 0 { + // new required field + reqFields |= mask + required-- + } + } else { + // This is imprecise. It can be fooled by a required field + // with a tag > 64 that is encoded twice; that's very rare. + // A fully correct implementation would require allocating + // a data structure, which we would like to avoid. + required-- + } + } + } + if err == nil { + if is_group { + return io.ErrUnexpectedEOF + } + if state.err != nil { + return state.err + } + if required > 0 { + // Not enough information to determine the exact field. If we use extra + // CPU, we could determine the field only if the missing required field + // has a tag <= 64 and we check reqFields. + return &RequiredNotSetError{"{Unknown}"} + } + } + return err +} + +// Individual type decoders +// For each, +// u is the decoded value, +// v is a pointer to the field (pointer) in the struct + +// Sizes of the pools to allocate inside the Buffer. +// The goal is modest amortization and allocation +// on at least 16-byte boundaries. +const ( + boolPoolSize = 16 + uint32PoolSize = 8 + uint64PoolSize = 4 +) + +// Decode a bool. +func (o *Buffer) dec_bool(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + if len(o.bools) == 0 { + o.bools = make([]bool, boolPoolSize) + } + o.bools[0] = u != 0 + *structPointer_Bool(base, p.field) = &o.bools[0] + o.bools = o.bools[1:] + return nil +} + +func (o *Buffer) dec_proto3_bool(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + *structPointer_BoolVal(base, p.field) = u != 0 + return nil +} + +// Decode an int32. +func (o *Buffer) dec_int32(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + word32_Set(structPointer_Word32(base, p.field), o, uint32(u)) + return nil +} + +func (o *Buffer) dec_proto3_int32(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + word32Val_Set(structPointer_Word32Val(base, p.field), uint32(u)) + return nil +} + +// Decode an int64. +func (o *Buffer) dec_int64(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + word64_Set(structPointer_Word64(base, p.field), o, u) + return nil +} + +func (o *Buffer) dec_proto3_int64(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + word64Val_Set(structPointer_Word64Val(base, p.field), o, u) + return nil +} + +// Decode a string. +func (o *Buffer) dec_string(p *Properties, base structPointer) error { + s, err := o.DecodeStringBytes() + if err != nil { + return err + } + *structPointer_String(base, p.field) = &s + return nil +} + +func (o *Buffer) dec_proto3_string(p *Properties, base structPointer) error { + s, err := o.DecodeStringBytes() + if err != nil { + return err + } + *structPointer_StringVal(base, p.field) = s + return nil +} + +// Decode a slice of bytes ([]byte). +func (o *Buffer) dec_slice_byte(p *Properties, base structPointer) error { + b, err := o.DecodeRawBytes(true) + if err != nil { + return err + } + *structPointer_Bytes(base, p.field) = b + return nil +} + +// Decode a slice of bools ([]bool). +func (o *Buffer) dec_slice_bool(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + v := structPointer_BoolSlice(base, p.field) + *v = append(*v, u != 0) + return nil +} + +// Decode a slice of bools ([]bool) in packed format. +func (o *Buffer) dec_slice_packed_bool(p *Properties, base structPointer) error { + v := structPointer_BoolSlice(base, p.field) + + nn, err := o.DecodeVarint() + if err != nil { + return err + } + nb := int(nn) // number of bytes of encoded bools + + y := *v + for i := 0; i < nb; i++ { + u, err := p.valDec(o) + if err != nil { + return err + } + y = append(y, u != 0) + } + + *v = y + return nil +} + +// Decode a slice of int32s ([]int32). +func (o *Buffer) dec_slice_int32(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + structPointer_Word32Slice(base, p.field).Append(uint32(u)) + return nil +} + +// Decode a slice of int32s ([]int32) in packed format. +func (o *Buffer) dec_slice_packed_int32(p *Properties, base structPointer) error { + v := structPointer_Word32Slice(base, p.field) + + nn, err := o.DecodeVarint() + if err != nil { + return err + } + nb := int(nn) // number of bytes of encoded int32s + + fin := o.index + nb + if fin < o.index { + return errOverflow + } + for o.index < fin { + u, err := p.valDec(o) + if err != nil { + return err + } + v.Append(uint32(u)) + } + return nil +} + +// Decode a slice of int64s ([]int64). +func (o *Buffer) dec_slice_int64(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + + structPointer_Word64Slice(base, p.field).Append(u) + return nil +} + +// Decode a slice of int64s ([]int64) in packed format. +func (o *Buffer) dec_slice_packed_int64(p *Properties, base structPointer) error { + v := structPointer_Word64Slice(base, p.field) + + nn, err := o.DecodeVarint() + if err != nil { + return err + } + nb := int(nn) // number of bytes of encoded int64s + + fin := o.index + nb + if fin < o.index { + return errOverflow + } + for o.index < fin { + u, err := p.valDec(o) + if err != nil { + return err + } + v.Append(u) + } + return nil +} + +// Decode a slice of strings ([]string). +func (o *Buffer) dec_slice_string(p *Properties, base structPointer) error { + s, err := o.DecodeStringBytes() + if err != nil { + return err + } + v := structPointer_StringSlice(base, p.field) + *v = append(*v, s) + return nil +} + +// Decode a slice of slice of bytes ([][]byte). +func (o *Buffer) dec_slice_slice_byte(p *Properties, base structPointer) error { + b, err := o.DecodeRawBytes(true) + if err != nil { + return err + } + v := structPointer_BytesSlice(base, p.field) + *v = append(*v, b) + return nil +} + +// Decode a map field. +func (o *Buffer) dec_new_map(p *Properties, base structPointer) error { + raw, err := o.DecodeRawBytes(false) + if err != nil { + return err + } + oi := o.index // index at the end of this map entry + o.index -= len(raw) // move buffer back to start of map entry + + mptr := structPointer_NewAt(base, p.field, p.mtype) // *map[K]V + if mptr.Elem().IsNil() { + mptr.Elem().Set(reflect.MakeMap(mptr.Type().Elem())) + } + v := mptr.Elem() // map[K]V + + // Prepare addressable doubly-indirect placeholders for the key and value types. + // See enc_new_map for why. + keyptr := reflect.New(reflect.PtrTo(p.mtype.Key())).Elem() // addressable *K + keybase := toStructPointer(keyptr.Addr()) // **K + + var valbase structPointer + var valptr reflect.Value + switch p.mtype.Elem().Kind() { + case reflect.Slice: + // []byte + var dummy []byte + valptr = reflect.ValueOf(&dummy) // *[]byte + valbase = toStructPointer(valptr) // *[]byte + case reflect.Ptr: + // message; valptr is **Msg; need to allocate the intermediate pointer + valptr = reflect.New(reflect.PtrTo(p.mtype.Elem())).Elem() // addressable *V + valptr.Set(reflect.New(valptr.Type().Elem())) + valbase = toStructPointer(valptr) + default: + // everything else + valptr = reflect.New(reflect.PtrTo(p.mtype.Elem())).Elem() // addressable *V + valbase = toStructPointer(valptr.Addr()) // **V + } + + // Decode. + // This parses a restricted wire format, namely the encoding of a message + // with two fields. See enc_new_map for the format. + for o.index < oi { + // tagcode for key and value properties are always a single byte + // because they have tags 1 and 2. + tagcode := o.buf[o.index] + o.index++ + switch tagcode { + case p.mkeyprop.tagcode[0]: + if err := p.mkeyprop.dec(o, p.mkeyprop, keybase); err != nil { + return err + } + case p.mvalprop.tagcode[0]: + if err := p.mvalprop.dec(o, p.mvalprop, valbase); err != nil { + return err + } + default: + // TODO: Should we silently skip this instead? + return fmt.Errorf("proto: bad map data tag %d", raw[0]) + } + } + keyelem, valelem := keyptr.Elem(), valptr.Elem() + if !keyelem.IsValid() || !valelem.IsValid() { + // We did not decode the key or the value in the map entry. + // Either way, it's an invalid map entry. + return fmt.Errorf("proto: bad map data: missing key/val") + } + + v.SetMapIndex(keyelem, valelem) + return nil +} + +// Decode a group. +func (o *Buffer) dec_struct_group(p *Properties, base structPointer) error { + bas := structPointer_GetStructPointer(base, p.field) + if structPointer_IsNil(bas) { + // allocate new nested message + bas = toStructPointer(reflect.New(p.stype)) + structPointer_SetStructPointer(base, p.field, bas) + } + return o.unmarshalType(p.stype, p.sprop, true, bas) +} + +// Decode an embedded message. +func (o *Buffer) dec_struct_message(p *Properties, base structPointer) (err error) { + raw, e := o.DecodeRawBytes(false) + if e != nil { + return e + } + + bas := structPointer_GetStructPointer(base, p.field) + if structPointer_IsNil(bas) { + // allocate new nested message + bas = toStructPointer(reflect.New(p.stype)) + structPointer_SetStructPointer(base, p.field, bas) + } + + // If the object can unmarshal itself, let it. + if p.isUnmarshaler { + iv := structPointer_Interface(bas, p.stype) + return iv.(Unmarshaler).Unmarshal(raw) + } + + obuf := o.buf + oi := o.index + o.buf = raw + o.index = 0 + + err = o.unmarshalType(p.stype, p.sprop, false, bas) + o.buf = obuf + o.index = oi + + return err +} + +// Decode a slice of embedded messages. +func (o *Buffer) dec_slice_struct_message(p *Properties, base structPointer) error { + return o.dec_slice_struct(p, false, base) +} + +// Decode a slice of embedded groups. +func (o *Buffer) dec_slice_struct_group(p *Properties, base structPointer) error { + return o.dec_slice_struct(p, true, base) +} + +// Decode a slice of structs ([]*struct). +func (o *Buffer) dec_slice_struct(p *Properties, is_group bool, base structPointer) error { + v := reflect.New(p.stype) + bas := toStructPointer(v) + structPointer_StructPointerSlice(base, p.field).Append(bas) + + if is_group { + err := o.unmarshalType(p.stype, p.sprop, is_group, bas) + return err + } + + raw, err := o.DecodeRawBytes(false) + if err != nil { + return err + } + + // If the object can unmarshal itself, let it. + if p.isUnmarshaler { + iv := v.Interface() + return iv.(Unmarshaler).Unmarshal(raw) + } + + obuf := o.buf + oi := o.index + o.buf = raw + o.index = 0 + + err = o.unmarshalType(p.stype, p.sprop, is_group, bas) + + o.buf = obuf + o.index = oi + + return err +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/decode_gogo.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/decode_gogo.go new file mode 100644 index 0000000000000..6a77aad7661a7 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/decode_gogo.go @@ -0,0 +1,175 @@ +// Copyright (c) 2013, Vastech SA (PTY) LTD. All rights reserved. +// http://github.com/gogo/protobuf/gogoproto +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +import ( + "reflect" +) + +// Decode a reference to a struct pointer. +func (o *Buffer) dec_ref_struct_message(p *Properties, base structPointer) (err error) { + raw, e := o.DecodeRawBytes(false) + if e != nil { + return e + } + + // If the object can unmarshal itself, let it. + if p.isUnmarshaler { + panic("not supported, since this is a pointer receiver") + } + + obuf := o.buf + oi := o.index + o.buf = raw + o.index = 0 + + bas := structPointer_FieldPointer(base, p.field) + + err = o.unmarshalType(p.stype, p.sprop, false, bas) + o.buf = obuf + o.index = oi + + return err +} + +// Decode a slice of references to struct pointers ([]struct). +func (o *Buffer) dec_slice_ref_struct(p *Properties, is_group bool, base structPointer) error { + newBas := appendStructPointer(base, p.field, p.sstype) + + if is_group { + panic("not supported, maybe in future, if requested.") + } + + raw, err := o.DecodeRawBytes(false) + if err != nil { + return err + } + + // If the object can unmarshal itself, let it. + if p.isUnmarshaler { + panic("not supported, since this is not a pointer receiver.") + } + + obuf := o.buf + oi := o.index + o.buf = raw + o.index = 0 + + err = o.unmarshalType(p.stype, p.sprop, is_group, newBas) + + o.buf = obuf + o.index = oi + + return err +} + +// Decode a slice of references to struct pointers. +func (o *Buffer) dec_slice_ref_struct_message(p *Properties, base structPointer) error { + return o.dec_slice_ref_struct(p, false, base) +} + +func setPtrCustomType(base structPointer, f field, v interface{}) { + if v == nil { + return + } + structPointer_SetStructPointer(base, f, structPointer(reflect.ValueOf(v).Pointer())) +} + +func setCustomType(base structPointer, f field, value interface{}) { + if value == nil { + return + } + v := reflect.ValueOf(value).Elem() + t := reflect.TypeOf(value).Elem() + kind := t.Kind() + switch kind { + case reflect.Slice: + slice := reflect.MakeSlice(t, v.Len(), v.Cap()) + reflect.Copy(slice, v) + oldHeader := structPointer_GetSliceHeader(base, f) + oldHeader.Data = slice.Pointer() + oldHeader.Len = v.Len() + oldHeader.Cap = v.Cap() + default: + l := 1 + size := reflect.TypeOf(value).Elem().Size() + if kind == reflect.Array { + l = reflect.TypeOf(value).Elem().Len() + size = reflect.TypeOf(value).Size() + } + total := int(size) * l + structPointer_Copy(toStructPointer(reflect.ValueOf(value)), structPointer_Add(base, f), total) + } +} + +func (o *Buffer) dec_custom_bytes(p *Properties, base structPointer) error { + b, err := o.DecodeRawBytes(true) + if err != nil { + return err + } + i := reflect.New(p.ctype.Elem()).Interface() + custom := (i).(Unmarshaler) + if err := custom.Unmarshal(b); err != nil { + return err + } + setPtrCustomType(base, p.field, custom) + return nil +} + +func (o *Buffer) dec_custom_ref_bytes(p *Properties, base structPointer) error { + b, err := o.DecodeRawBytes(true) + if err != nil { + return err + } + i := reflect.New(p.ctype).Interface() + custom := (i).(Unmarshaler) + if err := custom.Unmarshal(b); err != nil { + return err + } + if custom != nil { + setCustomType(base, p.field, custom) + } + return nil +} + +// Decode a slice of bytes ([]byte) into a slice of custom types. +func (o *Buffer) dec_custom_slice_bytes(p *Properties, base structPointer) error { + b, err := o.DecodeRawBytes(true) + if err != nil { + return err + } + i := reflect.New(p.ctype.Elem()).Interface() + custom := (i).(Unmarshaler) + if err := custom.Unmarshal(b); err != nil { + return err + } + newBas := appendStructPointer(base, p.field, p.ctype) + + setCustomType(newBas, 0, custom) + + return nil +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/encode.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/encode.go new file mode 100644 index 0000000000000..91f3f0784d2c4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/encode.go @@ -0,0 +1,1293 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +/* + * Routines for encoding data into the wire format for protocol buffers. + */ + +import ( + "errors" + "fmt" + "reflect" + "sort" +) + +// RequiredNotSetError is the error returned if Marshal is called with +// a protocol buffer struct whose required fields have not +// all been initialized. It is also the error returned if Unmarshal is +// called with an encoded protocol buffer that does not include all the +// required fields. +// +// When printed, RequiredNotSetError reports the first unset required field in a +// message. If the field cannot be precisely determined, it is reported as +// "{Unknown}". +type RequiredNotSetError struct { + field string +} + +func (e *RequiredNotSetError) Error() string { + return fmt.Sprintf("proto: required field %q not set", e.field) +} + +var ( + // errRepeatedHasNil is the error returned if Marshal is called with + // a struct with a repeated field containing a nil element. + errRepeatedHasNil = errors.New("proto: repeated field has nil element") + + // ErrNil is the error returned if Marshal is called with nil. + ErrNil = errors.New("proto: Marshal called with nil") +) + +// The fundamental encoders that put bytes on the wire. +// Those that take integer types all accept uint64 and are +// therefore of type valueEncoder. + +const maxVarintBytes = 10 // maximum length of a varint + +// EncodeVarint returns the varint encoding of x. +// This is the format for the +// int32, int64, uint32, uint64, bool, and enum +// protocol buffer types. +// Not used by the package itself, but helpful to clients +// wishing to use the same encoding. +func EncodeVarint(x uint64) []byte { + var buf [maxVarintBytes]byte + var n int + for n = 0; x > 127; n++ { + buf[n] = 0x80 | uint8(x&0x7F) + x >>= 7 + } + buf[n] = uint8(x) + n++ + return buf[0:n] +} + +// EncodeVarint writes a varint-encoded integer to the Buffer. +// This is the format for the +// int32, int64, uint32, uint64, bool, and enum +// protocol buffer types. +func (p *Buffer) EncodeVarint(x uint64) error { + for x >= 1<<7 { + p.buf = append(p.buf, uint8(x&0x7f|0x80)) + x >>= 7 + } + p.buf = append(p.buf, uint8(x)) + return nil +} + +func sizeVarint(x uint64) (n int) { + for { + n++ + x >>= 7 + if x == 0 { + break + } + } + return n +} + +// EncodeFixed64 writes a 64-bit integer to the Buffer. +// This is the format for the +// fixed64, sfixed64, and double protocol buffer types. +func (p *Buffer) EncodeFixed64(x uint64) error { + p.buf = append(p.buf, + uint8(x), + uint8(x>>8), + uint8(x>>16), + uint8(x>>24), + uint8(x>>32), + uint8(x>>40), + uint8(x>>48), + uint8(x>>56)) + return nil +} + +func sizeFixed64(x uint64) int { + return 8 +} + +// EncodeFixed32 writes a 32-bit integer to the Buffer. +// This is the format for the +// fixed32, sfixed32, and float protocol buffer types. +func (p *Buffer) EncodeFixed32(x uint64) error { + p.buf = append(p.buf, + uint8(x), + uint8(x>>8), + uint8(x>>16), + uint8(x>>24)) + return nil +} + +func sizeFixed32(x uint64) int { + return 4 +} + +// EncodeZigzag64 writes a zigzag-encoded 64-bit integer +// to the Buffer. +// This is the format used for the sint64 protocol buffer type. +func (p *Buffer) EncodeZigzag64(x uint64) error { + // use signed number to get arithmetic right shift. + return p.EncodeVarint(uint64((x << 1) ^ uint64((int64(x) >> 63)))) +} + +func sizeZigzag64(x uint64) int { + return sizeVarint(uint64((x << 1) ^ uint64((int64(x) >> 63)))) +} + +// EncodeZigzag32 writes a zigzag-encoded 32-bit integer +// to the Buffer. +// This is the format used for the sint32 protocol buffer type. +func (p *Buffer) EncodeZigzag32(x uint64) error { + // use signed number to get arithmetic right shift. + return p.EncodeVarint(uint64((uint32(x) << 1) ^ uint32((int32(x) >> 31)))) +} + +func sizeZigzag32(x uint64) int { + return sizeVarint(uint64((uint32(x) << 1) ^ uint32((int32(x) >> 31)))) +} + +// EncodeRawBytes writes a count-delimited byte buffer to the Buffer. +// This is the format used for the bytes protocol buffer +// type and for embedded messages. +func (p *Buffer) EncodeRawBytes(b []byte) error { + p.EncodeVarint(uint64(len(b))) + p.buf = append(p.buf, b...) + return nil +} + +func sizeRawBytes(b []byte) int { + return sizeVarint(uint64(len(b))) + + len(b) +} + +// EncodeStringBytes writes an encoded string to the Buffer. +// This is the format used for the proto2 string type. +func (p *Buffer) EncodeStringBytes(s string) error { + p.EncodeVarint(uint64(len(s))) + p.buf = append(p.buf, s...) + return nil +} + +func sizeStringBytes(s string) int { + return sizeVarint(uint64(len(s))) + + len(s) +} + +// Marshaler is the interface representing objects that can marshal themselves. +type Marshaler interface { + Marshal() ([]byte, error) +} + +// Marshal takes the protocol buffer +// and encodes it into the wire format, returning the data. +func Marshal(pb Message) ([]byte, error) { + // Can the object marshal itself? + if m, ok := pb.(Marshaler); ok { + return m.Marshal() + } + p := NewBuffer(nil) + err := p.Marshal(pb) + var state errorState + if err != nil && !state.shouldContinue(err, nil) { + return nil, err + } + if p.buf == nil && err == nil { + // Return a non-nil slice on success. + return []byte{}, nil + } + return p.buf, err +} + +// Marshal takes the protocol buffer +// and encodes it into the wire format, writing the result to the +// Buffer. +func (p *Buffer) Marshal(pb Message) error { + // Can the object marshal itself? + if m, ok := pb.(Marshaler); ok { + data, err := m.Marshal() + if err != nil { + return err + } + p.buf = append(p.buf, data...) + return nil + } + + t, base, err := getbase(pb) + if structPointer_IsNil(base) { + return ErrNil + } + if err == nil { + err = p.enc_struct(GetProperties(t.Elem()), base) + } + + if collectStats { + stats.Encode++ + } + + return err +} + +// Size returns the encoded size of a protocol buffer. +func Size(pb Message) (n int) { + // Can the object marshal itself? If so, Size is slow. + // TODO: add Size to Marshaler, or add a Sizer interface. + if m, ok := pb.(Marshaler); ok { + b, _ := m.Marshal() + return len(b) + } + + t, base, err := getbase(pb) + if structPointer_IsNil(base) { + return 0 + } + if err == nil { + n = size_struct(GetProperties(t.Elem()), base) + } + + if collectStats { + stats.Size++ + } + + return +} + +// Individual type encoders. + +// Encode a bool. +func (o *Buffer) enc_bool(p *Properties, base structPointer) error { + v := *structPointer_Bool(base, p.field) + if v == nil { + return ErrNil + } + x := 0 + if *v { + x = 1 + } + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, uint64(x)) + return nil +} + +func (o *Buffer) enc_proto3_bool(p *Properties, base structPointer) error { + v := *structPointer_BoolVal(base, p.field) + if !v { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, 1) + return nil +} + +func size_bool(p *Properties, base structPointer) int { + v := *structPointer_Bool(base, p.field) + if v == nil { + return 0 + } + return len(p.tagcode) + 1 // each bool takes exactly one byte +} + +func size_proto3_bool(p *Properties, base structPointer) int { + v := *structPointer_BoolVal(base, p.field) + if !v { + return 0 + } + return len(p.tagcode) + 1 // each bool takes exactly one byte +} + +// Encode an int32. +func (o *Buffer) enc_int32(p *Properties, base structPointer) error { + v := structPointer_Word32(base, p.field) + if word32_IsNil(v) { + return ErrNil + } + x := int32(word32_Get(v)) // permit sign extension to use full 64-bit range + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, uint64(x)) + return nil +} + +func (o *Buffer) enc_proto3_int32(p *Properties, base structPointer) error { + v := structPointer_Word32Val(base, p.field) + x := int32(word32Val_Get(v)) // permit sign extension to use full 64-bit range + if x == 0 { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, uint64(x)) + return nil +} + +func size_int32(p *Properties, base structPointer) (n int) { + v := structPointer_Word32(base, p.field) + if word32_IsNil(v) { + return 0 + } + x := int32(word32_Get(v)) // permit sign extension to use full 64-bit range + n += len(p.tagcode) + n += p.valSize(uint64(x)) + return +} + +func size_proto3_int32(p *Properties, base structPointer) (n int) { + v := structPointer_Word32Val(base, p.field) + x := int32(word32Val_Get(v)) // permit sign extension to use full 64-bit range + if x == 0 { + return 0 + } + n += len(p.tagcode) + n += p.valSize(uint64(x)) + return +} + +// Encode a uint32. +// Exactly the same as int32, except for no sign extension. +func (o *Buffer) enc_uint32(p *Properties, base structPointer) error { + v := structPointer_Word32(base, p.field) + if word32_IsNil(v) { + return ErrNil + } + x := word32_Get(v) + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, uint64(x)) + return nil +} + +func (o *Buffer) enc_proto3_uint32(p *Properties, base structPointer) error { + v := structPointer_Word32Val(base, p.field) + x := word32Val_Get(v) + if x == 0 { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, uint64(x)) + return nil +} + +func size_uint32(p *Properties, base structPointer) (n int) { + v := structPointer_Word32(base, p.field) + if word32_IsNil(v) { + return 0 + } + x := word32_Get(v) + n += len(p.tagcode) + n += p.valSize(uint64(x)) + return +} + +func size_proto3_uint32(p *Properties, base structPointer) (n int) { + v := structPointer_Word32Val(base, p.field) + x := word32Val_Get(v) + if x == 0 { + return 0 + } + n += len(p.tagcode) + n += p.valSize(uint64(x)) + return +} + +// Encode an int64. +func (o *Buffer) enc_int64(p *Properties, base structPointer) error { + v := structPointer_Word64(base, p.field) + if word64_IsNil(v) { + return ErrNil + } + x := word64_Get(v) + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, x) + return nil +} + +func (o *Buffer) enc_proto3_int64(p *Properties, base structPointer) error { + v := structPointer_Word64Val(base, p.field) + x := word64Val_Get(v) + if x == 0 { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, x) + return nil +} + +func size_int64(p *Properties, base structPointer) (n int) { + v := structPointer_Word64(base, p.field) + if word64_IsNil(v) { + return 0 + } + x := word64_Get(v) + n += len(p.tagcode) + n += p.valSize(x) + return +} + +func size_proto3_int64(p *Properties, base structPointer) (n int) { + v := structPointer_Word64Val(base, p.field) + x := word64Val_Get(v) + if x == 0 { + return 0 + } + n += len(p.tagcode) + n += p.valSize(x) + return +} + +// Encode a string. +func (o *Buffer) enc_string(p *Properties, base structPointer) error { + v := *structPointer_String(base, p.field) + if v == nil { + return ErrNil + } + x := *v + o.buf = append(o.buf, p.tagcode...) + o.EncodeStringBytes(x) + return nil +} + +func (o *Buffer) enc_proto3_string(p *Properties, base structPointer) error { + v := *structPointer_StringVal(base, p.field) + if v == "" { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeStringBytes(v) + return nil +} + +func size_string(p *Properties, base structPointer) (n int) { + v := *structPointer_String(base, p.field) + if v == nil { + return 0 + } + x := *v + n += len(p.tagcode) + n += sizeStringBytes(x) + return +} + +func size_proto3_string(p *Properties, base structPointer) (n int) { + v := *structPointer_StringVal(base, p.field) + if v == "" { + return 0 + } + n += len(p.tagcode) + n += sizeStringBytes(v) + return +} + +// All protocol buffer fields are nillable, but be careful. +func isNil(v reflect.Value) bool { + switch v.Kind() { + case reflect.Interface, reflect.Map, reflect.Ptr, reflect.Slice: + return v.IsNil() + } + return false +} + +// Encode a message struct. +func (o *Buffer) enc_struct_message(p *Properties, base structPointer) error { + var state errorState + structp := structPointer_GetStructPointer(base, p.field) + if structPointer_IsNil(structp) { + return ErrNil + } + + // Can the object marshal itself? + if p.isMarshaler { + m := structPointer_Interface(structp, p.stype).(Marshaler) + data, err := m.Marshal() + if err != nil && !state.shouldContinue(err, nil) { + return err + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(data) + return state.err + } + + o.buf = append(o.buf, p.tagcode...) + return o.enc_len_struct(p.sprop, structp, &state) +} + +func size_struct_message(p *Properties, base structPointer) int { + structp := structPointer_GetStructPointer(base, p.field) + if structPointer_IsNil(structp) { + return 0 + } + + // Can the object marshal itself? + if p.isMarshaler { + m := structPointer_Interface(structp, p.stype).(Marshaler) + data, _ := m.Marshal() + n0 := len(p.tagcode) + n1 := sizeRawBytes(data) + return n0 + n1 + } + + n0 := len(p.tagcode) + n1 := size_struct(p.sprop, structp) + n2 := sizeVarint(uint64(n1)) // size of encoded length + return n0 + n1 + n2 +} + +// Encode a group struct. +func (o *Buffer) enc_struct_group(p *Properties, base structPointer) error { + var state errorState + b := structPointer_GetStructPointer(base, p.field) + if structPointer_IsNil(b) { + return ErrNil + } + + o.EncodeVarint(uint64((p.Tag << 3) | WireStartGroup)) + err := o.enc_struct(p.sprop, b) + if err != nil && !state.shouldContinue(err, nil) { + return err + } + o.EncodeVarint(uint64((p.Tag << 3) | WireEndGroup)) + return state.err +} + +func size_struct_group(p *Properties, base structPointer) (n int) { + b := structPointer_GetStructPointer(base, p.field) + if structPointer_IsNil(b) { + return 0 + } + + n += sizeVarint(uint64((p.Tag << 3) | WireStartGroup)) + n += size_struct(p.sprop, b) + n += sizeVarint(uint64((p.Tag << 3) | WireEndGroup)) + return +} + +// Encode a slice of bools ([]bool). +func (o *Buffer) enc_slice_bool(p *Properties, base structPointer) error { + s := *structPointer_BoolSlice(base, p.field) + l := len(s) + if l == 0 { + return ErrNil + } + for _, x := range s { + o.buf = append(o.buf, p.tagcode...) + v := uint64(0) + if x { + v = 1 + } + p.valEnc(o, v) + } + return nil +} + +func size_slice_bool(p *Properties, base structPointer) int { + s := *structPointer_BoolSlice(base, p.field) + l := len(s) + if l == 0 { + return 0 + } + return l * (len(p.tagcode) + 1) // each bool takes exactly one byte +} + +// Encode a slice of bools ([]bool) in packed format. +func (o *Buffer) enc_slice_packed_bool(p *Properties, base structPointer) error { + s := *structPointer_BoolSlice(base, p.field) + l := len(s) + if l == 0 { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeVarint(uint64(l)) // each bool takes exactly one byte + for _, x := range s { + v := uint64(0) + if x { + v = 1 + } + p.valEnc(o, v) + } + return nil +} + +func size_slice_packed_bool(p *Properties, base structPointer) (n int) { + s := *structPointer_BoolSlice(base, p.field) + l := len(s) + if l == 0 { + return 0 + } + n += len(p.tagcode) + n += sizeVarint(uint64(l)) + n += l // each bool takes exactly one byte + return +} + +// Encode a slice of bytes ([]byte). +func (o *Buffer) enc_slice_byte(p *Properties, base structPointer) error { + s := *structPointer_Bytes(base, p.field) + if s == nil { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(s) + return nil +} + +func (o *Buffer) enc_proto3_slice_byte(p *Properties, base structPointer) error { + s := *structPointer_Bytes(base, p.field) + if len(s) == 0 { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(s) + return nil +} + +func size_slice_byte(p *Properties, base structPointer) (n int) { + s := *structPointer_Bytes(base, p.field) + if s == nil { + return 0 + } + n += len(p.tagcode) + n += sizeRawBytes(s) + return +} + +func size_proto3_slice_byte(p *Properties, base structPointer) (n int) { + s := *structPointer_Bytes(base, p.field) + if len(s) == 0 { + return 0 + } + n += len(p.tagcode) + n += sizeRawBytes(s) + return +} + +// Encode a slice of int32s ([]int32). +func (o *Buffer) enc_slice_int32(p *Properties, base structPointer) error { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return ErrNil + } + for i := 0; i < l; i++ { + o.buf = append(o.buf, p.tagcode...) + x := int32(s.Index(i)) // permit sign extension to use full 64-bit range + p.valEnc(o, uint64(x)) + } + return nil +} + +func size_slice_int32(p *Properties, base structPointer) (n int) { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return 0 + } + for i := 0; i < l; i++ { + n += len(p.tagcode) + x := int32(s.Index(i)) // permit sign extension to use full 64-bit range + n += p.valSize(uint64(x)) + } + return +} + +// Encode a slice of int32s ([]int32) in packed format. +func (o *Buffer) enc_slice_packed_int32(p *Properties, base structPointer) error { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return ErrNil + } + // TODO: Reuse a Buffer. + buf := NewBuffer(nil) + for i := 0; i < l; i++ { + x := int32(s.Index(i)) // permit sign extension to use full 64-bit range + p.valEnc(buf, uint64(x)) + } + + o.buf = append(o.buf, p.tagcode...) + o.EncodeVarint(uint64(len(buf.buf))) + o.buf = append(o.buf, buf.buf...) + return nil +} + +func size_slice_packed_int32(p *Properties, base structPointer) (n int) { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return 0 + } + var bufSize int + for i := 0; i < l; i++ { + x := int32(s.Index(i)) // permit sign extension to use full 64-bit range + bufSize += p.valSize(uint64(x)) + } + + n += len(p.tagcode) + n += sizeVarint(uint64(bufSize)) + n += bufSize + return +} + +// Encode a slice of uint32s ([]uint32). +// Exactly the same as int32, except for no sign extension. +func (o *Buffer) enc_slice_uint32(p *Properties, base structPointer) error { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return ErrNil + } + for i := 0; i < l; i++ { + o.buf = append(o.buf, p.tagcode...) + x := s.Index(i) + p.valEnc(o, uint64(x)) + } + return nil +} + +func size_slice_uint32(p *Properties, base structPointer) (n int) { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return 0 + } + for i := 0; i < l; i++ { + n += len(p.tagcode) + x := s.Index(i) + n += p.valSize(uint64(x)) + } + return +} + +// Encode a slice of uint32s ([]uint32) in packed format. +// Exactly the same as int32, except for no sign extension. +func (o *Buffer) enc_slice_packed_uint32(p *Properties, base structPointer) error { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return ErrNil + } + // TODO: Reuse a Buffer. + buf := NewBuffer(nil) + for i := 0; i < l; i++ { + p.valEnc(buf, uint64(s.Index(i))) + } + + o.buf = append(o.buf, p.tagcode...) + o.EncodeVarint(uint64(len(buf.buf))) + o.buf = append(o.buf, buf.buf...) + return nil +} + +func size_slice_packed_uint32(p *Properties, base structPointer) (n int) { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return 0 + } + var bufSize int + for i := 0; i < l; i++ { + bufSize += p.valSize(uint64(s.Index(i))) + } + + n += len(p.tagcode) + n += sizeVarint(uint64(bufSize)) + n += bufSize + return +} + +// Encode a slice of int64s ([]int64). +func (o *Buffer) enc_slice_int64(p *Properties, base structPointer) error { + s := structPointer_Word64Slice(base, p.field) + l := s.Len() + if l == 0 { + return ErrNil + } + for i := 0; i < l; i++ { + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, s.Index(i)) + } + return nil +} + +func size_slice_int64(p *Properties, base structPointer) (n int) { + s := structPointer_Word64Slice(base, p.field) + l := s.Len() + if l == 0 { + return 0 + } + for i := 0; i < l; i++ { + n += len(p.tagcode) + n += p.valSize(s.Index(i)) + } + return +} + +// Encode a slice of int64s ([]int64) in packed format. +func (o *Buffer) enc_slice_packed_int64(p *Properties, base structPointer) error { + s := structPointer_Word64Slice(base, p.field) + l := s.Len() + if l == 0 { + return ErrNil + } + // TODO: Reuse a Buffer. + buf := NewBuffer(nil) + for i := 0; i < l; i++ { + p.valEnc(buf, s.Index(i)) + } + + o.buf = append(o.buf, p.tagcode...) + o.EncodeVarint(uint64(len(buf.buf))) + o.buf = append(o.buf, buf.buf...) + return nil +} + +func size_slice_packed_int64(p *Properties, base structPointer) (n int) { + s := structPointer_Word64Slice(base, p.field) + l := s.Len() + if l == 0 { + return 0 + } + var bufSize int + for i := 0; i < l; i++ { + bufSize += p.valSize(s.Index(i)) + } + + n += len(p.tagcode) + n += sizeVarint(uint64(bufSize)) + n += bufSize + return +} + +// Encode a slice of slice of bytes ([][]byte). +func (o *Buffer) enc_slice_slice_byte(p *Properties, base structPointer) error { + ss := *structPointer_BytesSlice(base, p.field) + l := len(ss) + if l == 0 { + return ErrNil + } + for i := 0; i < l; i++ { + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(ss[i]) + } + return nil +} + +func size_slice_slice_byte(p *Properties, base structPointer) (n int) { + ss := *structPointer_BytesSlice(base, p.field) + l := len(ss) + if l == 0 { + return 0 + } + n += l * len(p.tagcode) + for i := 0; i < l; i++ { + n += sizeRawBytes(ss[i]) + } + return +} + +// Encode a slice of strings ([]string). +func (o *Buffer) enc_slice_string(p *Properties, base structPointer) error { + ss := *structPointer_StringSlice(base, p.field) + l := len(ss) + for i := 0; i < l; i++ { + o.buf = append(o.buf, p.tagcode...) + o.EncodeStringBytes(ss[i]) + } + return nil +} + +func size_slice_string(p *Properties, base structPointer) (n int) { + ss := *structPointer_StringSlice(base, p.field) + l := len(ss) + n += l * len(p.tagcode) + for i := 0; i < l; i++ { + n += sizeStringBytes(ss[i]) + } + return +} + +// Encode a slice of message structs ([]*struct). +func (o *Buffer) enc_slice_struct_message(p *Properties, base structPointer) error { + var state errorState + s := structPointer_StructPointerSlice(base, p.field) + l := s.Len() + + for i := 0; i < l; i++ { + structp := s.Index(i) + if structPointer_IsNil(structp) { + return errRepeatedHasNil + } + + // Can the object marshal itself? + if p.isMarshaler { + m := structPointer_Interface(structp, p.stype).(Marshaler) + data, err := m.Marshal() + if err != nil && !state.shouldContinue(err, nil) { + return err + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(data) + continue + } + + o.buf = append(o.buf, p.tagcode...) + err := o.enc_len_struct(p.sprop, structp, &state) + if err != nil && !state.shouldContinue(err, nil) { + if err == ErrNil { + return errRepeatedHasNil + } + return err + } + } + return state.err +} + +func size_slice_struct_message(p *Properties, base structPointer) (n int) { + s := structPointer_StructPointerSlice(base, p.field) + l := s.Len() + n += l * len(p.tagcode) + for i := 0; i < l; i++ { + structp := s.Index(i) + if structPointer_IsNil(structp) { + return // return the size up to this point + } + + // Can the object marshal itself? + if p.isMarshaler { + m := structPointer_Interface(structp, p.stype).(Marshaler) + data, _ := m.Marshal() + n += len(p.tagcode) + n += sizeRawBytes(data) + continue + } + + n0 := size_struct(p.sprop, structp) + n1 := sizeVarint(uint64(n0)) // size of encoded length + n += n0 + n1 + } + return +} + +// Encode a slice of group structs ([]*struct). +func (o *Buffer) enc_slice_struct_group(p *Properties, base structPointer) error { + var state errorState + s := structPointer_StructPointerSlice(base, p.field) + l := s.Len() + + for i := 0; i < l; i++ { + b := s.Index(i) + if structPointer_IsNil(b) { + return errRepeatedHasNil + } + + o.EncodeVarint(uint64((p.Tag << 3) | WireStartGroup)) + + err := o.enc_struct(p.sprop, b) + + if err != nil && !state.shouldContinue(err, nil) { + if err == ErrNil { + return errRepeatedHasNil + } + return err + } + + o.EncodeVarint(uint64((p.Tag << 3) | WireEndGroup)) + } + return state.err +} + +func size_slice_struct_group(p *Properties, base structPointer) (n int) { + s := structPointer_StructPointerSlice(base, p.field) + l := s.Len() + + n += l * sizeVarint(uint64((p.Tag<<3)|WireStartGroup)) + n += l * sizeVarint(uint64((p.Tag<<3)|WireEndGroup)) + for i := 0; i < l; i++ { + b := s.Index(i) + if structPointer_IsNil(b) { + return // return size up to this point + } + + n += size_struct(p.sprop, b) + } + return +} + +// Encode an extension map. +func (o *Buffer) enc_map(p *Properties, base structPointer) error { + v := *structPointer_ExtMap(base, p.field) + if err := encodeExtensionMap(v); err != nil { + return err + } + // Fast-path for common cases: zero or one extensions. + if len(v) <= 1 { + for _, e := range v { + o.buf = append(o.buf, e.enc...) + } + return nil + } + + // Sort keys to provide a deterministic encoding. + keys := make([]int, 0, len(v)) + for k := range v { + keys = append(keys, int(k)) + } + sort.Ints(keys) + + for _, k := range keys { + o.buf = append(o.buf, v[int32(k)].enc...) + } + return nil +} + +func size_map(p *Properties, base structPointer) int { + v := *structPointer_ExtMap(base, p.field) + return sizeExtensionMap(v) +} + +// Encode a map field. +func (o *Buffer) enc_new_map(p *Properties, base structPointer) error { + var state errorState // XXX: or do we need to plumb this through? + + /* + A map defined as + map map_field = N; + is encoded in the same way as + message MapFieldEntry { + key_type key = 1; + value_type value = 2; + } + repeated MapFieldEntry map_field = N; + */ + + v := structPointer_NewAt(base, p.field, p.mtype).Elem() // map[K]V + if v.Len() == 0 { + return nil + } + + keycopy, valcopy, keybase, valbase := mapEncodeScratch(p.mtype) + + enc := func() error { + if err := p.mkeyprop.enc(o, p.mkeyprop, keybase); err != nil { + return err + } + if err := p.mvalprop.enc(o, p.mvalprop, valbase); err != nil { + return err + } + return nil + } + + keys := v.MapKeys() + sort.Sort(mapKeys(keys)) + for _, key := range keys { + val := v.MapIndex(key) + + // The only illegal map entry values are nil message pointers. + if val.Kind() == reflect.Ptr && val.IsNil() { + return errors.New("proto: map has nil element") + } + + keycopy.Set(key) + valcopy.Set(val) + + o.buf = append(o.buf, p.tagcode...) + if err := o.enc_len_thing(enc, &state); err != nil { + return err + } + } + return nil +} + +func size_new_map(p *Properties, base structPointer) int { + v := structPointer_NewAt(base, p.field, p.mtype).Elem() // map[K]V + + keycopy, valcopy, keybase, valbase := mapEncodeScratch(p.mtype) + + n := 0 + for _, key := range v.MapKeys() { + val := v.MapIndex(key) + keycopy.Set(key) + valcopy.Set(val) + + // Tag codes for key and val are the responsibility of the sub-sizer. + keysize := p.mkeyprop.size(p.mkeyprop, keybase) + valsize := p.mvalprop.size(p.mvalprop, valbase) + entry := keysize + valsize + // Add on tag code and length of map entry itself. + n += len(p.tagcode) + sizeVarint(uint64(entry)) + entry + } + return n +} + +// mapEncodeScratch returns a new reflect.Value matching the map's value type, +// and a structPointer suitable for passing to an encoder or sizer. +func mapEncodeScratch(mapType reflect.Type) (keycopy, valcopy reflect.Value, keybase, valbase structPointer) { + // Prepare addressable doubly-indirect placeholders for the key and value types. + // This is needed because the element-type encoders expect **T, but the map iteration produces T. + + keycopy = reflect.New(mapType.Key()).Elem() // addressable K + keyptr := reflect.New(reflect.PtrTo(keycopy.Type())).Elem() // addressable *K + keyptr.Set(keycopy.Addr()) // + keybase = toStructPointer(keyptr.Addr()) // **K + + // Value types are more varied and require special handling. + switch mapType.Elem().Kind() { + case reflect.Slice: + // []byte + var dummy []byte + valcopy = reflect.ValueOf(&dummy).Elem() // addressable []byte + valbase = toStructPointer(valcopy.Addr()) + case reflect.Ptr: + // message; the generated field type is map[K]*Msg (so V is *Msg), + // so we only need one level of indirection. + valcopy = reflect.New(mapType.Elem()).Elem() // addressable V + valbase = toStructPointer(valcopy.Addr()) + default: + // everything else + valcopy = reflect.New(mapType.Elem()).Elem() // addressable V + valptr := reflect.New(reflect.PtrTo(valcopy.Type())).Elem() // addressable *V + valptr.Set(valcopy.Addr()) // + valbase = toStructPointer(valptr.Addr()) // **V + } + return +} + +// Encode a struct. +func (o *Buffer) enc_struct(prop *StructProperties, base structPointer) error { + var state errorState + // Encode fields in tag order so that decoders may use optimizations + // that depend on the ordering. + // https://developers.google.com/protocol-buffers/docs/encoding#order + for _, i := range prop.order { + p := prop.Prop[i] + if p.enc != nil { + err := p.enc(o, p, base) + if err != nil { + if err == ErrNil { + if p.Required && state.err == nil { + state.err = &RequiredNotSetError{p.Name} + } + } else if err == errRepeatedHasNil { + // Give more context to nil values in repeated fields. + return errors.New("repeated field " + p.OrigName + " has nil element") + } else if !state.shouldContinue(err, p) { + return err + } + } + } + } + + // Add unrecognized fields at the end. + if prop.unrecField.IsValid() { + v := *structPointer_Bytes(base, prop.unrecField) + if len(v) > 0 { + o.buf = append(o.buf, v...) + } + } + + return state.err +} + +func size_struct(prop *StructProperties, base structPointer) (n int) { + for _, i := range prop.order { + p := prop.Prop[i] + if p.size != nil { + n += p.size(p, base) + } + } + + // Add unrecognized fields at the end. + if prop.unrecField.IsValid() { + v := *structPointer_Bytes(base, prop.unrecField) + n += len(v) + } + + return +} + +var zeroes [20]byte // longer than any conceivable sizeVarint + +// Encode a struct, preceded by its encoded length (as a varint). +func (o *Buffer) enc_len_struct(prop *StructProperties, base structPointer, state *errorState) error { + return o.enc_len_thing(func() error { return o.enc_struct(prop, base) }, state) +} + +// Encode something, preceded by its encoded length (as a varint). +func (o *Buffer) enc_len_thing(enc func() error, state *errorState) error { + iLen := len(o.buf) + o.buf = append(o.buf, 0, 0, 0, 0) // reserve four bytes for length + iMsg := len(o.buf) + err := enc() + if err != nil && !state.shouldContinue(err, nil) { + return err + } + lMsg := len(o.buf) - iMsg + lLen := sizeVarint(uint64(lMsg)) + switch x := lLen - (iMsg - iLen); { + case x > 0: // actual length is x bytes larger than the space we reserved + // Move msg x bytes right. + o.buf = append(o.buf, zeroes[:x]...) + copy(o.buf[iMsg+x:], o.buf[iMsg:iMsg+lMsg]) + case x < 0: // actual length is x bytes smaller than the space we reserved + // Move msg x bytes left. + copy(o.buf[iMsg+x:], o.buf[iMsg:iMsg+lMsg]) + o.buf = o.buf[:len(o.buf)+x] // x is negative + } + // Encode the length in the reserved space. + o.buf = o.buf[:iLen] + o.EncodeVarint(uint64(lMsg)) + o.buf = o.buf[:len(o.buf)+lMsg] + return state.err +} + +// errorState maintains the first error that occurs and updates that error +// with additional context. +type errorState struct { + err error +} + +// shouldContinue reports whether encoding should continue upon encountering the +// given error. If the error is RequiredNotSetError, shouldContinue returns true +// and, if this is the first appearance of that error, remembers it for future +// reporting. +// +// If prop is not nil, it may update any error with additional context about the +// field with the error. +func (s *errorState) shouldContinue(err error, prop *Properties) bool { + // Ignore unset required fields. + reqNotSet, ok := err.(*RequiredNotSetError) + if !ok { + return false + } + if s.err == nil { + if prop != nil { + err = &RequiredNotSetError{prop.Name + "." + reqNotSet.field} + } + s.err = err + } + return true +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/encode_gogo.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/encode_gogo.go new file mode 100644 index 0000000000000..f77cfb1eea4db --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/encode_gogo.go @@ -0,0 +1,354 @@ +// Extensions for Protocol Buffers to create more go like structures. +// +// Copyright (c) 2013, Vastech SA (PTY) LTD. All rights reserved. +// http://github.com/gogo/protobuf/gogoproto +// +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// http://github.com/golang/protobuf/ +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +import ( + "reflect" +) + +func NewRequiredNotSetError(field string) *RequiredNotSetError { + return &RequiredNotSetError{field} +} + +type Sizer interface { + Size() int +} + +func (o *Buffer) enc_ext_slice_byte(p *Properties, base structPointer) error { + s := *structPointer_Bytes(base, p.field) + if s == nil { + return ErrNil + } + o.buf = append(o.buf, s...) + return nil +} + +func size_ext_slice_byte(p *Properties, base structPointer) (n int) { + s := *structPointer_Bytes(base, p.field) + if s == nil { + return 0 + } + n += len(s) + return +} + +// Encode a reference to bool pointer. +func (o *Buffer) enc_ref_bool(p *Properties, base structPointer) error { + v := *structPointer_BoolVal(base, p.field) + x := 0 + if v { + x = 1 + } + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, uint64(x)) + return nil +} + +func size_ref_bool(p *Properties, base structPointer) int { + return len(p.tagcode) + 1 // each bool takes exactly one byte +} + +// Encode a reference to int32 pointer. +func (o *Buffer) enc_ref_int32(p *Properties, base structPointer) error { + v := structPointer_Word32Val(base, p.field) + x := int32(word32Val_Get(v)) + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, uint64(x)) + return nil +} + +func size_ref_int32(p *Properties, base structPointer) (n int) { + v := structPointer_Word32Val(base, p.field) + x := int32(word32Val_Get(v)) + n += len(p.tagcode) + n += p.valSize(uint64(x)) + return +} + +func (o *Buffer) enc_ref_uint32(p *Properties, base structPointer) error { + v := structPointer_Word32Val(base, p.field) + x := word32Val_Get(v) + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, uint64(x)) + return nil +} + +func size_ref_uint32(p *Properties, base structPointer) (n int) { + v := structPointer_Word32Val(base, p.field) + x := word32Val_Get(v) + n += len(p.tagcode) + n += p.valSize(uint64(x)) + return +} + +// Encode a reference to an int64 pointer. +func (o *Buffer) enc_ref_int64(p *Properties, base structPointer) error { + v := structPointer_Word64Val(base, p.field) + x := word64Val_Get(v) + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, x) + return nil +} + +func size_ref_int64(p *Properties, base structPointer) (n int) { + v := structPointer_Word64Val(base, p.field) + x := word64Val_Get(v) + n += len(p.tagcode) + n += p.valSize(x) + return +} + +// Encode a reference to a string pointer. +func (o *Buffer) enc_ref_string(p *Properties, base structPointer) error { + v := *structPointer_StringVal(base, p.field) + o.buf = append(o.buf, p.tagcode...) + o.EncodeStringBytes(v) + return nil +} + +func size_ref_string(p *Properties, base structPointer) (n int) { + v := *structPointer_StringVal(base, p.field) + n += len(p.tagcode) + n += sizeStringBytes(v) + return +} + +// Encode a reference to a message struct. +func (o *Buffer) enc_ref_struct_message(p *Properties, base structPointer) error { + var state errorState + structp := structPointer_GetRefStructPointer(base, p.field) + if structPointer_IsNil(structp) { + return ErrNil + } + + // Can the object marshal itself? + if p.isMarshaler { + m := structPointer_Interface(structp, p.stype).(Marshaler) + data, err := m.Marshal() + if err != nil && !state.shouldContinue(err, nil) { + return err + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(data) + return nil + } + + o.buf = append(o.buf, p.tagcode...) + return o.enc_len_struct(p.sprop, structp, &state) +} + +//TODO this is only copied, please fix this +func size_ref_struct_message(p *Properties, base structPointer) int { + structp := structPointer_GetRefStructPointer(base, p.field) + if structPointer_IsNil(structp) { + return 0 + } + + // Can the object marshal itself? + if p.isMarshaler { + m := structPointer_Interface(structp, p.stype).(Marshaler) + data, _ := m.Marshal() + n0 := len(p.tagcode) + n1 := sizeRawBytes(data) + return n0 + n1 + } + + n0 := len(p.tagcode) + n1 := size_struct(p.sprop, structp) + n2 := sizeVarint(uint64(n1)) // size of encoded length + return n0 + n1 + n2 +} + +// Encode a slice of references to message struct pointers ([]struct). +func (o *Buffer) enc_slice_ref_struct_message(p *Properties, base structPointer) error { + var state errorState + ss := structPointer_GetStructPointer(base, p.field) + ss1 := structPointer_GetRefStructPointer(ss, field(0)) + size := p.stype.Size() + l := structPointer_Len(base, p.field) + for i := 0; i < l; i++ { + structp := structPointer_Add(ss1, field(uintptr(i)*size)) + if structPointer_IsNil(structp) { + return errRepeatedHasNil + } + + // Can the object marshal itself? + if p.isMarshaler { + m := structPointer_Interface(structp, p.stype).(Marshaler) + data, err := m.Marshal() + if err != nil && !state.shouldContinue(err, nil) { + return err + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(data) + continue + } + + o.buf = append(o.buf, p.tagcode...) + err := o.enc_len_struct(p.sprop, structp, &state) + if err != nil && !state.shouldContinue(err, nil) { + if err == ErrNil { + return errRepeatedHasNil + } + return err + } + + } + return state.err +} + +//TODO this is only copied, please fix this +func size_slice_ref_struct_message(p *Properties, base structPointer) (n int) { + ss := structPointer_GetStructPointer(base, p.field) + ss1 := structPointer_GetRefStructPointer(ss, field(0)) + size := p.stype.Size() + l := structPointer_Len(base, p.field) + n += l * len(p.tagcode) + for i := 0; i < l; i++ { + structp := structPointer_Add(ss1, field(uintptr(i)*size)) + if structPointer_IsNil(structp) { + return // return the size up to this point + } + + // Can the object marshal itself? + if p.isMarshaler { + m := structPointer_Interface(structp, p.stype).(Marshaler) + data, _ := m.Marshal() + n += len(p.tagcode) + n += sizeRawBytes(data) + continue + } + + n0 := size_struct(p.sprop, structp) + n1 := sizeVarint(uint64(n0)) // size of encoded length + n += n0 + n1 + } + return +} + +func (o *Buffer) enc_custom_bytes(p *Properties, base structPointer) error { + i := structPointer_InterfaceRef(base, p.field, p.ctype) + if i == nil { + return ErrNil + } + custom := i.(Marshaler) + data, err := custom.Marshal() + if err != nil { + return err + } + if data == nil { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(data) + return nil +} + +func size_custom_bytes(p *Properties, base structPointer) (n int) { + n += len(p.tagcode) + i := structPointer_InterfaceRef(base, p.field, p.ctype) + if i == nil { + return 0 + } + custom := i.(Marshaler) + data, _ := custom.Marshal() + n += sizeRawBytes(data) + return +} + +func (o *Buffer) enc_custom_ref_bytes(p *Properties, base structPointer) error { + custom := structPointer_InterfaceAt(base, p.field, p.ctype).(Marshaler) + data, err := custom.Marshal() + if err != nil { + return err + } + if data == nil { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(data) + return nil +} + +func size_custom_ref_bytes(p *Properties, base structPointer) (n int) { + n += len(p.tagcode) + i := structPointer_InterfaceAt(base, p.field, p.ctype) + if i == nil { + return 0 + } + custom := i.(Marshaler) + data, _ := custom.Marshal() + n += sizeRawBytes(data) + return +} + +func (o *Buffer) enc_custom_slice_bytes(p *Properties, base structPointer) error { + inter := structPointer_InterfaceRef(base, p.field, p.ctype) + if inter == nil { + return ErrNil + } + slice := reflect.ValueOf(inter) + l := slice.Len() + for i := 0; i < l; i++ { + v := slice.Index(i) + custom := v.Interface().(Marshaler) + data, err := custom.Marshal() + if err != nil { + return err + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(data) + } + return nil +} + +func size_custom_slice_bytes(p *Properties, base structPointer) (n int) { + inter := structPointer_InterfaceRef(base, p.field, p.ctype) + if inter == nil { + return 0 + } + slice := reflect.ValueOf(inter) + l := slice.Len() + n += l * len(p.tagcode) + for i := 0; i < l; i++ { + v := slice.Index(i) + custom := v.Interface().(Marshaler) + data, _ := custom.Marshal() + n += sizeRawBytes(data) + } + return +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/equal.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/equal.go new file mode 100644 index 0000000000000..d8673a3e97aef --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/equal.go @@ -0,0 +1,256 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2011 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// Protocol buffer comparison. +// TODO: MessageSet. + +package proto + +import ( + "bytes" + "log" + "reflect" + "strings" +) + +/* +Equal returns true iff protocol buffers a and b are equal. +The arguments must both be pointers to protocol buffer structs. + +Equality is defined in this way: + - Two messages are equal iff they are the same type, + corresponding fields are equal, unknown field sets + are equal, and extensions sets are equal. + - Two set scalar fields are equal iff their values are equal. + If the fields are of a floating-point type, remember that + NaN != x for all x, including NaN. + - Two repeated fields are equal iff their lengths are the same, + and their corresponding elements are equal (a "bytes" field, + although represented by []byte, is not a repeated field) + - Two unset fields are equal. + - Two unknown field sets are equal if their current + encoded state is equal. + - Two extension sets are equal iff they have corresponding + elements that are pairwise equal. + - Every other combination of things are not equal. + +The return value is undefined if a and b are not protocol buffers. +*/ +func Equal(a, b Message) bool { + if a == nil || b == nil { + return a == b + } + v1, v2 := reflect.ValueOf(a), reflect.ValueOf(b) + if v1.Type() != v2.Type() { + return false + } + if v1.Kind() == reflect.Ptr { + if v1.IsNil() { + return v2.IsNil() + } + if v2.IsNil() { + return false + } + v1, v2 = v1.Elem(), v2.Elem() + } + if v1.Kind() != reflect.Struct { + return false + } + return equalStruct(v1, v2) +} + +// v1 and v2 are known to have the same type. +func equalStruct(v1, v2 reflect.Value) bool { + for i := 0; i < v1.NumField(); i++ { + f := v1.Type().Field(i) + if strings.HasPrefix(f.Name, "XXX_") { + continue + } + f1, f2 := v1.Field(i), v2.Field(i) + if f.Type.Kind() == reflect.Ptr { + if n1, n2 := f1.IsNil(), f2.IsNil(); n1 && n2 { + // both unset + continue + } else if n1 != n2 { + // set/unset mismatch + return false + } + b1, ok := f1.Interface().(raw) + if ok { + b2 := f2.Interface().(raw) + // RawMessage + if !bytes.Equal(b1.Bytes(), b2.Bytes()) { + return false + } + continue + } + f1, f2 = f1.Elem(), f2.Elem() + } + if !equalAny(f1, f2) { + return false + } + } + + if em1 := v1.FieldByName("XXX_extensions"); em1.IsValid() { + em2 := v2.FieldByName("XXX_extensions") + if !equalExtensions(v1.Type(), em1.Interface().(map[int32]Extension), em2.Interface().(map[int32]Extension)) { + return false + } + } + + uf := v1.FieldByName("XXX_unrecognized") + if !uf.IsValid() { + return true + } + + u1 := uf.Bytes() + u2 := v2.FieldByName("XXX_unrecognized").Bytes() + if !bytes.Equal(u1, u2) { + return false + } + + return true +} + +// v1 and v2 are known to have the same type. +func equalAny(v1, v2 reflect.Value) bool { + if v1.Type() == protoMessageType { + m1, _ := v1.Interface().(Message) + m2, _ := v2.Interface().(Message) + return Equal(m1, m2) + } + switch v1.Kind() { + case reflect.Bool: + return v1.Bool() == v2.Bool() + case reflect.Float32, reflect.Float64: + return v1.Float() == v2.Float() + case reflect.Int32, reflect.Int64: + return v1.Int() == v2.Int() + case reflect.Map: + if v1.Len() != v2.Len() { + return false + } + for _, key := range v1.MapKeys() { + val2 := v2.MapIndex(key) + if !val2.IsValid() { + // This key was not found in the second map. + return false + } + if !equalAny(v1.MapIndex(key), val2) { + return false + } + } + return true + case reflect.Ptr: + return equalAny(v1.Elem(), v2.Elem()) + case reflect.Slice: + if v1.Type().Elem().Kind() == reflect.Uint8 { + // short circuit: []byte + if v1.IsNil() != v2.IsNil() { + return false + } + return bytes.Equal(v1.Interface().([]byte), v2.Interface().([]byte)) + } + + if v1.Len() != v2.Len() { + return false + } + for i := 0; i < v1.Len(); i++ { + if !equalAny(v1.Index(i), v2.Index(i)) { + return false + } + } + return true + case reflect.String: + return v1.Interface().(string) == v2.Interface().(string) + case reflect.Struct: + return equalStruct(v1, v2) + case reflect.Uint32, reflect.Uint64: + return v1.Uint() == v2.Uint() + } + + // unknown type, so not a protocol buffer + log.Printf("proto: don't know how to compare %v", v1) + return false +} + +// base is the struct type that the extensions are based on. +// em1 and em2 are extension maps. +func equalExtensions(base reflect.Type, em1, em2 map[int32]Extension) bool { + if len(em1) != len(em2) { + return false + } + + for extNum, e1 := range em1 { + e2, ok := em2[extNum] + if !ok { + return false + } + + m1, m2 := e1.value, e2.value + + if m1 != nil && m2 != nil { + // Both are unencoded. + if !equalAny(reflect.ValueOf(m1), reflect.ValueOf(m2)) { + return false + } + continue + } + + // At least one is encoded. To do a semantically correct comparison + // we need to unmarshal them first. + var desc *ExtensionDesc + if m := extensionMaps[base]; m != nil { + desc = m[extNum] + } + if desc == nil { + log.Printf("proto: don't know how to compare extension %d of %v", extNum, base) + continue + } + var err error + if m1 == nil { + m1, err = decodeExtension(e1.enc, desc) + } + if m2 == nil && err == nil { + m2, err = decodeExtension(e2.enc, desc) + } + if err != nil { + // The encoded form is invalid. + log.Printf("proto: badly encoded extension %d of %v: %v", extNum, base, err) + return false + } + if !equalAny(reflect.ValueOf(m1), reflect.ValueOf(m2)) { + return false + } + } + + return true +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/equal_test.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/equal_test.go new file mode 100644 index 0000000000000..ef6048008d3e1 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/equal_test.go @@ -0,0 +1,191 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2011 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "testing" + + . "github.com/gogo/protobuf/proto" + pb "github.com/gogo/protobuf/proto/testdata" +) + +// Four identical base messages. +// The init function adds extensions to some of them. +var messageWithoutExtension = &pb.MyMessage{Count: Int32(7)} +var messageWithExtension1a = &pb.MyMessage{Count: Int32(7)} +var messageWithExtension1b = &pb.MyMessage{Count: Int32(7)} +var messageWithExtension2 = &pb.MyMessage{Count: Int32(7)} + +// Two messages with non-message extensions. +var messageWithInt32Extension1 = &pb.MyMessage{Count: Int32(8)} +var messageWithInt32Extension2 = &pb.MyMessage{Count: Int32(8)} + +func init() { + ext1 := &pb.Ext{Data: String("Kirk")} + ext2 := &pb.Ext{Data: String("Picard")} + + // messageWithExtension1a has ext1, but never marshals it. + if err := SetExtension(messageWithExtension1a, pb.E_Ext_More, ext1); err != nil { + panic("SetExtension on 1a failed: " + err.Error()) + } + + // messageWithExtension1b is the unmarshaled form of messageWithExtension1a. + if err := SetExtension(messageWithExtension1b, pb.E_Ext_More, ext1); err != nil { + panic("SetExtension on 1b failed: " + err.Error()) + } + buf, err := Marshal(messageWithExtension1b) + if err != nil { + panic("Marshal of 1b failed: " + err.Error()) + } + messageWithExtension1b.Reset() + if err := Unmarshal(buf, messageWithExtension1b); err != nil { + panic("Unmarshal of 1b failed: " + err.Error()) + } + + // messageWithExtension2 has ext2. + if err := SetExtension(messageWithExtension2, pb.E_Ext_More, ext2); err != nil { + panic("SetExtension on 2 failed: " + err.Error()) + } + + if err := SetExtension(messageWithInt32Extension1, pb.E_Ext_Number, Int32(23)); err != nil { + panic("SetExtension on Int32-1 failed: " + err.Error()) + } + if err := SetExtension(messageWithInt32Extension1, pb.E_Ext_Number, Int32(24)); err != nil { + panic("SetExtension on Int32-2 failed: " + err.Error()) + } +} + +var EqualTests = []struct { + desc string + a, b Message + exp bool +}{ + {"different types", &pb.GoEnum{}, &pb.GoTestField{}, false}, + {"equal empty", &pb.GoEnum{}, &pb.GoEnum{}, true}, + {"nil vs nil", nil, nil, true}, + {"typed nil vs typed nil", (*pb.GoEnum)(nil), (*pb.GoEnum)(nil), true}, + {"typed nil vs empty", (*pb.GoEnum)(nil), &pb.GoEnum{}, false}, + {"different typed nil", (*pb.GoEnum)(nil), (*pb.GoTestField)(nil), false}, + + {"one set field, one unset field", &pb.GoTestField{Label: String("foo")}, &pb.GoTestField{}, false}, + {"one set field zero, one unset field", &pb.GoTest{Param: Int32(0)}, &pb.GoTest{}, false}, + {"different set fields", &pb.GoTestField{Label: String("foo")}, &pb.GoTestField{Label: String("bar")}, false}, + {"equal set", &pb.GoTestField{Label: String("foo")}, &pb.GoTestField{Label: String("foo")}, true}, + + {"repeated, one set", &pb.GoTest{F_Int32Repeated: []int32{2, 3}}, &pb.GoTest{}, false}, + {"repeated, different length", &pb.GoTest{F_Int32Repeated: []int32{2, 3}}, &pb.GoTest{F_Int32Repeated: []int32{2}}, false}, + {"repeated, different value", &pb.GoTest{F_Int32Repeated: []int32{2}}, &pb.GoTest{F_Int32Repeated: []int32{3}}, false}, + {"repeated, equal", &pb.GoTest{F_Int32Repeated: []int32{2, 4}}, &pb.GoTest{F_Int32Repeated: []int32{2, 4}}, true}, + {"repeated, nil equal nil", &pb.GoTest{F_Int32Repeated: nil}, &pb.GoTest{F_Int32Repeated: nil}, true}, + {"repeated, nil equal empty", &pb.GoTest{F_Int32Repeated: nil}, &pb.GoTest{F_Int32Repeated: []int32{}}, true}, + {"repeated, empty equal nil", &pb.GoTest{F_Int32Repeated: []int32{}}, &pb.GoTest{F_Int32Repeated: nil}, true}, + + { + "nested, different", + &pb.GoTest{RequiredField: &pb.GoTestField{Label: String("foo")}}, + &pb.GoTest{RequiredField: &pb.GoTestField{Label: String("bar")}}, + false, + }, + { + "nested, equal", + &pb.GoTest{RequiredField: &pb.GoTestField{Label: String("wow")}}, + &pb.GoTest{RequiredField: &pb.GoTestField{Label: String("wow")}}, + true, + }, + + {"bytes", &pb.OtherMessage{Value: []byte("foo")}, &pb.OtherMessage{Value: []byte("foo")}, true}, + {"bytes, empty", &pb.OtherMessage{Value: []byte{}}, &pb.OtherMessage{Value: []byte{}}, true}, + {"bytes, empty vs nil", &pb.OtherMessage{Value: []byte{}}, &pb.OtherMessage{Value: nil}, false}, + { + "repeated bytes", + &pb.MyMessage{RepBytes: [][]byte{[]byte("sham"), []byte("wow")}}, + &pb.MyMessage{RepBytes: [][]byte{[]byte("sham"), []byte("wow")}}, + true, + }, + + {"extension vs. no extension", messageWithoutExtension, messageWithExtension1a, false}, + {"extension vs. same extension", messageWithExtension1a, messageWithExtension1b, true}, + {"extension vs. different extension", messageWithExtension1a, messageWithExtension2, false}, + + {"int32 extension vs. itself", messageWithInt32Extension1, messageWithInt32Extension1, true}, + {"int32 extension vs. a different int32", messageWithInt32Extension1, messageWithInt32Extension2, false}, + + { + "message with group", + &pb.MyMessage{ + Count: Int32(1), + Somegroup: &pb.MyMessage_SomeGroup{ + GroupField: Int32(5), + }, + }, + &pb.MyMessage{ + Count: Int32(1), + Somegroup: &pb.MyMessage_SomeGroup{ + GroupField: Int32(5), + }, + }, + true, + }, + + { + "map same", + &pb.MessageWithMap{NameMapping: map[int32]string{1: "Ken"}}, + &pb.MessageWithMap{NameMapping: map[int32]string{1: "Ken"}}, + true, + }, + { + "map different entry", + &pb.MessageWithMap{NameMapping: map[int32]string{1: "Ken"}}, + &pb.MessageWithMap{NameMapping: map[int32]string{2: "Rob"}}, + false, + }, + { + "map different key only", + &pb.MessageWithMap{NameMapping: map[int32]string{1: "Ken"}}, + &pb.MessageWithMap{NameMapping: map[int32]string{2: "Ken"}}, + false, + }, + { + "map different value only", + &pb.MessageWithMap{NameMapping: map[int32]string{1: "Ken"}}, + &pb.MessageWithMap{NameMapping: map[int32]string{1: "Rob"}}, + false, + }, +} + +func TestEqual(t *testing.T) { + for _, tc := range EqualTests { + if res := Equal(tc.a, tc.b); res != tc.exp { + t.Errorf("%v: Equal(%v, %v) = %v, want %v", tc.desc, tc.a, tc.b, res, tc.exp) + } + } +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/extensions.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/extensions.go new file mode 100644 index 0000000000000..9a6374fdbdec4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/extensions.go @@ -0,0 +1,519 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +/* + * Types and routines for supporting protocol buffer extensions. + */ + +import ( + "errors" + "fmt" + "reflect" + "strconv" + "sync" +) + +// ErrMissingExtension is the error returned by GetExtension if the named extension is not in the message. +var ErrMissingExtension = errors.New("proto: missing extension") + +// ExtensionRange represents a range of message extensions for a protocol buffer. +// Used in code generated by the protocol compiler. +type ExtensionRange struct { + Start, End int32 // both inclusive +} + +// extendableProto is an interface implemented by any protocol buffer that may be extended. +type extendableProto interface { + Message + ExtensionRangeArray() []ExtensionRange +} + +type extensionsMap interface { + extendableProto + ExtensionMap() map[int32]Extension +} + +type extensionsBytes interface { + extendableProto + GetExtensions() *[]byte +} + +var extendableProtoType = reflect.TypeOf((*extendableProto)(nil)).Elem() + +// ExtensionDesc represents an extension specification. +// Used in generated code from the protocol compiler. +type ExtensionDesc struct { + ExtendedType Message // nil pointer to the type that is being extended + ExtensionType interface{} // nil pointer to the extension type + Field int32 // field number + Name string // fully-qualified name of extension, for text formatting + Tag string // protobuf tag style +} + +func (ed *ExtensionDesc) repeated() bool { + t := reflect.TypeOf(ed.ExtensionType) + return t.Kind() == reflect.Slice && t.Elem().Kind() != reflect.Uint8 +} + +// Extension represents an extension in a message. +type Extension struct { + // When an extension is stored in a message using SetExtension + // only desc and value are set. When the message is marshaled + // enc will be set to the encoded form of the message. + // + // When a message is unmarshaled and contains extensions, each + // extension will have only enc set. When such an extension is + // accessed using GetExtension (or GetExtensions) desc and value + // will be set. + desc *ExtensionDesc + value interface{} + enc []byte +} + +// SetRawExtension is for testing only. +func SetRawExtension(base extendableProto, id int32, b []byte) { + if ebase, ok := base.(extensionsMap); ok { + ebase.ExtensionMap()[id] = Extension{enc: b} + } else if ebase, ok := base.(extensionsBytes); ok { + clearExtension(base, id) + ext := ebase.GetExtensions() + *ext = append(*ext, b...) + } else { + panic("unreachable") + } +} + +// isExtensionField returns true iff the given field number is in an extension range. +func isExtensionField(pb extendableProto, field int32) bool { + for _, er := range pb.ExtensionRangeArray() { + if er.Start <= field && field <= er.End { + return true + } + } + return false +} + +// checkExtensionTypes checks that the given extension is valid for pb. +func checkExtensionTypes(pb extendableProto, extension *ExtensionDesc) error { + // Check the extended type. + if a, b := reflect.TypeOf(pb), reflect.TypeOf(extension.ExtendedType); a != b { + return errors.New("proto: bad extended type; " + b.String() + " does not extend " + a.String()) + } + // Check the range. + if !isExtensionField(pb, extension.Field) { + return errors.New("proto: bad extension number; not in declared ranges") + } + return nil +} + +// extPropKey is sufficient to uniquely identify an extension. +type extPropKey struct { + base reflect.Type + field int32 +} + +var extProp = struct { + sync.RWMutex + m map[extPropKey]*Properties +}{ + m: make(map[extPropKey]*Properties), +} + +func extensionProperties(ed *ExtensionDesc) *Properties { + key := extPropKey{base: reflect.TypeOf(ed.ExtendedType), field: ed.Field} + + extProp.RLock() + if prop, ok := extProp.m[key]; ok { + extProp.RUnlock() + return prop + } + extProp.RUnlock() + + extProp.Lock() + defer extProp.Unlock() + // Check again. + if prop, ok := extProp.m[key]; ok { + return prop + } + + prop := new(Properties) + prop.Init(reflect.TypeOf(ed.ExtensionType), "unknown_name", ed.Tag, nil) + extProp.m[key] = prop + return prop +} + +// encodeExtensionMap encodes any unmarshaled (unencoded) extensions in m. +func encodeExtensionMap(m map[int32]Extension) error { + for k, e := range m { + err := encodeExtension(&e) + if err != nil { + return err + } + m[k] = e + } + return nil +} + +func encodeExtension(e *Extension) error { + if e.value == nil || e.desc == nil { + // Extension is only in its encoded form. + return nil + } + // We don't skip extensions that have an encoded form set, + // because the extension value may have been mutated after + // the last time this function was called. + + et := reflect.TypeOf(e.desc.ExtensionType) + props := extensionProperties(e.desc) + + p := NewBuffer(nil) + // If e.value has type T, the encoder expects a *struct{ X T }. + // Pass a *T with a zero field and hope it all works out. + x := reflect.New(et) + x.Elem().Set(reflect.ValueOf(e.value)) + if err := props.enc(p, props, toStructPointer(x)); err != nil { + return err + } + e.enc = p.buf + return nil +} + +func sizeExtensionMap(m map[int32]Extension) (n int) { + for _, e := range m { + if e.value == nil || e.desc == nil { + // Extension is only in its encoded form. + n += len(e.enc) + continue + } + + // We don't skip extensions that have an encoded form set, + // because the extension value may have been mutated after + // the last time this function was called. + + et := reflect.TypeOf(e.desc.ExtensionType) + props := extensionProperties(e.desc) + + // If e.value has type T, the encoder expects a *struct{ X T }. + // Pass a *T with a zero field and hope it all works out. + x := reflect.New(et) + x.Elem().Set(reflect.ValueOf(e.value)) + n += props.size(props, toStructPointer(x)) + } + return +} + +// HasExtension returns whether the given extension is present in pb. +func HasExtension(pb extendableProto, extension *ExtensionDesc) bool { + // TODO: Check types, field numbers, etc.? + if epb, doki := pb.(extensionsMap); doki { + _, ok := epb.ExtensionMap()[extension.Field] + return ok + } else if epb, doki := pb.(extensionsBytes); doki { + ext := epb.GetExtensions() + buf := *ext + o := 0 + for o < len(buf) { + tag, n := DecodeVarint(buf[o:]) + fieldNum := int32(tag >> 3) + if int32(fieldNum) == extension.Field { + return true + } + wireType := int(tag & 0x7) + o += n + l, err := size(buf[o:], wireType) + if err != nil { + return false + } + o += l + } + return false + } + panic("unreachable") +} + +func deleteExtension(pb extensionsBytes, theFieldNum int32, offset int) int { + ext := pb.GetExtensions() + for offset < len(*ext) { + tag, n1 := DecodeVarint((*ext)[offset:]) + fieldNum := int32(tag >> 3) + wireType := int(tag & 0x7) + n2, err := size((*ext)[offset+n1:], wireType) + if err != nil { + panic(err) + } + newOffset := offset + n1 + n2 + if fieldNum == theFieldNum { + *ext = append((*ext)[:offset], (*ext)[newOffset:]...) + return offset + } + offset = newOffset + } + return -1 +} + +func clearExtension(pb extendableProto, fieldNum int32) { + if epb, doki := pb.(extensionsMap); doki { + delete(epb.ExtensionMap(), fieldNum) + } else if epb, doki := pb.(extensionsBytes); doki { + offset := 0 + for offset != -1 { + offset = deleteExtension(epb, fieldNum, offset) + } + } else { + panic("unreachable") + } +} + +// ClearExtension removes the given extension from pb. +func ClearExtension(pb extendableProto, extension *ExtensionDesc) { + // TODO: Check types, field numbers, etc.? + clearExtension(pb, extension.Field) +} + +// GetExtension parses and returns the given extension of pb. +// If the extension is not present it returns ErrMissingExtension. +func GetExtension(pb extendableProto, extension *ExtensionDesc) (interface{}, error) { + if err := checkExtensionTypes(pb, extension); err != nil { + return nil, err + } + + if epb, doki := pb.(extensionsMap); doki { + emap := epb.ExtensionMap() + e, ok := emap[extension.Field] + if !ok { + // defaultExtensionValue returns the default value or + // ErrMissingExtension if there is no default. + return defaultExtensionValue(extension) + } + if e.value != nil { + // Already decoded. Check the descriptor, though. + if e.desc != extension { + // This shouldn't happen. If it does, it means that + // GetExtension was called twice with two different + // descriptors with the same field number. + return nil, errors.New("proto: descriptor conflict") + } + return e.value, nil + } + + v, err := decodeExtension(e.enc, extension) + if err != nil { + return nil, err + } + + // Remember the decoded version and drop the encoded version. + // That way it is safe to mutate what we return. + e.value = v + e.desc = extension + e.enc = nil + emap[extension.Field] = e + return e.value, nil + } else if epb, doki := pb.(extensionsBytes); doki { + ext := epb.GetExtensions() + o := 0 + for o < len(*ext) { + tag, n := DecodeVarint((*ext)[o:]) + fieldNum := int32(tag >> 3) + wireType := int(tag & 0x7) + l, err := size((*ext)[o+n:], wireType) + if err != nil { + return nil, err + } + if int32(fieldNum) == extension.Field { + v, err := decodeExtension((*ext)[o:o+n+l], extension) + if err != nil { + return nil, err + } + return v, nil + } + o += n + l + } + return defaultExtensionValue(extension) + } + panic("unreachable") +} + +// defaultExtensionValue returns the default value for extension. +// If no default for an extension is defined ErrMissingExtension is returned. +func defaultExtensionValue(extension *ExtensionDesc) (interface{}, error) { + t := reflect.TypeOf(extension.ExtensionType) + props := extensionProperties(extension) + + sf, _, err := fieldDefault(t, props) + if err != nil { + return nil, err + } + + if sf == nil || sf.value == nil { + // There is no default value. + return nil, ErrMissingExtension + } + + if t.Kind() != reflect.Ptr { + // We do not need to return a Ptr, we can directly return sf.value. + return sf.value, nil + } + + // We need to return an interface{} that is a pointer to sf.value. + value := reflect.New(t).Elem() + value.Set(reflect.New(value.Type().Elem())) + if sf.kind == reflect.Int32 { + // We may have an int32 or an enum, but the underlying data is int32. + // Since we can't set an int32 into a non int32 reflect.value directly + // set it as a int32. + value.Elem().SetInt(int64(sf.value.(int32))) + } else { + value.Elem().Set(reflect.ValueOf(sf.value)) + } + return value.Interface(), nil +} + +// decodeExtension decodes an extension encoded in b. +func decodeExtension(b []byte, extension *ExtensionDesc) (interface{}, error) { + o := NewBuffer(b) + + t := reflect.TypeOf(extension.ExtensionType) + rep := extension.repeated() + + props := extensionProperties(extension) + + // t is a pointer to a struct, pointer to basic type or a slice. + // Allocate a "field" to store the pointer/slice itself; the + // pointer/slice will be stored here. We pass + // the address of this field to props.dec. + // This passes a zero field and a *t and lets props.dec + // interpret it as a *struct{ x t }. + value := reflect.New(t).Elem() + + for { + // Discard wire type and field number varint. It isn't needed. + if _, err := o.DecodeVarint(); err != nil { + return nil, err + } + + if err := props.dec(o, props, toStructPointer(value.Addr())); err != nil { + return nil, err + } + + if !rep || o.index >= len(o.buf) { + break + } + } + return value.Interface(), nil +} + +// GetExtensions returns a slice of the extensions present in pb that are also listed in es. +// The returned slice has the same length as es; missing extensions will appear as nil elements. +func GetExtensions(pb Message, es []*ExtensionDesc) (extensions []interface{}, err error) { + epb, ok := pb.(extendableProto) + if !ok { + err = errors.New("proto: not an extendable proto") + return + } + extensions = make([]interface{}, len(es)) + for i, e := range es { + extensions[i], err = GetExtension(epb, e) + if err == ErrMissingExtension { + err = nil + } + if err != nil { + return + } + } + return +} + +// SetExtension sets the specified extension of pb to the specified value. +func SetExtension(pb extendableProto, extension *ExtensionDesc, value interface{}) error { + if err := checkExtensionTypes(pb, extension); err != nil { + return err + } + typ := reflect.TypeOf(extension.ExtensionType) + if typ != reflect.TypeOf(value) { + return errors.New("proto: bad extension value type") + } + // nil extension values need to be caught early, because the + // encoder can't distinguish an ErrNil due to a nil extension + // from an ErrNil due to a missing field. Extensions are + // always optional, so the encoder would just swallow the error + // and drop all the extensions from the encoded message. + if reflect.ValueOf(value).IsNil() { + return fmt.Errorf("proto: SetExtension called with nil value of type %T", value) + } + return setExtension(pb, extension, value) +} + +func setExtension(pb extendableProto, extension *ExtensionDesc, value interface{}) error { + if epb, doki := pb.(extensionsMap); doki { + epb.ExtensionMap()[extension.Field] = Extension{desc: extension, value: value} + } else if epb, doki := pb.(extensionsBytes); doki { + ClearExtension(pb, extension) + ext := epb.GetExtensions() + et := reflect.TypeOf(extension.ExtensionType) + props := extensionProperties(extension) + p := NewBuffer(nil) + x := reflect.New(et) + x.Elem().Set(reflect.ValueOf(value)) + if err := props.enc(p, props, toStructPointer(x)); err != nil { + return err + } + *ext = append(*ext, p.buf...) + } + return nil +} + +// A global registry of extensions. +// The generated code will register the generated descriptors by calling RegisterExtension. + +var extensionMaps = make(map[reflect.Type]map[int32]*ExtensionDesc) + +// RegisterExtension is called from the generated code. +func RegisterExtension(desc *ExtensionDesc) { + st := reflect.TypeOf(desc.ExtendedType).Elem() + m := extensionMaps[st] + if m == nil { + m = make(map[int32]*ExtensionDesc) + extensionMaps[st] = m + } + if _, ok := m[desc.Field]; ok { + panic("proto: duplicate extension registered: " + st.String() + " " + strconv.Itoa(int(desc.Field))) + } + m[desc.Field] = desc +} + +// RegisteredExtensions returns a map of the registered extensions of a +// protocol buffer struct, indexed by the extension number. +// The argument pb should be a nil pointer to the struct type. +func RegisteredExtensions(pb Message) map[int32]*ExtensionDesc { + return extensionMaps[reflect.TypeOf(pb).Elem()] +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/extensions_gogo.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/extensions_gogo.go new file mode 100644 index 0000000000000..bd55fb68b61ef --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/extensions_gogo.go @@ -0,0 +1,221 @@ +// Copyright (c) 2013, Vastech SA (PTY) LTD. All rights reserved. +// http://github.com/gogo/protobuf/gogoproto +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +import ( + "bytes" + "errors" + "fmt" + "reflect" + "sort" + "strings" +) + +func GetBoolExtension(pb extendableProto, extension *ExtensionDesc, ifnotset bool) bool { + if reflect.ValueOf(pb).IsNil() { + return ifnotset + } + value, err := GetExtension(pb, extension) + if err != nil { + return ifnotset + } + if value == nil { + return ifnotset + } + if value.(*bool) == nil { + return ifnotset + } + return *(value.(*bool)) +} + +func (this *Extension) Equal(that *Extension) bool { + return bytes.Equal(this.enc, that.enc) +} + +func SizeOfExtensionMap(m map[int32]Extension) (n int) { + return sizeExtensionMap(m) +} + +type sortableMapElem struct { + field int32 + ext Extension +} + +func newSortableExtensionsFromMap(m map[int32]Extension) sortableExtensions { + s := make(sortableExtensions, 0, len(m)) + for k, v := range m { + s = append(s, &sortableMapElem{field: k, ext: v}) + } + return s +} + +type sortableExtensions []*sortableMapElem + +func (this sortableExtensions) Len() int { return len(this) } + +func (this sortableExtensions) Swap(i, j int) { this[i], this[j] = this[j], this[i] } + +func (this sortableExtensions) Less(i, j int) bool { return this[i].field < this[j].field } + +func (this sortableExtensions) String() string { + sort.Sort(this) + ss := make([]string, len(this)) + for i := range this { + ss[i] = fmt.Sprintf("%d: %v", this[i].field, this[i].ext) + } + return "map[" + strings.Join(ss, ",") + "]" +} + +func StringFromExtensionsMap(m map[int32]Extension) string { + return newSortableExtensionsFromMap(m).String() +} + +func StringFromExtensionsBytes(ext []byte) string { + m, err := BytesToExtensionsMap(ext) + if err != nil { + panic(err) + } + return StringFromExtensionsMap(m) +} + +func EncodeExtensionMap(m map[int32]Extension, data []byte) (n int, err error) { + if err := encodeExtensionMap(m); err != nil { + return 0, err + } + keys := make([]int, 0, len(m)) + for k := range m { + keys = append(keys, int(k)) + } + sort.Ints(keys) + for _, k := range keys { + n += copy(data[n:], m[int32(k)].enc) + } + return n, nil +} + +func GetRawExtension(m map[int32]Extension, id int32) ([]byte, error) { + if m[id].value == nil || m[id].desc == nil { + return m[id].enc, nil + } + if err := encodeExtensionMap(m); err != nil { + return nil, err + } + return m[id].enc, nil +} + +func size(buf []byte, wire int) (int, error) { + switch wire { + case WireVarint: + _, n := DecodeVarint(buf) + return n, nil + case WireFixed64: + return 8, nil + case WireBytes: + v, n := DecodeVarint(buf) + return int(v) + n, nil + case WireFixed32: + return 4, nil + case WireStartGroup: + offset := 0 + for { + u, n := DecodeVarint(buf[offset:]) + fwire := int(u & 0x7) + offset += n + if fwire == WireEndGroup { + return offset, nil + } + s, err := size(buf[offset:], wire) + if err != nil { + return 0, err + } + offset += s + } + } + return 0, fmt.Errorf("proto: can't get size for unknown wire type %d", wire) +} + +func BytesToExtensionsMap(buf []byte) (map[int32]Extension, error) { + m := make(map[int32]Extension) + i := 0 + for i < len(buf) { + tag, n := DecodeVarint(buf[i:]) + if n <= 0 { + return nil, fmt.Errorf("unable to decode varint") + } + fieldNum := int32(tag >> 3) + wireType := int(tag & 0x7) + l, err := size(buf[i+n:], wireType) + if err != nil { + return nil, err + } + end := i + int(l) + n + m[int32(fieldNum)] = Extension{enc: buf[i:end]} + i = end + } + return m, nil +} + +func NewExtension(e []byte) Extension { + ee := Extension{enc: make([]byte, len(e))} + copy(ee.enc, e) + return ee +} + +func (this Extension) GoString() string { + if this.enc == nil { + if err := encodeExtension(&this); err != nil { + panic(err) + } + } + return fmt.Sprintf("proto.NewExtension(%#v)", this.enc) +} + +func SetUnsafeExtension(pb extendableProto, fieldNum int32, value interface{}) error { + typ := reflect.TypeOf(pb).Elem() + ext, ok := extensionMaps[typ] + if !ok { + return fmt.Errorf("proto: bad extended type; %s is not extendable", typ.String()) + } + desc, ok := ext[fieldNum] + if !ok { + return errors.New("proto: bad extension number; not in declared ranges") + } + return setExtension(pb, desc, value) +} + +func GetUnsafeExtension(pb extendableProto, fieldNum int32) (interface{}, error) { + typ := reflect.TypeOf(pb).Elem() + ext, ok := extensionMaps[typ] + if !ok { + return nil, fmt.Errorf("proto: bad extended type; %s is not extendable", typ.String()) + } + desc, ok := ext[fieldNum] + if !ok { + return nil, fmt.Errorf("unregistered field number %d", fieldNum) + } + return GetExtension(pb, desc) +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/extensions_test.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/extensions_test.go new file mode 100644 index 0000000000000..86e3006d73982 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/extensions_test.go @@ -0,0 +1,292 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2014 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "fmt" + "reflect" + "testing" + + "github.com/gogo/protobuf/proto" + pb "github.com/gogo/protobuf/proto/testdata" +) + +func TestGetExtensionsWithMissingExtensions(t *testing.T) { + msg := &pb.MyMessage{} + ext1 := &pb.Ext{} + if err := proto.SetExtension(msg, pb.E_Ext_More, ext1); err != nil { + t.Fatalf("Could not set ext1: %s", ext1) + } + exts, err := proto.GetExtensions(msg, []*proto.ExtensionDesc{ + pb.E_Ext_More, + pb.E_Ext_Text, + }) + if err != nil { + t.Fatalf("GetExtensions() failed: %s", err) + } + if exts[0] != ext1 { + t.Errorf("ext1 not in returned extensions: %T %v", exts[0], exts[0]) + } + if exts[1] != nil { + t.Errorf("ext2 in returned extensions: %T %v", exts[1], exts[1]) + } +} + +func TestGetExtensionStability(t *testing.T) { + check := func(m *pb.MyMessage) bool { + ext1, err := proto.GetExtension(m, pb.E_Ext_More) + if err != nil { + t.Fatalf("GetExtension() failed: %s", err) + } + ext2, err := proto.GetExtension(m, pb.E_Ext_More) + if err != nil { + t.Fatalf("GetExtension() failed: %s", err) + } + return ext1 == ext2 + } + msg := &pb.MyMessage{Count: proto.Int32(4)} + ext0 := &pb.Ext{} + if err := proto.SetExtension(msg, pb.E_Ext_More, ext0); err != nil { + t.Fatalf("Could not set ext1: %s", ext0) + } + if !check(msg) { + t.Errorf("GetExtension() not stable before marshaling") + } + bb, err := proto.Marshal(msg) + if err != nil { + t.Fatalf("Marshal() failed: %s", err) + } + msg1 := &pb.MyMessage{} + err = proto.Unmarshal(bb, msg1) + if err != nil { + t.Fatalf("Unmarshal() failed: %s", err) + } + if !check(msg1) { + t.Errorf("GetExtension() not stable after unmarshaling") + } +} + +func TestGetExtensionDefaults(t *testing.T) { + var setFloat64 float64 = 1 + var setFloat32 float32 = 2 + var setInt32 int32 = 3 + var setInt64 int64 = 4 + var setUint32 uint32 = 5 + var setUint64 uint64 = 6 + var setBool = true + var setBool2 = false + var setString = "Goodnight string" + var setBytes = []byte("Goodnight bytes") + var setEnum = pb.DefaultsMessage_TWO + + type testcase struct { + ext *proto.ExtensionDesc // Extension we are testing. + want interface{} // Expected value of extension, or nil (meaning that GetExtension will fail). + def interface{} // Expected value of extension after ClearExtension(). + } + tests := []testcase{ + {pb.E_NoDefaultDouble, setFloat64, nil}, + {pb.E_NoDefaultFloat, setFloat32, nil}, + {pb.E_NoDefaultInt32, setInt32, nil}, + {pb.E_NoDefaultInt64, setInt64, nil}, + {pb.E_NoDefaultUint32, setUint32, nil}, + {pb.E_NoDefaultUint64, setUint64, nil}, + {pb.E_NoDefaultSint32, setInt32, nil}, + {pb.E_NoDefaultSint64, setInt64, nil}, + {pb.E_NoDefaultFixed32, setUint32, nil}, + {pb.E_NoDefaultFixed64, setUint64, nil}, + {pb.E_NoDefaultSfixed32, setInt32, nil}, + {pb.E_NoDefaultSfixed64, setInt64, nil}, + {pb.E_NoDefaultBool, setBool, nil}, + {pb.E_NoDefaultBool, setBool2, nil}, + {pb.E_NoDefaultString, setString, nil}, + {pb.E_NoDefaultBytes, setBytes, nil}, + {pb.E_NoDefaultEnum, setEnum, nil}, + {pb.E_DefaultDouble, setFloat64, float64(3.1415)}, + {pb.E_DefaultFloat, setFloat32, float32(3.14)}, + {pb.E_DefaultInt32, setInt32, int32(42)}, + {pb.E_DefaultInt64, setInt64, int64(43)}, + {pb.E_DefaultUint32, setUint32, uint32(44)}, + {pb.E_DefaultUint64, setUint64, uint64(45)}, + {pb.E_DefaultSint32, setInt32, int32(46)}, + {pb.E_DefaultSint64, setInt64, int64(47)}, + {pb.E_DefaultFixed32, setUint32, uint32(48)}, + {pb.E_DefaultFixed64, setUint64, uint64(49)}, + {pb.E_DefaultSfixed32, setInt32, int32(50)}, + {pb.E_DefaultSfixed64, setInt64, int64(51)}, + {pb.E_DefaultBool, setBool, true}, + {pb.E_DefaultBool, setBool2, true}, + {pb.E_DefaultString, setString, "Hello, string"}, + {pb.E_DefaultBytes, setBytes, []byte("Hello, bytes")}, + {pb.E_DefaultEnum, setEnum, pb.DefaultsMessage_ONE}, + } + + checkVal := func(test testcase, msg *pb.DefaultsMessage, valWant interface{}) error { + val, err := proto.GetExtension(msg, test.ext) + if err != nil { + if valWant != nil { + return fmt.Errorf("GetExtension(): %s", err) + } + if want := proto.ErrMissingExtension; err != want { + return fmt.Errorf("Unexpected error: got %v, want %v", err, want) + } + return nil + } + + // All proto2 extension values are either a pointer to a value or a slice of values. + ty := reflect.TypeOf(val) + tyWant := reflect.TypeOf(test.ext.ExtensionType) + if got, want := ty, tyWant; got != want { + return fmt.Errorf("unexpected reflect.TypeOf(): got %v want %v", got, want) + } + tye := ty.Elem() + tyeWant := tyWant.Elem() + if got, want := tye, tyeWant; got != want { + return fmt.Errorf("unexpected reflect.TypeOf().Elem(): got %v want %v", got, want) + } + + // Check the name of the type of the value. + // If it is an enum it will be type int32 with the name of the enum. + if got, want := tye.Name(), tye.Name(); got != want { + return fmt.Errorf("unexpected reflect.TypeOf().Elem().Name(): got %v want %v", got, want) + } + + // Check that value is what we expect. + // If we have a pointer in val, get the value it points to. + valExp := val + if ty.Kind() == reflect.Ptr { + valExp = reflect.ValueOf(val).Elem().Interface() + } + if got, want := valExp, valWant; !reflect.DeepEqual(got, want) { + return fmt.Errorf("unexpected reflect.DeepEqual(): got %v want %v", got, want) + } + + return nil + } + + setTo := func(test testcase) interface{} { + setTo := reflect.ValueOf(test.want) + if typ := reflect.TypeOf(test.ext.ExtensionType); typ.Kind() == reflect.Ptr { + setTo = reflect.New(typ).Elem() + setTo.Set(reflect.New(setTo.Type().Elem())) + setTo.Elem().Set(reflect.ValueOf(test.want)) + } + return setTo.Interface() + } + + for _, test := range tests { + msg := &pb.DefaultsMessage{} + name := test.ext.Name + + // Check the initial value. + if err := checkVal(test, msg, test.def); err != nil { + t.Errorf("%s: %v", name, err) + } + + // Set the per-type value and check value. + name = fmt.Sprintf("%s (set to %T %v)", name, test.want, test.want) + if err := proto.SetExtension(msg, test.ext, setTo(test)); err != nil { + t.Errorf("%s: SetExtension(): %v", name, err) + continue + } + if err := checkVal(test, msg, test.want); err != nil { + t.Errorf("%s: %v", name, err) + continue + } + + // Set and check the value. + name += " (cleared)" + proto.ClearExtension(msg, test.ext) + if err := checkVal(test, msg, test.def); err != nil { + t.Errorf("%s: %v", name, err) + } + } +} + +func TestExtensionsRoundTrip(t *testing.T) { + msg := &pb.MyMessage{} + ext1 := &pb.Ext{ + Data: proto.String("hi"), + } + ext2 := &pb.Ext{ + Data: proto.String("there"), + } + exists := proto.HasExtension(msg, pb.E_Ext_More) + if exists { + t.Error("Extension More present unexpectedly") + } + if err := proto.SetExtension(msg, pb.E_Ext_More, ext1); err != nil { + t.Error(err) + } + if err := proto.SetExtension(msg, pb.E_Ext_More, ext2); err != nil { + t.Error(err) + } + e, err := proto.GetExtension(msg, pb.E_Ext_More) + if err != nil { + t.Error(err) + } + x, ok := e.(*pb.Ext) + if !ok { + t.Errorf("e has type %T, expected testdata.Ext", e) + } else if *x.Data != "there" { + t.Errorf("SetExtension failed to overwrite, got %+v, not 'there'", x) + } + proto.ClearExtension(msg, pb.E_Ext_More) + if _, err = proto.GetExtension(msg, pb.E_Ext_More); err != proto.ErrMissingExtension { + t.Errorf("got %v, expected ErrMissingExtension", e) + } + if _, err := proto.GetExtension(msg, pb.E_X215); err == nil { + t.Error("expected bad extension error, got nil") + } + if err := proto.SetExtension(msg, pb.E_X215, 12); err == nil { + t.Error("expected extension err") + } + if err := proto.SetExtension(msg, pb.E_Ext_More, 12); err == nil { + t.Error("expected some sort of type mismatch error, got nil") + } +} + +func TestNilExtension(t *testing.T) { + msg := &pb.MyMessage{ + Count: proto.Int32(1), + } + if err := proto.SetExtension(msg, pb.E_Ext_Text, proto.String("hello")); err != nil { + t.Fatal(err) + } + if err := proto.SetExtension(msg, pb.E_Ext_More, (*pb.Ext)(nil)); err == nil { + t.Error("expected SetExtension to fail due to a nil extension") + } else if want := "proto: SetExtension called with nil value of type *testdata.Ext"; err.Error() != want { + t.Errorf("expected error %v, got %v", want, err) + } + // Note: if the behavior of Marshal is ever changed to ignore nil extensions, update + // this test to verify that E_Ext_Text is properly propagated through marshal->unmarshal. +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/lib.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/lib.go new file mode 100644 index 0000000000000..d36f9ad1293b4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/lib.go @@ -0,0 +1,841 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +/* +Package proto converts data structures to and from the wire format of +protocol buffers. It works in concert with the Go source code generated +for .proto files by the protocol compiler. + +A summary of the properties of the protocol buffer interface +for a protocol buffer variable v: + + - Names are turned from camel_case to CamelCase for export. + - There are no methods on v to set fields; just treat + them as structure fields. + - There are getters that return a field's value if set, + and return the field's default value if unset. + The getters work even if the receiver is a nil message. + - The zero value for a struct is its correct initialization state. + All desired fields must be set before marshaling. + - A Reset() method will restore a protobuf struct to its zero state. + - Non-repeated fields are pointers to the values; nil means unset. + That is, optional or required field int32 f becomes F *int32. + - Repeated fields are slices. + - Helper functions are available to aid the setting of fields. + msg.Foo = proto.String("hello") // set field + - Constants are defined to hold the default values of all fields that + have them. They have the form Default_StructName_FieldName. + Because the getter methods handle defaulted values, + direct use of these constants should be rare. + - Enums are given type names and maps from names to values. + Enum values are prefixed by the enclosing message's name, or by the + enum's type name if it is a top-level enum. Enum types have a String + method, and a Enum method to assist in message construction. + - Nested messages, groups and enums have type names prefixed with the name of + the surrounding message type. + - Extensions are given descriptor names that start with E_, + followed by an underscore-delimited list of the nested messages + that contain it (if any) followed by the CamelCased name of the + extension field itself. HasExtension, ClearExtension, GetExtension + and SetExtension are functions for manipulating extensions. + - Marshal and Unmarshal are functions to encode and decode the wire format. + +The simplest way to describe this is to see an example. +Given file test.proto, containing + + package example; + + enum FOO { X = 17; } + + message Test { + required string label = 1; + optional int32 type = 2 [default=77]; + repeated int64 reps = 3; + optional group OptionalGroup = 4 { + required string RequiredField = 5; + } + } + +The resulting file, test.pb.go, is: + + package example + + import proto "github.com/gogo/protobuf/proto" + import math "math" + + type FOO int32 + const ( + FOO_X FOO = 17 + ) + var FOO_name = map[int32]string{ + 17: "X", + } + var FOO_value = map[string]int32{ + "X": 17, + } + + func (x FOO) Enum() *FOO { + p := new(FOO) + *p = x + return p + } + func (x FOO) String() string { + return proto.EnumName(FOO_name, int32(x)) + } + func (x *FOO) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(FOO_value, data) + if err != nil { + return err + } + *x = FOO(value) + return nil + } + + type Test struct { + Label *string `protobuf:"bytes,1,req,name=label" json:"label,omitempty"` + Type *int32 `protobuf:"varint,2,opt,name=type,def=77" json:"type,omitempty"` + Reps []int64 `protobuf:"varint,3,rep,name=reps" json:"reps,omitempty"` + Optionalgroup *Test_OptionalGroup `protobuf:"group,4,opt,name=OptionalGroup" json:"optionalgroup,omitempty"` + XXX_unrecognized []byte `json:"-"` + } + func (m *Test) Reset() { *m = Test{} } + func (m *Test) String() string { return proto.CompactTextString(m) } + func (*Test) ProtoMessage() {} + const Default_Test_Type int32 = 77 + + func (m *Test) GetLabel() string { + if m != nil && m.Label != nil { + return *m.Label + } + return "" + } + + func (m *Test) GetType() int32 { + if m != nil && m.Type != nil { + return *m.Type + } + return Default_Test_Type + } + + func (m *Test) GetOptionalgroup() *Test_OptionalGroup { + if m != nil { + return m.Optionalgroup + } + return nil + } + + type Test_OptionalGroup struct { + RequiredField *string `protobuf:"bytes,5,req" json:"RequiredField,omitempty"` + } + func (m *Test_OptionalGroup) Reset() { *m = Test_OptionalGroup{} } + func (m *Test_OptionalGroup) String() string { return proto.CompactTextString(m) } + + func (m *Test_OptionalGroup) GetRequiredField() string { + if m != nil && m.RequiredField != nil { + return *m.RequiredField + } + return "" + } + + func init() { + proto.RegisterEnum("example.FOO", FOO_name, FOO_value) + } + +To create and play with a Test object: + +package main + + import ( + "log" + + "github.com/gogo/protobuf/proto" + pb "./example.pb" + ) + + func main() { + test := &pb.Test{ + Label: proto.String("hello"), + Type: proto.Int32(17), + Optionalgroup: &pb.Test_OptionalGroup{ + RequiredField: proto.String("good bye"), + }, + } + data, err := proto.Marshal(test) + if err != nil { + log.Fatal("marshaling error: ", err) + } + newTest := &pb.Test{} + err = proto.Unmarshal(data, newTest) + if err != nil { + log.Fatal("unmarshaling error: ", err) + } + // Now test and newTest contain the same data. + if test.GetLabel() != newTest.GetLabel() { + log.Fatalf("data mismatch %q != %q", test.GetLabel(), newTest.GetLabel()) + } + // etc. + } +*/ +package proto + +import ( + "encoding/json" + "fmt" + "log" + "reflect" + "sort" + "strconv" + "sync" +) + +// Message is implemented by generated protocol buffer messages. +type Message interface { + Reset() + String() string + ProtoMessage() +} + +// Stats records allocation details about the protocol buffer encoders +// and decoders. Useful for tuning the library itself. +type Stats struct { + Emalloc uint64 // mallocs in encode + Dmalloc uint64 // mallocs in decode + Encode uint64 // number of encodes + Decode uint64 // number of decodes + Chit uint64 // number of cache hits + Cmiss uint64 // number of cache misses + Size uint64 // number of sizes +} + +// Set to true to enable stats collection. +const collectStats = false + +var stats Stats + +// GetStats returns a copy of the global Stats structure. +func GetStats() Stats { return stats } + +// A Buffer is a buffer manager for marshaling and unmarshaling +// protocol buffers. It may be reused between invocations to +// reduce memory usage. It is not necessary to use a Buffer; +// the global functions Marshal and Unmarshal create a +// temporary Buffer and are fine for most applications. +type Buffer struct { + buf []byte // encode/decode byte stream + index int // write point + + // pools of basic types to amortize allocation. + bools []bool + uint32s []uint32 + uint64s []uint64 + + // extra pools, only used with pointer_reflect.go + int32s []int32 + int64s []int64 + float32s []float32 + float64s []float64 +} + +// NewBuffer allocates a new Buffer and initializes its internal data to +// the contents of the argument slice. +func NewBuffer(e []byte) *Buffer { + return &Buffer{buf: e} +} + +// Reset resets the Buffer, ready for marshaling a new protocol buffer. +func (p *Buffer) Reset() { + p.buf = p.buf[0:0] // for reading/writing + p.index = 0 // for reading +} + +// SetBuf replaces the internal buffer with the slice, +// ready for unmarshaling the contents of the slice. +func (p *Buffer) SetBuf(s []byte) { + p.buf = s + p.index = 0 +} + +// Bytes returns the contents of the Buffer. +func (p *Buffer) Bytes() []byte { return p.buf } + +/* + * Helper routines for simplifying the creation of optional fields of basic type. + */ + +// Bool is a helper routine that allocates a new bool value +// to store v and returns a pointer to it. +func Bool(v bool) *bool { + return &v +} + +// Int32 is a helper routine that allocates a new int32 value +// to store v and returns a pointer to it. +func Int32(v int32) *int32 { + return &v +} + +// Int is a helper routine that allocates a new int32 value +// to store v and returns a pointer to it, but unlike Int32 +// its argument value is an int. +func Int(v int) *int32 { + p := new(int32) + *p = int32(v) + return p +} + +// Int64 is a helper routine that allocates a new int64 value +// to store v and returns a pointer to it. +func Int64(v int64) *int64 { + return &v +} + +// Float32 is a helper routine that allocates a new float32 value +// to store v and returns a pointer to it. +func Float32(v float32) *float32 { + return &v +} + +// Float64 is a helper routine that allocates a new float64 value +// to store v and returns a pointer to it. +func Float64(v float64) *float64 { + return &v +} + +// Uint32 is a helper routine that allocates a new uint32 value +// to store v and returns a pointer to it. +func Uint32(v uint32) *uint32 { + return &v +} + +// Uint64 is a helper routine that allocates a new uint64 value +// to store v and returns a pointer to it. +func Uint64(v uint64) *uint64 { + return &v +} + +// String is a helper routine that allocates a new string value +// to store v and returns a pointer to it. +func String(v string) *string { + return &v +} + +// EnumName is a helper function to simplify printing protocol buffer enums +// by name. Given an enum map and a value, it returns a useful string. +func EnumName(m map[int32]string, v int32) string { + s, ok := m[v] + if ok { + return s + } + return strconv.Itoa(int(v)) +} + +// UnmarshalJSONEnum is a helper function to simplify recovering enum int values +// from their JSON-encoded representation. Given a map from the enum's symbolic +// names to its int values, and a byte buffer containing the JSON-encoded +// value, it returns an int32 that can be cast to the enum type by the caller. +// +// The function can deal with both JSON representations, numeric and symbolic. +func UnmarshalJSONEnum(m map[string]int32, data []byte, enumName string) (int32, error) { + if data[0] == '"' { + // New style: enums are strings. + var repr string + if err := json.Unmarshal(data, &repr); err != nil { + return -1, err + } + val, ok := m[repr] + if !ok { + return 0, fmt.Errorf("unrecognized enum %s value %q", enumName, repr) + } + return val, nil + } + // Old style: enums are ints. + var val int32 + if err := json.Unmarshal(data, &val); err != nil { + return 0, fmt.Errorf("cannot unmarshal %#q into enum %s", data, enumName) + } + return val, nil +} + +// DebugPrint dumps the encoded data in b in a debugging format with a header +// including the string s. Used in testing but made available for general debugging. +func (p *Buffer) DebugPrint(s string, b []byte) { + var u uint64 + + obuf := p.buf + index := p.index + p.buf = b + p.index = 0 + depth := 0 + + fmt.Printf("\n--- %s ---\n", s) + +out: + for { + for i := 0; i < depth; i++ { + fmt.Print(" ") + } + + index := p.index + if index == len(p.buf) { + break + } + + op, err := p.DecodeVarint() + if err != nil { + fmt.Printf("%3d: fetching op err %v\n", index, err) + break out + } + tag := op >> 3 + wire := op & 7 + + switch wire { + default: + fmt.Printf("%3d: t=%3d unknown wire=%d\n", + index, tag, wire) + break out + + case WireBytes: + var r []byte + + r, err = p.DecodeRawBytes(false) + if err != nil { + break out + } + fmt.Printf("%3d: t=%3d bytes [%d]", index, tag, len(r)) + if len(r) <= 6 { + for i := 0; i < len(r); i++ { + fmt.Printf(" %.2x", r[i]) + } + } else { + for i := 0; i < 3; i++ { + fmt.Printf(" %.2x", r[i]) + } + fmt.Printf(" ..") + for i := len(r) - 3; i < len(r); i++ { + fmt.Printf(" %.2x", r[i]) + } + } + fmt.Printf("\n") + + case WireFixed32: + u, err = p.DecodeFixed32() + if err != nil { + fmt.Printf("%3d: t=%3d fix32 err %v\n", index, tag, err) + break out + } + fmt.Printf("%3d: t=%3d fix32 %d\n", index, tag, u) + + case WireFixed64: + u, err = p.DecodeFixed64() + if err != nil { + fmt.Printf("%3d: t=%3d fix64 err %v\n", index, tag, err) + break out + } + fmt.Printf("%3d: t=%3d fix64 %d\n", index, tag, u) + break + + case WireVarint: + u, err = p.DecodeVarint() + if err != nil { + fmt.Printf("%3d: t=%3d varint err %v\n", index, tag, err) + break out + } + fmt.Printf("%3d: t=%3d varint %d\n", index, tag, u) + + case WireStartGroup: + if err != nil { + fmt.Printf("%3d: t=%3d start err %v\n", index, tag, err) + break out + } + fmt.Printf("%3d: t=%3d start\n", index, tag) + depth++ + + case WireEndGroup: + depth-- + if err != nil { + fmt.Printf("%3d: t=%3d end err %v\n", index, tag, err) + break out + } + fmt.Printf("%3d: t=%3d end\n", index, tag) + } + } + + if depth != 0 { + fmt.Printf("%3d: start-end not balanced %d\n", p.index, depth) + } + fmt.Printf("\n") + + p.buf = obuf + p.index = index +} + +// SetDefaults sets unset protocol buffer fields to their default values. +// It only modifies fields that are both unset and have defined defaults. +// It recursively sets default values in any non-nil sub-messages. +func SetDefaults(pb Message) { + setDefaults(reflect.ValueOf(pb), true, false) +} + +// v is a pointer to a struct. +func setDefaults(v reflect.Value, recur, zeros bool) { + v = v.Elem() + + defaultMu.RLock() + dm, ok := defaults[v.Type()] + defaultMu.RUnlock() + if !ok { + dm = buildDefaultMessage(v.Type()) + defaultMu.Lock() + defaults[v.Type()] = dm + defaultMu.Unlock() + } + + for _, sf := range dm.scalars { + f := v.Field(sf.index) + if !f.IsNil() { + // field already set + continue + } + dv := sf.value + if dv == nil && !zeros { + // no explicit default, and don't want to set zeros + continue + } + fptr := f.Addr().Interface() // **T + // TODO: Consider batching the allocations we do here. + switch sf.kind { + case reflect.Bool: + b := new(bool) + if dv != nil { + *b = dv.(bool) + } + *(fptr.(**bool)) = b + case reflect.Float32: + f := new(float32) + if dv != nil { + *f = dv.(float32) + } + *(fptr.(**float32)) = f + case reflect.Float64: + f := new(float64) + if dv != nil { + *f = dv.(float64) + } + *(fptr.(**float64)) = f + case reflect.Int32: + // might be an enum + if ft := f.Type(); ft != int32PtrType { + // enum + f.Set(reflect.New(ft.Elem())) + if dv != nil { + f.Elem().SetInt(int64(dv.(int32))) + } + } else { + // int32 field + i := new(int32) + if dv != nil { + *i = dv.(int32) + } + *(fptr.(**int32)) = i + } + case reflect.Int64: + i := new(int64) + if dv != nil { + *i = dv.(int64) + } + *(fptr.(**int64)) = i + case reflect.String: + s := new(string) + if dv != nil { + *s = dv.(string) + } + *(fptr.(**string)) = s + case reflect.Uint8: + // exceptional case: []byte + var b []byte + if dv != nil { + db := dv.([]byte) + b = make([]byte, len(db)) + copy(b, db) + } else { + b = []byte{} + } + *(fptr.(*[]byte)) = b + case reflect.Uint32: + u := new(uint32) + if dv != nil { + *u = dv.(uint32) + } + *(fptr.(**uint32)) = u + case reflect.Uint64: + u := new(uint64) + if dv != nil { + *u = dv.(uint64) + } + *(fptr.(**uint64)) = u + default: + log.Printf("proto: can't set default for field %v (sf.kind=%v)", f, sf.kind) + } + } + + for _, ni := range dm.nested { + f := v.Field(ni) + // f is *T or []*T or map[T]*T + switch f.Kind() { + case reflect.Ptr: + if f.IsNil() { + continue + } + setDefaults(f, recur, zeros) + + case reflect.Slice: + for i := 0; i < f.Len(); i++ { + e := f.Index(i) + if e.IsNil() { + continue + } + setDefaults(e, recur, zeros) + } + + case reflect.Map: + for _, k := range f.MapKeys() { + e := f.MapIndex(k) + if e.IsNil() { + continue + } + setDefaults(e, recur, zeros) + } + } + } +} + +var ( + // defaults maps a protocol buffer struct type to a slice of the fields, + // with its scalar fields set to their proto-declared non-zero default values. + defaultMu sync.RWMutex + defaults = make(map[reflect.Type]defaultMessage) + + int32PtrType = reflect.TypeOf((*int32)(nil)) +) + +// defaultMessage represents information about the default values of a message. +type defaultMessage struct { + scalars []scalarField + nested []int // struct field index of nested messages +} + +type scalarField struct { + index int // struct field index + kind reflect.Kind // element type (the T in *T or []T) + value interface{} // the proto-declared default value, or nil +} + +// t is a struct type. +func buildDefaultMessage(t reflect.Type) (dm defaultMessage) { + sprop := GetProperties(t) + for _, prop := range sprop.Prop { + fi, ok := sprop.decoderTags.get(prop.Tag) + if !ok { + // XXX_unrecognized + continue + } + ft := t.Field(fi).Type + + sf, nested, err := fieldDefault(ft, prop) + switch { + case err != nil: + log.Print(err) + case nested: + dm.nested = append(dm.nested, fi) + case sf != nil: + sf.index = fi + dm.scalars = append(dm.scalars, *sf) + } + } + + return dm +} + +// fieldDefault returns the scalarField for field type ft. +// sf will be nil if the field can not have a default. +// nestedMessage will be true if this is a nested message. +// Note that sf.index is not set on return. +func fieldDefault(ft reflect.Type, prop *Properties) (sf *scalarField, nestedMessage bool, err error) { + var canHaveDefault bool + switch ft.Kind() { + case reflect.Ptr: + if ft.Elem().Kind() == reflect.Struct { + nestedMessage = true + } else { + canHaveDefault = true // proto2 scalar field + } + + case reflect.Slice: + switch ft.Elem().Kind() { + case reflect.Ptr: + nestedMessage = true // repeated message + case reflect.Uint8: + canHaveDefault = true // bytes field + } + + case reflect.Map: + if ft.Elem().Kind() == reflect.Ptr { + nestedMessage = true // map with message values + } + } + + if !canHaveDefault { + if nestedMessage { + return nil, true, nil + } + return nil, false, nil + } + + // We now know that ft is a pointer or slice. + sf = &scalarField{kind: ft.Elem().Kind()} + + // scalar fields without defaults + if !prop.HasDefault { + return sf, false, nil + } + + // a scalar field: either *T or []byte + switch ft.Elem().Kind() { + case reflect.Bool: + x, err := strconv.ParseBool(prop.Default) + if err != nil { + return nil, false, fmt.Errorf("proto: bad default bool %q: %v", prop.Default, err) + } + sf.value = x + case reflect.Float32: + x, err := strconv.ParseFloat(prop.Default, 32) + if err != nil { + return nil, false, fmt.Errorf("proto: bad default float32 %q: %v", prop.Default, err) + } + sf.value = float32(x) + case reflect.Float64: + x, err := strconv.ParseFloat(prop.Default, 64) + if err != nil { + return nil, false, fmt.Errorf("proto: bad default float64 %q: %v", prop.Default, err) + } + sf.value = x + case reflect.Int32: + x, err := strconv.ParseInt(prop.Default, 10, 32) + if err != nil { + return nil, false, fmt.Errorf("proto: bad default int32 %q: %v", prop.Default, err) + } + sf.value = int32(x) + case reflect.Int64: + x, err := strconv.ParseInt(prop.Default, 10, 64) + if err != nil { + return nil, false, fmt.Errorf("proto: bad default int64 %q: %v", prop.Default, err) + } + sf.value = x + case reflect.String: + sf.value = prop.Default + case reflect.Uint8: + // []byte (not *uint8) + sf.value = []byte(prop.Default) + case reflect.Uint32: + x, err := strconv.ParseUint(prop.Default, 10, 32) + if err != nil { + return nil, false, fmt.Errorf("proto: bad default uint32 %q: %v", prop.Default, err) + } + sf.value = uint32(x) + case reflect.Uint64: + x, err := strconv.ParseUint(prop.Default, 10, 64) + if err != nil { + return nil, false, fmt.Errorf("proto: bad default uint64 %q: %v", prop.Default, err) + } + sf.value = x + default: + return nil, false, fmt.Errorf("proto: unhandled def kind %v", ft.Elem().Kind()) + } + + return sf, false, nil +} + +// Map fields may have key types of non-float scalars, strings and enums. +// The easiest way to sort them in some deterministic order is to use fmt. +// If this turns out to be inefficient we can always consider other options, +// such as doing a Schwartzian transform. + +func mapKeys(vs []reflect.Value) sort.Interface { + s := mapKeySorter{ + vs: vs, + // default Less function: textual comparison + less: func(a, b reflect.Value) bool { + return fmt.Sprint(a.Interface()) < fmt.Sprint(b.Interface()) + }, + } + + // Type specialization per https://developers.google.com/protocol-buffers/docs/proto#maps; + // numeric keys are sorted numerically. + if len(vs) == 0 { + return s + } + switch vs[0].Kind() { + case reflect.Int32, reflect.Int64: + s.less = func(a, b reflect.Value) bool { return a.Int() < b.Int() } + case reflect.Uint32, reflect.Uint64: + s.less = func(a, b reflect.Value) bool { return a.Uint() < b.Uint() } + } + + return s +} + +type mapKeySorter struct { + vs []reflect.Value + less func(a, b reflect.Value) bool +} + +func (s mapKeySorter) Len() int { return len(s.vs) } +func (s mapKeySorter) Swap(i, j int) { s.vs[i], s.vs[j] = s.vs[j], s.vs[i] } +func (s mapKeySorter) Less(i, j int) bool { + return s.less(s.vs[i], s.vs[j]) +} + +// isProto3Zero reports whether v is a zero proto3 value. +func isProto3Zero(v reflect.Value) bool { + switch v.Kind() { + case reflect.Bool: + return !v.Bool() + case reflect.Int32, reflect.Int64: + return v.Int() == 0 + case reflect.Uint32, reflect.Uint64: + return v.Uint() == 0 + case reflect.Float32, reflect.Float64: + return v.Float() == 0 + case reflect.String: + return v.String() == "" + } + return false +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/lib_gogo.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/lib_gogo.go new file mode 100644 index 0000000000000..a6c2c06b23de2 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/lib_gogo.go @@ -0,0 +1,40 @@ +// Copyright (c) 2013, Vastech SA (PTY) LTD. All rights reserved. +// http://github.com/gogo/protobuf/gogoproto +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +import ( + "encoding/json" + "strconv" +) + +func MarshalJSONEnum(m map[int32]string, value int32) ([]byte, error) { + s, ok := m[value] + if !ok { + s = strconv.Itoa(int(value)) + } + return json.Marshal(s) +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/message_set.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/message_set.go new file mode 100644 index 0000000000000..9d912bce19bb6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/message_set.go @@ -0,0 +1,287 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +/* + * Support for message sets. + */ + +import ( + "bytes" + "encoding/json" + "errors" + "fmt" + "reflect" + "sort" +) + +// ErrNoMessageTypeId occurs when a protocol buffer does not have a message type ID. +// A message type ID is required for storing a protocol buffer in a message set. +var ErrNoMessageTypeId = errors.New("proto does not have a message type ID") + +// The first two types (_MessageSet_Item and MessageSet) +// model what the protocol compiler produces for the following protocol message: +// message MessageSet { +// repeated group Item = 1 { +// required int32 type_id = 2; +// required string message = 3; +// }; +// } +// That is the MessageSet wire format. We can't use a proto to generate these +// because that would introduce a circular dependency between it and this package. +// +// When a proto1 proto has a field that looks like: +// optional message info = 3; +// the protocol compiler produces a field in the generated struct that looks like: +// Info *_proto_.MessageSet `protobuf:"bytes,3,opt,name=info"` +// The package is automatically inserted so there is no need for that proto file to +// import this package. + +type _MessageSet_Item struct { + TypeId *int32 `protobuf:"varint,2,req,name=type_id"` + Message []byte `protobuf:"bytes,3,req,name=message"` +} + +type MessageSet struct { + Item []*_MessageSet_Item `protobuf:"group,1,rep"` + XXX_unrecognized []byte + // TODO: caching? +} + +// Make sure MessageSet is a Message. +var _ Message = (*MessageSet)(nil) + +// messageTypeIder is an interface satisfied by a protocol buffer type +// that may be stored in a MessageSet. +type messageTypeIder interface { + MessageTypeId() int32 +} + +func (ms *MessageSet) find(pb Message) *_MessageSet_Item { + mti, ok := pb.(messageTypeIder) + if !ok { + return nil + } + id := mti.MessageTypeId() + for _, item := range ms.Item { + if *item.TypeId == id { + return item + } + } + return nil +} + +func (ms *MessageSet) Has(pb Message) bool { + if ms.find(pb) != nil { + return true + } + return false +} + +func (ms *MessageSet) Unmarshal(pb Message) error { + if item := ms.find(pb); item != nil { + return Unmarshal(item.Message, pb) + } + if _, ok := pb.(messageTypeIder); !ok { + return ErrNoMessageTypeId + } + return nil // TODO: return error instead? +} + +func (ms *MessageSet) Marshal(pb Message) error { + msg, err := Marshal(pb) + if err != nil { + return err + } + if item := ms.find(pb); item != nil { + // reuse existing item + item.Message = msg + return nil + } + + mti, ok := pb.(messageTypeIder) + if !ok { + return ErrNoMessageTypeId + } + + mtid := mti.MessageTypeId() + ms.Item = append(ms.Item, &_MessageSet_Item{ + TypeId: &mtid, + Message: msg, + }) + return nil +} + +func (ms *MessageSet) Reset() { *ms = MessageSet{} } +func (ms *MessageSet) String() string { return CompactTextString(ms) } +func (*MessageSet) ProtoMessage() {} + +// Support for the message_set_wire_format message option. + +func skipVarint(buf []byte) []byte { + i := 0 + for ; buf[i]&0x80 != 0; i++ { + } + return buf[i+1:] +} + +// MarshalMessageSet encodes the extension map represented by m in the message set wire format. +// It is called by generated Marshal methods on protocol buffer messages with the message_set_wire_format option. +func MarshalMessageSet(m map[int32]Extension) ([]byte, error) { + if err := encodeExtensionMap(m); err != nil { + return nil, err + } + + // Sort extension IDs to provide a deterministic encoding. + // See also enc_map in encode.go. + ids := make([]int, 0, len(m)) + for id := range m { + ids = append(ids, int(id)) + } + sort.Ints(ids) + + ms := &MessageSet{Item: make([]*_MessageSet_Item, 0, len(m))} + for _, id := range ids { + e := m[int32(id)] + // Remove the wire type and field number varint, as well as the length varint. + msg := skipVarint(skipVarint(e.enc)) + + ms.Item = append(ms.Item, &_MessageSet_Item{ + TypeId: Int32(int32(id)), + Message: msg, + }) + } + return Marshal(ms) +} + +// UnmarshalMessageSet decodes the extension map encoded in buf in the message set wire format. +// It is called by generated Unmarshal methods on protocol buffer messages with the message_set_wire_format option. +func UnmarshalMessageSet(buf []byte, m map[int32]Extension) error { + ms := new(MessageSet) + if err := Unmarshal(buf, ms); err != nil { + return err + } + for _, item := range ms.Item { + id := *item.TypeId + msg := item.Message + + // Restore wire type and field number varint, plus length varint. + // Be careful to preserve duplicate items. + b := EncodeVarint(uint64(id)<<3 | WireBytes) + if ext, ok := m[id]; ok { + // Existing data; rip off the tag and length varint + // so we join the new data correctly. + // We can assume that ext.enc is set because we are unmarshaling. + o := ext.enc[len(b):] // skip wire type and field number + _, n := DecodeVarint(o) // calculate length of length varint + o = o[n:] // skip length varint + msg = append(o, msg...) // join old data and new data + } + b = append(b, EncodeVarint(uint64(len(msg)))...) + b = append(b, msg...) + + m[id] = Extension{enc: b} + } + return nil +} + +// MarshalMessageSetJSON encodes the extension map represented by m in JSON format. +// It is called by generated MarshalJSON methods on protocol buffer messages with the message_set_wire_format option. +func MarshalMessageSetJSON(m map[int32]Extension) ([]byte, error) { + var b bytes.Buffer + b.WriteByte('{') + + // Process the map in key order for deterministic output. + ids := make([]int32, 0, len(m)) + for id := range m { + ids = append(ids, id) + } + sort.Sort(int32Slice(ids)) // int32Slice defined in text.go + + for i, id := range ids { + ext := m[id] + if i > 0 { + b.WriteByte(',') + } + + msd, ok := messageSetMap[id] + if !ok { + // Unknown type; we can't render it, so skip it. + continue + } + fmt.Fprintf(&b, `"[%s]":`, msd.name) + + x := ext.value + if x == nil { + x = reflect.New(msd.t.Elem()).Interface() + if err := Unmarshal(ext.enc, x.(Message)); err != nil { + return nil, err + } + } + d, err := json.Marshal(x) + if err != nil { + return nil, err + } + b.Write(d) + } + b.WriteByte('}') + return b.Bytes(), nil +} + +// UnmarshalMessageSetJSON decodes the extension map encoded in buf in JSON format. +// It is called by generated UnmarshalJSON methods on protocol buffer messages with the message_set_wire_format option. +func UnmarshalMessageSetJSON(buf []byte, m map[int32]Extension) error { + // Common-case fast path. + if len(buf) == 0 || bytes.Equal(buf, []byte("{}")) { + return nil + } + + // This is fairly tricky, and it's not clear that it is needed. + return errors.New("TODO: UnmarshalMessageSetJSON not yet implemented") +} + +// A global registry of types that can be used in a MessageSet. + +var messageSetMap = make(map[int32]messageSetDesc) + +type messageSetDesc struct { + t reflect.Type // pointer to struct + name string +} + +// RegisterMessageSetType is called from the generated code. +func RegisterMessageSetType(m Message, fieldNum int32, name string) { + messageSetMap[fieldNum] = messageSetDesc{ + t: reflect.TypeOf(m), + name: name, + } +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/message_set_test.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/message_set_test.go new file mode 100644 index 0000000000000..7c29bccf4b1d9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/message_set_test.go @@ -0,0 +1,66 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2014 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +import ( + "bytes" + "testing" +) + +func TestUnmarshalMessageSetWithDuplicate(t *testing.T) { + // Check that a repeated message set entry will be concatenated. + in := &MessageSet{ + Item: []*_MessageSet_Item{ + {TypeId: Int32(12345), Message: []byte("hoo")}, + {TypeId: Int32(12345), Message: []byte("hah")}, + }, + } + b, err := Marshal(in) + if err != nil { + t.Fatalf("Marshal: %v", err) + } + t.Logf("Marshaled bytes: %q", b) + + m := make(map[int32]Extension) + if err := UnmarshalMessageSet(b, m); err != nil { + t.Fatalf("UnmarshalMessageSet: %v", err) + } + ext, ok := m[12345] + if !ok { + t.Fatalf("Didn't retrieve extension 12345; map is %v", m) + } + // Skip wire type/field number and length varints. + got := skipVarint(skipVarint(ext.enc)) + if want := []byte("hoohah"); !bytes.Equal(got, want) { + t.Errorf("Combined extension is %q, want %q", got, want) + } +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/pointer_reflect.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/pointer_reflect.go new file mode 100644 index 0000000000000..749919d250a1a --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/pointer_reflect.go @@ -0,0 +1,479 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2012 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// +build appengine + +// This file contains an implementation of proto field accesses using package reflect. +// It is slower than the code in pointer_unsafe.go but it avoids package unsafe and can +// be used on App Engine. + +package proto + +import ( + "math" + "reflect" +) + +// A structPointer is a pointer to a struct. +type structPointer struct { + v reflect.Value +} + +// toStructPointer returns a structPointer equivalent to the given reflect value. +// The reflect value must itself be a pointer to a struct. +func toStructPointer(v reflect.Value) structPointer { + return structPointer{v} +} + +// IsNil reports whether p is nil. +func structPointer_IsNil(p structPointer) bool { + return p.v.IsNil() +} + +// Interface returns the struct pointer as an interface value. +func structPointer_Interface(p structPointer, _ reflect.Type) interface{} { + return p.v.Interface() +} + +// A field identifies a field in a struct, accessible from a structPointer. +// In this implementation, a field is identified by the sequence of field indices +// passed to reflect's FieldByIndex. +type field []int + +// toField returns a field equivalent to the given reflect field. +func toField(f *reflect.StructField) field { + return f.Index +} + +// invalidField is an invalid field identifier. +var invalidField = field(nil) + +// IsValid reports whether the field identifier is valid. +func (f field) IsValid() bool { return f != nil } + +// field returns the given field in the struct as a reflect value. +func structPointer_field(p structPointer, f field) reflect.Value { + // Special case: an extension map entry with a value of type T + // passes a *T to the struct-handling code with a zero field, + // expecting that it will be treated as equivalent to *struct{ X T }, + // which has the same memory layout. We have to handle that case + // specially, because reflect will panic if we call FieldByIndex on a + // non-struct. + if f == nil { + return p.v.Elem() + } + + return p.v.Elem().FieldByIndex(f) +} + +// ifield returns the given field in the struct as an interface value. +func structPointer_ifield(p structPointer, f field) interface{} { + return structPointer_field(p, f).Addr().Interface() +} + +// Bytes returns the address of a []byte field in the struct. +func structPointer_Bytes(p structPointer, f field) *[]byte { + return structPointer_ifield(p, f).(*[]byte) +} + +// BytesSlice returns the address of a [][]byte field in the struct. +func structPointer_BytesSlice(p structPointer, f field) *[][]byte { + return structPointer_ifield(p, f).(*[][]byte) +} + +// Bool returns the address of a *bool field in the struct. +func structPointer_Bool(p structPointer, f field) **bool { + return structPointer_ifield(p, f).(**bool) +} + +// BoolVal returns the address of a bool field in the struct. +func structPointer_BoolVal(p structPointer, f field) *bool { + return structPointer_ifield(p, f).(*bool) +} + +// BoolSlice returns the address of a []bool field in the struct. +func structPointer_BoolSlice(p structPointer, f field) *[]bool { + return structPointer_ifield(p, f).(*[]bool) +} + +// String returns the address of a *string field in the struct. +func structPointer_String(p structPointer, f field) **string { + return structPointer_ifield(p, f).(**string) +} + +// StringVal returns the address of a string field in the struct. +func structPointer_StringVal(p structPointer, f field) *string { + return structPointer_ifield(p, f).(*string) +} + +// StringSlice returns the address of a []string field in the struct. +func structPointer_StringSlice(p structPointer, f field) *[]string { + return structPointer_ifield(p, f).(*[]string) +} + +// ExtMap returns the address of an extension map field in the struct. +func structPointer_ExtMap(p structPointer, f field) *map[int32]Extension { + return structPointer_ifield(p, f).(*map[int32]Extension) +} + +// NewAt returns the reflect.Value for a pointer to a field in the struct. +func structPointer_NewAt(p structPointer, f field, typ reflect.Type) reflect.Value { + return structPointer_field(p, f).Addr() +} + +// SetStructPointer writes a *struct field in the struct. +func structPointer_SetStructPointer(p structPointer, f field, q structPointer) { + structPointer_field(p, f).Set(q.v) +} + +// GetStructPointer reads a *struct field in the struct. +func structPointer_GetStructPointer(p structPointer, f field) structPointer { + return structPointer{structPointer_field(p, f)} +} + +// StructPointerSlice the address of a []*struct field in the struct. +func structPointer_StructPointerSlice(p structPointer, f field) structPointerSlice { + return structPointerSlice{structPointer_field(p, f)} +} + +// A structPointerSlice represents the address of a slice of pointers to structs +// (themselves messages or groups). That is, v.Type() is *[]*struct{...}. +type structPointerSlice struct { + v reflect.Value +} + +func (p structPointerSlice) Len() int { return p.v.Len() } +func (p structPointerSlice) Index(i int) structPointer { return structPointer{p.v.Index(i)} } +func (p structPointerSlice) Append(q structPointer) { + p.v.Set(reflect.Append(p.v, q.v)) +} + +var ( + int32Type = reflect.TypeOf(int32(0)) + uint32Type = reflect.TypeOf(uint32(0)) + float32Type = reflect.TypeOf(float32(0)) + int64Type = reflect.TypeOf(int64(0)) + uint64Type = reflect.TypeOf(uint64(0)) + float64Type = reflect.TypeOf(float64(0)) +) + +// A word32 represents a field of type *int32, *uint32, *float32, or *enum. +// That is, v.Type() is *int32, *uint32, *float32, or *enum and v is assignable. +type word32 struct { + v reflect.Value +} + +// IsNil reports whether p is nil. +func word32_IsNil(p word32) bool { + return p.v.IsNil() +} + +// Set sets p to point at a newly allocated word with bits set to x. +func word32_Set(p word32, o *Buffer, x uint32) { + t := p.v.Type().Elem() + switch t { + case int32Type: + if len(o.int32s) == 0 { + o.int32s = make([]int32, uint32PoolSize) + } + o.int32s[0] = int32(x) + p.v.Set(reflect.ValueOf(&o.int32s[0])) + o.int32s = o.int32s[1:] + return + case uint32Type: + if len(o.uint32s) == 0 { + o.uint32s = make([]uint32, uint32PoolSize) + } + o.uint32s[0] = x + p.v.Set(reflect.ValueOf(&o.uint32s[0])) + o.uint32s = o.uint32s[1:] + return + case float32Type: + if len(o.float32s) == 0 { + o.float32s = make([]float32, uint32PoolSize) + } + o.float32s[0] = math.Float32frombits(x) + p.v.Set(reflect.ValueOf(&o.float32s[0])) + o.float32s = o.float32s[1:] + return + } + + // must be enum + p.v.Set(reflect.New(t)) + p.v.Elem().SetInt(int64(int32(x))) +} + +// Get gets the bits pointed at by p, as a uint32. +func word32_Get(p word32) uint32 { + elem := p.v.Elem() + switch elem.Kind() { + case reflect.Int32: + return uint32(elem.Int()) + case reflect.Uint32: + return uint32(elem.Uint()) + case reflect.Float32: + return math.Float32bits(float32(elem.Float())) + } + panic("unreachable") +} + +// Word32 returns a reference to a *int32, *uint32, *float32, or *enum field in the struct. +func structPointer_Word32(p structPointer, f field) word32 { + return word32{structPointer_field(p, f)} +} + +// A word32Val represents a field of type int32, uint32, float32, or enum. +// That is, v.Type() is int32, uint32, float32, or enum and v is assignable. +type word32Val struct { + v reflect.Value +} + +// Set sets *p to x. +func word32Val_Set(p word32Val, x uint32) { + switch p.v.Type() { + case int32Type: + p.v.SetInt(int64(x)) + return + case uint32Type: + p.v.SetUint(uint64(x)) + return + case float32Type: + p.v.SetFloat(float64(math.Float32frombits(x))) + return + } + + // must be enum + p.v.SetInt(int64(int32(x))) +} + +// Get gets the bits pointed at by p, as a uint32. +func word32Val_Get(p word32Val) uint32 { + elem := p.v + switch elem.Kind() { + case reflect.Int32: + return uint32(elem.Int()) + case reflect.Uint32: + return uint32(elem.Uint()) + case reflect.Float32: + return math.Float32bits(float32(elem.Float())) + } + panic("unreachable") +} + +// Word32Val returns a reference to a int32, uint32, float32, or enum field in the struct. +func structPointer_Word32Val(p structPointer, f field) word32Val { + return word32Val{structPointer_field(p, f)} +} + +// A word32Slice is a slice of 32-bit values. +// That is, v.Type() is []int32, []uint32, []float32, or []enum. +type word32Slice struct { + v reflect.Value +} + +func (p word32Slice) Append(x uint32) { + n, m := p.v.Len(), p.v.Cap() + if n < m { + p.v.SetLen(n + 1) + } else { + t := p.v.Type().Elem() + p.v.Set(reflect.Append(p.v, reflect.Zero(t))) + } + elem := p.v.Index(n) + switch elem.Kind() { + case reflect.Int32: + elem.SetInt(int64(int32(x))) + case reflect.Uint32: + elem.SetUint(uint64(x)) + case reflect.Float32: + elem.SetFloat(float64(math.Float32frombits(x))) + } +} + +func (p word32Slice) Len() int { + return p.v.Len() +} + +func (p word32Slice) Index(i int) uint32 { + elem := p.v.Index(i) + switch elem.Kind() { + case reflect.Int32: + return uint32(elem.Int()) + case reflect.Uint32: + return uint32(elem.Uint()) + case reflect.Float32: + return math.Float32bits(float32(elem.Float())) + } + panic("unreachable") +} + +// Word32Slice returns a reference to a []int32, []uint32, []float32, or []enum field in the struct. +func structPointer_Word32Slice(p structPointer, f field) word32Slice { + return word32Slice{structPointer_field(p, f)} +} + +// word64 is like word32 but for 64-bit values. +type word64 struct { + v reflect.Value +} + +func word64_Set(p word64, o *Buffer, x uint64) { + t := p.v.Type().Elem() + switch t { + case int64Type: + if len(o.int64s) == 0 { + o.int64s = make([]int64, uint64PoolSize) + } + o.int64s[0] = int64(x) + p.v.Set(reflect.ValueOf(&o.int64s[0])) + o.int64s = o.int64s[1:] + return + case uint64Type: + if len(o.uint64s) == 0 { + o.uint64s = make([]uint64, uint64PoolSize) + } + o.uint64s[0] = x + p.v.Set(reflect.ValueOf(&o.uint64s[0])) + o.uint64s = o.uint64s[1:] + return + case float64Type: + if len(o.float64s) == 0 { + o.float64s = make([]float64, uint64PoolSize) + } + o.float64s[0] = math.Float64frombits(x) + p.v.Set(reflect.ValueOf(&o.float64s[0])) + o.float64s = o.float64s[1:] + return + } + panic("unreachable") +} + +func word64_IsNil(p word64) bool { + return p.v.IsNil() +} + +func word64_Get(p word64) uint64 { + elem := p.v.Elem() + switch elem.Kind() { + case reflect.Int64: + return uint64(elem.Int()) + case reflect.Uint64: + return elem.Uint() + case reflect.Float64: + return math.Float64bits(elem.Float()) + } + panic("unreachable") +} + +func structPointer_Word64(p structPointer, f field) word64 { + return word64{structPointer_field(p, f)} +} + +// word64Val is like word32Val but for 64-bit values. +type word64Val struct { + v reflect.Value +} + +func word64Val_Set(p word64Val, o *Buffer, x uint64) { + switch p.v.Type() { + case int64Type: + p.v.SetInt(int64(x)) + return + case uint64Type: + p.v.SetUint(x) + return + case float64Type: + p.v.SetFloat(math.Float64frombits(x)) + return + } + panic("unreachable") +} + +func word64Val_Get(p word64Val) uint64 { + elem := p.v + switch elem.Kind() { + case reflect.Int64: + return uint64(elem.Int()) + case reflect.Uint64: + return elem.Uint() + case reflect.Float64: + return math.Float64bits(elem.Float()) + } + panic("unreachable") +} + +func structPointer_Word64Val(p structPointer, f field) word64Val { + return word64Val{structPointer_field(p, f)} +} + +type word64Slice struct { + v reflect.Value +} + +func (p word64Slice) Append(x uint64) { + n, m := p.v.Len(), p.v.Cap() + if n < m { + p.v.SetLen(n + 1) + } else { + t := p.v.Type().Elem() + p.v.Set(reflect.Append(p.v, reflect.Zero(t))) + } + elem := p.v.Index(n) + switch elem.Kind() { + case reflect.Int64: + elem.SetInt(int64(int64(x))) + case reflect.Uint64: + elem.SetUint(uint64(x)) + case reflect.Float64: + elem.SetFloat(float64(math.Float64frombits(x))) + } +} + +func (p word64Slice) Len() int { + return p.v.Len() +} + +func (p word64Slice) Index(i int) uint64 { + elem := p.v.Index(i) + switch elem.Kind() { + case reflect.Int64: + return uint64(elem.Int()) + case reflect.Uint64: + return uint64(elem.Uint()) + case reflect.Float64: + return math.Float64bits(float64(elem.Float())) + } + panic("unreachable") +} + +func structPointer_Word64Slice(p structPointer, f field) word64Slice { + return word64Slice{structPointer_field(p, f)} +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/pointer_unsafe.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/pointer_unsafe.go new file mode 100644 index 0000000000000..e9be0fe92ee70 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/pointer_unsafe.go @@ -0,0 +1,266 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2012 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// +build !appengine + +// This file contains the implementation of the proto field accesses using package unsafe. + +package proto + +import ( + "reflect" + "unsafe" +) + +// NOTE: These type_Foo functions would more idiomatically be methods, +// but Go does not allow methods on pointer types, and we must preserve +// some pointer type for the garbage collector. We use these +// funcs with clunky names as our poor approximation to methods. +// +// An alternative would be +// type structPointer struct { p unsafe.Pointer } +// but that does not registerize as well. + +// A structPointer is a pointer to a struct. +type structPointer unsafe.Pointer + +// toStructPointer returns a structPointer equivalent to the given reflect value. +func toStructPointer(v reflect.Value) structPointer { + return structPointer(unsafe.Pointer(v.Pointer())) +} + +// IsNil reports whether p is nil. +func structPointer_IsNil(p structPointer) bool { + return p == nil +} + +// Interface returns the struct pointer, assumed to have element type t, +// as an interface value. +func structPointer_Interface(p structPointer, t reflect.Type) interface{} { + return reflect.NewAt(t, unsafe.Pointer(p)).Interface() +} + +// A field identifies a field in a struct, accessible from a structPointer. +// In this implementation, a field is identified by its byte offset from the start of the struct. +type field uintptr + +// toField returns a field equivalent to the given reflect field. +func toField(f *reflect.StructField) field { + return field(f.Offset) +} + +// invalidField is an invalid field identifier. +const invalidField = ^field(0) + +// IsValid reports whether the field identifier is valid. +func (f field) IsValid() bool { + return f != ^field(0) +} + +// Bytes returns the address of a []byte field in the struct. +func structPointer_Bytes(p structPointer, f field) *[]byte { + return (*[]byte)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// BytesSlice returns the address of a [][]byte field in the struct. +func structPointer_BytesSlice(p structPointer, f field) *[][]byte { + return (*[][]byte)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// Bool returns the address of a *bool field in the struct. +func structPointer_Bool(p structPointer, f field) **bool { + return (**bool)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// BoolVal returns the address of a bool field in the struct. +func structPointer_BoolVal(p structPointer, f field) *bool { + return (*bool)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// BoolSlice returns the address of a []bool field in the struct. +func structPointer_BoolSlice(p structPointer, f field) *[]bool { + return (*[]bool)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// String returns the address of a *string field in the struct. +func structPointer_String(p structPointer, f field) **string { + return (**string)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// StringVal returns the address of a string field in the struct. +func structPointer_StringVal(p structPointer, f field) *string { + return (*string)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// StringSlice returns the address of a []string field in the struct. +func structPointer_StringSlice(p structPointer, f field) *[]string { + return (*[]string)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// ExtMap returns the address of an extension map field in the struct. +func structPointer_ExtMap(p structPointer, f field) *map[int32]Extension { + return (*map[int32]Extension)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// NewAt returns the reflect.Value for a pointer to a field in the struct. +func structPointer_NewAt(p structPointer, f field, typ reflect.Type) reflect.Value { + return reflect.NewAt(typ, unsafe.Pointer(uintptr(p)+uintptr(f))) +} + +// SetStructPointer writes a *struct field in the struct. +func structPointer_SetStructPointer(p structPointer, f field, q structPointer) { + *(*structPointer)(unsafe.Pointer(uintptr(p) + uintptr(f))) = q +} + +// GetStructPointer reads a *struct field in the struct. +func structPointer_GetStructPointer(p structPointer, f field) structPointer { + return *(*structPointer)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// StructPointerSlice the address of a []*struct field in the struct. +func structPointer_StructPointerSlice(p structPointer, f field) *structPointerSlice { + return (*structPointerSlice)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// A structPointerSlice represents a slice of pointers to structs (themselves submessages or groups). +type structPointerSlice []structPointer + +func (v *structPointerSlice) Len() int { return len(*v) } +func (v *structPointerSlice) Index(i int) structPointer { return (*v)[i] } +func (v *structPointerSlice) Append(p structPointer) { *v = append(*v, p) } + +// A word32 is the address of a "pointer to 32-bit value" field. +type word32 **uint32 + +// IsNil reports whether *v is nil. +func word32_IsNil(p word32) bool { + return *p == nil +} + +// Set sets *v to point at a newly allocated word set to x. +func word32_Set(p word32, o *Buffer, x uint32) { + if len(o.uint32s) == 0 { + o.uint32s = make([]uint32, uint32PoolSize) + } + o.uint32s[0] = x + *p = &o.uint32s[0] + o.uint32s = o.uint32s[1:] +} + +// Get gets the value pointed at by *v. +func word32_Get(p word32) uint32 { + return **p +} + +// Word32 returns the address of a *int32, *uint32, *float32, or *enum field in the struct. +func structPointer_Word32(p structPointer, f field) word32 { + return word32((**uint32)(unsafe.Pointer(uintptr(p) + uintptr(f)))) +} + +// A word32Val is the address of a 32-bit value field. +type word32Val *uint32 + +// Set sets *p to x. +func word32Val_Set(p word32Val, x uint32) { + *p = x +} + +// Get gets the value pointed at by p. +func word32Val_Get(p word32Val) uint32 { + return *p +} + +// Word32Val returns the address of a *int32, *uint32, *float32, or *enum field in the struct. +func structPointer_Word32Val(p structPointer, f field) word32Val { + return word32Val((*uint32)(unsafe.Pointer(uintptr(p) + uintptr(f)))) +} + +// A word32Slice is a slice of 32-bit values. +type word32Slice []uint32 + +func (v *word32Slice) Append(x uint32) { *v = append(*v, x) } +func (v *word32Slice) Len() int { return len(*v) } +func (v *word32Slice) Index(i int) uint32 { return (*v)[i] } + +// Word32Slice returns the address of a []int32, []uint32, []float32, or []enum field in the struct. +func structPointer_Word32Slice(p structPointer, f field) *word32Slice { + return (*word32Slice)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// word64 is like word32 but for 64-bit values. +type word64 **uint64 + +func word64_Set(p word64, o *Buffer, x uint64) { + if len(o.uint64s) == 0 { + o.uint64s = make([]uint64, uint64PoolSize) + } + o.uint64s[0] = x + *p = &o.uint64s[0] + o.uint64s = o.uint64s[1:] +} + +func word64_IsNil(p word64) bool { + return *p == nil +} + +func word64_Get(p word64) uint64 { + return **p +} + +func structPointer_Word64(p structPointer, f field) word64 { + return word64((**uint64)(unsafe.Pointer(uintptr(p) + uintptr(f)))) +} + +// word64Val is like word32Val but for 64-bit values. +type word64Val *uint64 + +func word64Val_Set(p word64Val, o *Buffer, x uint64) { + *p = x +} + +func word64Val_Get(p word64Val) uint64 { + return *p +} + +func structPointer_Word64Val(p structPointer, f field) word64Val { + return word64Val((*uint64)(unsafe.Pointer(uintptr(p) + uintptr(f)))) +} + +// word64Slice is like word32Slice but for 64-bit values. +type word64Slice []uint64 + +func (v *word64Slice) Append(x uint64) { *v = append(*v, x) } +func (v *word64Slice) Len() int { return len(*v) } +func (v *word64Slice) Index(i int) uint64 { return (*v)[i] } + +func structPointer_Word64Slice(p structPointer, f field) *word64Slice { + return (*word64Slice)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/pointer_unsafe_gogo.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/pointer_unsafe_gogo.go new file mode 100644 index 0000000000000..6bc85fa9873f4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/pointer_unsafe_gogo.go @@ -0,0 +1,108 @@ +// Copyright (c) 2013, Vastech SA (PTY) LTD. All rights reserved. +// http://github.com/gogo/protobuf/gogoproto +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// +build !appengine + +// This file contains the implementation of the proto field accesses using package unsafe. + +package proto + +import ( + "reflect" + "unsafe" +) + +func structPointer_InterfaceAt(p structPointer, f field, t reflect.Type) interface{} { + point := unsafe.Pointer(uintptr(p) + uintptr(f)) + r := reflect.NewAt(t, point) + return r.Interface() +} + +func structPointer_InterfaceRef(p structPointer, f field, t reflect.Type) interface{} { + point := unsafe.Pointer(uintptr(p) + uintptr(f)) + r := reflect.NewAt(t, point) + if r.Elem().IsNil() { + return nil + } + return r.Elem().Interface() +} + +func copyUintPtr(oldptr, newptr uintptr, size int) { + oldbytes := make([]byte, 0) + oldslice := (*reflect.SliceHeader)(unsafe.Pointer(&oldbytes)) + oldslice.Data = oldptr + oldslice.Len = size + oldslice.Cap = size + newbytes := make([]byte, 0) + newslice := (*reflect.SliceHeader)(unsafe.Pointer(&newbytes)) + newslice.Data = newptr + newslice.Len = size + newslice.Cap = size + copy(newbytes, oldbytes) +} + +func structPointer_Copy(oldptr structPointer, newptr structPointer, size int) { + copyUintPtr(uintptr(oldptr), uintptr(newptr), size) +} + +func appendStructPointer(base structPointer, f field, typ reflect.Type) structPointer { + size := typ.Elem().Size() + oldHeader := structPointer_GetSliceHeader(base, f) + newLen := oldHeader.Len + 1 + slice := reflect.MakeSlice(typ, newLen, newLen) + bas := toStructPointer(slice) + for i := 0; i < oldHeader.Len; i++ { + newElemptr := uintptr(bas) + uintptr(i)*size + oldElemptr := oldHeader.Data + uintptr(i)*size + copyUintPtr(oldElemptr, newElemptr, int(size)) + } + + oldHeader.Data = uintptr(bas) + oldHeader.Len = newLen + oldHeader.Cap = newLen + + return structPointer(unsafe.Pointer(uintptr(unsafe.Pointer(bas)) + uintptr(uintptr(newLen-1)*size))) +} + +func structPointer_FieldPointer(p structPointer, f field) structPointer { + return structPointer(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +func structPointer_GetRefStructPointer(p structPointer, f field) structPointer { + return structPointer((*structPointer)(unsafe.Pointer(uintptr(p) + uintptr(f)))) +} + +func structPointer_GetSliceHeader(p structPointer, f field) *reflect.SliceHeader { + return (*reflect.SliceHeader)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +func structPointer_Add(p structPointer, size field) structPointer { + return structPointer(unsafe.Pointer(uintptr(p) + uintptr(size))) +} + +func structPointer_Len(p structPointer, f field) int { + return len(*(*[]interface{})(unsafe.Pointer(structPointer_GetRefStructPointer(p, f)))) +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/properties.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/properties.go new file mode 100644 index 0000000000000..13245c00df240 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/properties.go @@ -0,0 +1,815 @@ +// Extensions for Protocol Buffers to create more go like structures. +// +// Copyright (c) 2013, Vastech SA (PTY) LTD. All rights reserved. +// http://github.com/gogo/protobuf/gogoproto +// +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +/* + * Routines for encoding data into the wire format for protocol buffers. + */ + +import ( + "fmt" + "os" + "reflect" + "sort" + "strconv" + "strings" + "sync" +) + +const debug bool = false + +// Constants that identify the encoding of a value on the wire. +const ( + WireVarint = 0 + WireFixed64 = 1 + WireBytes = 2 + WireStartGroup = 3 + WireEndGroup = 4 + WireFixed32 = 5 +) + +const startSize = 10 // initial slice/string sizes + +// Encoders are defined in encode.go +// An encoder outputs the full representation of a field, including its +// tag and encoder type. +type encoder func(p *Buffer, prop *Properties, base structPointer) error + +// A valueEncoder encodes a single integer in a particular encoding. +type valueEncoder func(o *Buffer, x uint64) error + +// Sizers are defined in encode.go +// A sizer returns the encoded size of a field, including its tag and encoder +// type. +type sizer func(prop *Properties, base structPointer) int + +// A valueSizer returns the encoded size of a single integer in a particular +// encoding. +type valueSizer func(x uint64) int + +// Decoders are defined in decode.go +// A decoder creates a value from its wire representation. +// Unrecognized subelements are saved in unrec. +type decoder func(p *Buffer, prop *Properties, base structPointer) error + +// A valueDecoder decodes a single integer in a particular encoding. +type valueDecoder func(o *Buffer) (x uint64, err error) + +// tagMap is an optimization over map[int]int for typical protocol buffer +// use-cases. Encoded protocol buffers are often in tag order with small tag +// numbers. +type tagMap struct { + fastTags []int + slowTags map[int]int +} + +// tagMapFastLimit is the upper bound on the tag number that will be stored in +// the tagMap slice rather than its map. +const tagMapFastLimit = 1024 + +func (p *tagMap) get(t int) (int, bool) { + if t > 0 && t < tagMapFastLimit { + if t >= len(p.fastTags) { + return 0, false + } + fi := p.fastTags[t] + return fi, fi >= 0 + } + fi, ok := p.slowTags[t] + return fi, ok +} + +func (p *tagMap) put(t int, fi int) { + if t > 0 && t < tagMapFastLimit { + for len(p.fastTags) < t+1 { + p.fastTags = append(p.fastTags, -1) + } + p.fastTags[t] = fi + return + } + if p.slowTags == nil { + p.slowTags = make(map[int]int) + } + p.slowTags[t] = fi +} + +// StructProperties represents properties for all the fields of a struct. +// decoderTags and decoderOrigNames should only be used by the decoder. +type StructProperties struct { + Prop []*Properties // properties for each field + reqCount int // required count + decoderTags tagMap // map from proto tag to struct field number + decoderOrigNames map[string]int // map from original name to struct field number + order []int // list of struct field numbers in tag order + unrecField field // field id of the XXX_unrecognized []byte field + extendable bool // is this an extendable proto +} + +// Implement the sorting interface so we can sort the fields in tag order, as recommended by the spec. +// See encode.go, (*Buffer).enc_struct. + +func (sp *StructProperties) Len() int { return len(sp.order) } +func (sp *StructProperties) Less(i, j int) bool { + return sp.Prop[sp.order[i]].Tag < sp.Prop[sp.order[j]].Tag +} +func (sp *StructProperties) Swap(i, j int) { sp.order[i], sp.order[j] = sp.order[j], sp.order[i] } + +// Properties represents the protocol-specific behavior of a single struct field. +type Properties struct { + Name string // name of the field, for error messages + OrigName string // original name before protocol compiler (always set) + Wire string + WireType int + Tag int + Required bool + Optional bool + Repeated bool + Packed bool // relevant for repeated primitives only + Enum string // set for enum types only + proto3 bool // whether this is known to be a proto3 field; set for []byte only + + Default string // default value + HasDefault bool // whether an explicit default was provided + CustomType string + def_uint64 uint64 + + enc encoder + valEnc valueEncoder // set for bool and numeric types only + field field + tagcode []byte // encoding of EncodeVarint((Tag<<3)|WireType) + tagbuf [8]byte + stype reflect.Type // set for struct types only + sstype reflect.Type // set for slices of structs types only + ctype reflect.Type // set for custom types only + sprop *StructProperties // set for struct types only + isMarshaler bool + isUnmarshaler bool + + mtype reflect.Type // set for map types only + mkeyprop *Properties // set for map types only + mvalprop *Properties // set for map types only + + size sizer + valSize valueSizer // set for bool and numeric types only + + dec decoder + valDec valueDecoder // set for bool and numeric types only + + // If this is a packable field, this will be the decoder for the packed version of the field. + packedDec decoder +} + +// String formats the properties in the protobuf struct field tag style. +func (p *Properties) String() string { + s := p.Wire + s = "," + s += strconv.Itoa(p.Tag) + if p.Required { + s += ",req" + } + if p.Optional { + s += ",opt" + } + if p.Repeated { + s += ",rep" + } + if p.Packed { + s += ",packed" + } + if p.OrigName != p.Name { + s += ",name=" + p.OrigName + } + if p.proto3 { + s += ",proto3" + } + if len(p.Enum) > 0 { + s += ",enum=" + p.Enum + } + if p.HasDefault { + s += ",def=" + p.Default + } + return s +} + +// Parse populates p by parsing a string in the protobuf struct field tag style. +func (p *Properties) Parse(s string) { + // "bytes,49,opt,name=foo,def=hello!" + fields := strings.Split(s, ",") // breaks def=, but handled below. + if len(fields) < 2 { + fmt.Fprintf(os.Stderr, "proto: tag has too few fields: %q\n", s) + return + } + + p.Wire = fields[0] + switch p.Wire { + case "varint": + p.WireType = WireVarint + p.valEnc = (*Buffer).EncodeVarint + p.valDec = (*Buffer).DecodeVarint + p.valSize = sizeVarint + case "fixed32": + p.WireType = WireFixed32 + p.valEnc = (*Buffer).EncodeFixed32 + p.valDec = (*Buffer).DecodeFixed32 + p.valSize = sizeFixed32 + case "fixed64": + p.WireType = WireFixed64 + p.valEnc = (*Buffer).EncodeFixed64 + p.valDec = (*Buffer).DecodeFixed64 + p.valSize = sizeFixed64 + case "zigzag32": + p.WireType = WireVarint + p.valEnc = (*Buffer).EncodeZigzag32 + p.valDec = (*Buffer).DecodeZigzag32 + p.valSize = sizeZigzag32 + case "zigzag64": + p.WireType = WireVarint + p.valEnc = (*Buffer).EncodeZigzag64 + p.valDec = (*Buffer).DecodeZigzag64 + p.valSize = sizeZigzag64 + case "bytes", "group": + p.WireType = WireBytes + // no numeric converter for non-numeric types + default: + fmt.Fprintf(os.Stderr, "proto: tag has unknown wire type: %q\n", s) + return + } + + var err error + p.Tag, err = strconv.Atoi(fields[1]) + if err != nil { + return + } + + for i := 2; i < len(fields); i++ { + f := fields[i] + switch { + case f == "req": + p.Required = true + case f == "opt": + p.Optional = true + case f == "rep": + p.Repeated = true + case f == "packed": + p.Packed = true + case strings.HasPrefix(f, "name="): + p.OrigName = f[5:] + case strings.HasPrefix(f, "enum="): + p.Enum = f[5:] + case f == "proto3": + p.proto3 = true + case strings.HasPrefix(f, "def="): + p.HasDefault = true + p.Default = f[4:] // rest of string + if i+1 < len(fields) { + // Commas aren't escaped, and def is always last. + p.Default += "," + strings.Join(fields[i+1:], ",") + break + } + case strings.HasPrefix(f, "embedded="): + p.OrigName = strings.Split(f, "=")[1] + case strings.HasPrefix(f, "customtype="): + p.CustomType = strings.Split(f, "=")[1] + } + } +} + +func logNoSliceEnc(t1, t2 reflect.Type) { + fmt.Fprintf(os.Stderr, "proto: no slice oenc for %T = []%T\n", t1, t2) +} + +var protoMessageType = reflect.TypeOf((*Message)(nil)).Elem() + +// Initialize the fields for encoding and decoding. +func (p *Properties) setEncAndDec(typ reflect.Type, f *reflect.StructField, lockGetProp bool) { + p.enc = nil + p.dec = nil + p.size = nil + if len(p.CustomType) > 0 { + p.setCustomEncAndDec(typ) + p.setTag(lockGetProp) + return + } + switch t1 := typ; t1.Kind() { + default: + fmt.Fprintf(os.Stderr, "proto: no coders for %v\n", t1) + + // proto3 scalar types + + case reflect.Bool: + if p.proto3 { + p.enc = (*Buffer).enc_proto3_bool + p.dec = (*Buffer).dec_proto3_bool + p.size = size_proto3_bool + } else { + p.enc = (*Buffer).enc_ref_bool + p.dec = (*Buffer).dec_proto3_bool + p.size = size_ref_bool + } + case reflect.Int32: + if p.proto3 { + p.enc = (*Buffer).enc_proto3_int32 + p.dec = (*Buffer).dec_proto3_int32 + p.size = size_proto3_int32 + } else { + p.enc = (*Buffer).enc_ref_int32 + p.dec = (*Buffer).dec_proto3_int32 + p.size = size_ref_int32 + } + case reflect.Uint32: + if p.proto3 { + p.enc = (*Buffer).enc_proto3_uint32 + p.dec = (*Buffer).dec_proto3_int32 // can reuse + p.size = size_proto3_uint32 + } else { + p.enc = (*Buffer).enc_ref_uint32 + p.dec = (*Buffer).dec_proto3_int32 // can reuse + p.size = size_ref_uint32 + } + case reflect.Int64, reflect.Uint64: + if p.proto3 { + p.enc = (*Buffer).enc_proto3_int64 + p.dec = (*Buffer).dec_proto3_int64 + p.size = size_proto3_int64 + } else { + p.enc = (*Buffer).enc_ref_int64 + p.dec = (*Buffer).dec_proto3_int64 + p.size = size_ref_int64 + } + case reflect.Float32: + if p.proto3 { + p.enc = (*Buffer).enc_proto3_uint32 // can just treat them as bits + p.dec = (*Buffer).dec_proto3_int32 + p.size = size_proto3_uint32 + } else { + p.enc = (*Buffer).enc_ref_uint32 // can just treat them as bits + p.dec = (*Buffer).dec_proto3_int32 + p.size = size_ref_uint32 + } + case reflect.Float64: + if p.proto3 { + p.enc = (*Buffer).enc_proto3_int64 // can just treat them as bits + p.dec = (*Buffer).dec_proto3_int64 + p.size = size_proto3_int64 + } else { + p.enc = (*Buffer).enc_ref_int64 // can just treat them as bits + p.dec = (*Buffer).dec_proto3_int64 + p.size = size_ref_int64 + } + case reflect.String: + if p.proto3 { + p.enc = (*Buffer).enc_proto3_string + p.dec = (*Buffer).dec_proto3_string + p.size = size_proto3_string + } else { + p.enc = (*Buffer).enc_ref_string + p.dec = (*Buffer).dec_proto3_string + p.size = size_ref_string + } + case reflect.Struct: + p.stype = typ + p.isMarshaler = isMarshaler(typ) + p.isUnmarshaler = isUnmarshaler(typ) + if p.Wire == "bytes" { + p.enc = (*Buffer).enc_ref_struct_message + p.dec = (*Buffer).dec_ref_struct_message + p.size = size_ref_struct_message + } else { + fmt.Fprintf(os.Stderr, "proto: no coders for struct %T\n", typ) + } + + case reflect.Ptr: + switch t2 := t1.Elem(); t2.Kind() { + default: + fmt.Fprintf(os.Stderr, "proto: no encoder function for %v -> %v\n", t1, t2) + break + case reflect.Bool: + p.enc = (*Buffer).enc_bool + p.dec = (*Buffer).dec_bool + p.size = size_bool + case reflect.Int32: + p.enc = (*Buffer).enc_int32 + p.dec = (*Buffer).dec_int32 + p.size = size_int32 + case reflect.Uint32: + p.enc = (*Buffer).enc_uint32 + p.dec = (*Buffer).dec_int32 // can reuse + p.size = size_uint32 + case reflect.Int64, reflect.Uint64: + p.enc = (*Buffer).enc_int64 + p.dec = (*Buffer).dec_int64 + p.size = size_int64 + case reflect.Float32: + p.enc = (*Buffer).enc_uint32 // can just treat them as bits + p.dec = (*Buffer).dec_int32 + p.size = size_uint32 + case reflect.Float64: + p.enc = (*Buffer).enc_int64 // can just treat them as bits + p.dec = (*Buffer).dec_int64 + p.size = size_int64 + case reflect.String: + p.enc = (*Buffer).enc_string + p.dec = (*Buffer).dec_string + p.size = size_string + case reflect.Struct: + p.stype = t1.Elem() + p.isMarshaler = isMarshaler(t1) + p.isUnmarshaler = isUnmarshaler(t1) + if p.Wire == "bytes" { + p.enc = (*Buffer).enc_struct_message + p.dec = (*Buffer).dec_struct_message + p.size = size_struct_message + } else { + p.enc = (*Buffer).enc_struct_group + p.dec = (*Buffer).dec_struct_group + p.size = size_struct_group + } + } + + case reflect.Slice: + switch t2 := t1.Elem(); t2.Kind() { + default: + logNoSliceEnc(t1, t2) + break + case reflect.Bool: + if p.Packed { + p.enc = (*Buffer).enc_slice_packed_bool + p.size = size_slice_packed_bool + } else { + p.enc = (*Buffer).enc_slice_bool + p.size = size_slice_bool + } + p.dec = (*Buffer).dec_slice_bool + p.packedDec = (*Buffer).dec_slice_packed_bool + case reflect.Int32: + if p.Packed { + p.enc = (*Buffer).enc_slice_packed_int32 + p.size = size_slice_packed_int32 + } else { + p.enc = (*Buffer).enc_slice_int32 + p.size = size_slice_int32 + } + p.dec = (*Buffer).dec_slice_int32 + p.packedDec = (*Buffer).dec_slice_packed_int32 + case reflect.Uint32: + if p.Packed { + p.enc = (*Buffer).enc_slice_packed_uint32 + p.size = size_slice_packed_uint32 + } else { + p.enc = (*Buffer).enc_slice_uint32 + p.size = size_slice_uint32 + } + p.dec = (*Buffer).dec_slice_int32 + p.packedDec = (*Buffer).dec_slice_packed_int32 + case reflect.Int64, reflect.Uint64: + if p.Packed { + p.enc = (*Buffer).enc_slice_packed_int64 + p.size = size_slice_packed_int64 + } else { + p.enc = (*Buffer).enc_slice_int64 + p.size = size_slice_int64 + } + p.dec = (*Buffer).dec_slice_int64 + p.packedDec = (*Buffer).dec_slice_packed_int64 + case reflect.Uint8: + p.enc = (*Buffer).enc_slice_byte + p.dec = (*Buffer).dec_slice_byte + p.size = size_slice_byte + // This is a []byte, which is either a bytes field, + // or the value of a map field. In the latter case, + // we always encode an empty []byte, so we should not + // use the proto3 enc/size funcs. + // f == nil iff this is the key/value of a map field. + if p.proto3 && f != nil { + p.enc = (*Buffer).enc_proto3_slice_byte + p.size = size_proto3_slice_byte + } + case reflect.Float32, reflect.Float64: + switch t2.Bits() { + case 32: + // can just treat them as bits + if p.Packed { + p.enc = (*Buffer).enc_slice_packed_uint32 + p.size = size_slice_packed_uint32 + } else { + p.enc = (*Buffer).enc_slice_uint32 + p.size = size_slice_uint32 + } + p.dec = (*Buffer).dec_slice_int32 + p.packedDec = (*Buffer).dec_slice_packed_int32 + case 64: + // can just treat them as bits + if p.Packed { + p.enc = (*Buffer).enc_slice_packed_int64 + p.size = size_slice_packed_int64 + } else { + p.enc = (*Buffer).enc_slice_int64 + p.size = size_slice_int64 + } + p.dec = (*Buffer).dec_slice_int64 + p.packedDec = (*Buffer).dec_slice_packed_int64 + default: + logNoSliceEnc(t1, t2) + break + } + case reflect.String: + p.enc = (*Buffer).enc_slice_string + p.dec = (*Buffer).dec_slice_string + p.size = size_slice_string + case reflect.Ptr: + switch t3 := t2.Elem(); t3.Kind() { + default: + fmt.Fprintf(os.Stderr, "proto: no ptr oenc for %T -> %T -> %T\n", t1, t2, t3) + break + case reflect.Struct: + p.stype = t2.Elem() + p.isMarshaler = isMarshaler(t2) + p.isUnmarshaler = isUnmarshaler(t2) + if p.Wire == "bytes" { + p.enc = (*Buffer).enc_slice_struct_message + p.dec = (*Buffer).dec_slice_struct_message + p.size = size_slice_struct_message + } else { + p.enc = (*Buffer).enc_slice_struct_group + p.dec = (*Buffer).dec_slice_struct_group + p.size = size_slice_struct_group + } + } + case reflect.Slice: + switch t2.Elem().Kind() { + default: + fmt.Fprintf(os.Stderr, "proto: no slice elem oenc for %T -> %T -> %T\n", t1, t2, t2.Elem()) + break + case reflect.Uint8: + p.enc = (*Buffer).enc_slice_slice_byte + p.dec = (*Buffer).dec_slice_slice_byte + p.size = size_slice_slice_byte + } + case reflect.Struct: + p.setSliceOfNonPointerStructs(t1) + } + + case reflect.Map: + p.enc = (*Buffer).enc_new_map + p.dec = (*Buffer).dec_new_map + p.size = size_new_map + + p.mtype = t1 + p.mkeyprop = &Properties{} + p.mkeyprop.init(reflect.PtrTo(p.mtype.Key()), "Key", f.Tag.Get("protobuf_key"), nil, lockGetProp) + p.mvalprop = &Properties{} + vtype := p.mtype.Elem() + if vtype.Kind() != reflect.Ptr && vtype.Kind() != reflect.Slice { + // The value type is not a message (*T) or bytes ([]byte), + // so we need encoders for the pointer to this type. + vtype = reflect.PtrTo(vtype) + } + p.mvalprop.init(vtype, "Value", f.Tag.Get("protobuf_val"), nil, lockGetProp) + } + p.setTag(lockGetProp) +} + +func (p *Properties) setTag(lockGetProp bool) { + // precalculate tag code + wire := p.WireType + if p.Packed { + wire = WireBytes + } + x := uint32(p.Tag)<<3 | uint32(wire) + i := 0 + for i = 0; x > 127; i++ { + p.tagbuf[i] = 0x80 | uint8(x&0x7F) + x >>= 7 + } + p.tagbuf[i] = uint8(x) + p.tagcode = p.tagbuf[0 : i+1] + + if p.stype != nil { + if lockGetProp { + p.sprop = GetProperties(p.stype) + } else { + p.sprop = getPropertiesLocked(p.stype) + } + } +} + +var ( + marshalerType = reflect.TypeOf((*Marshaler)(nil)).Elem() + unmarshalerType = reflect.TypeOf((*Unmarshaler)(nil)).Elem() +) + +// isMarshaler reports whether type t implements Marshaler. +func isMarshaler(t reflect.Type) bool { + return t.Implements(marshalerType) +} + +// isUnmarshaler reports whether type t implements Unmarshaler. +func isUnmarshaler(t reflect.Type) bool { + return t.Implements(unmarshalerType) +} + +// Init populates the properties from a protocol buffer struct tag. +func (p *Properties) Init(typ reflect.Type, name, tag string, f *reflect.StructField) { + p.init(typ, name, tag, f, true) +} + +func (p *Properties) init(typ reflect.Type, name, tag string, f *reflect.StructField, lockGetProp bool) { + // "bytes,49,opt,def=hello!" + p.Name = name + p.OrigName = name + if f != nil { + p.field = toField(f) + } + if tag == "" { + return + } + p.Parse(tag) + p.setEncAndDec(typ, f, lockGetProp) +} + +var ( + propertiesMu sync.RWMutex + propertiesMap = make(map[reflect.Type]*StructProperties) +) + +// GetProperties returns the list of properties for the type represented by t. +// t must represent a generated struct type of a protocol message. +func GetProperties(t reflect.Type) *StructProperties { + if t.Kind() != reflect.Struct { + panic("proto: type must have kind struct") + } + + // Most calls to GetProperties in a long-running program will be + // retrieving details for types we have seen before. + propertiesMu.RLock() + sprop, ok := propertiesMap[t] + propertiesMu.RUnlock() + if ok { + if collectStats { + stats.Chit++ + } + return sprop + } + + propertiesMu.Lock() + sprop = getPropertiesLocked(t) + propertiesMu.Unlock() + return sprop +} + +// getPropertiesLocked requires that propertiesMu is held. +func getPropertiesLocked(t reflect.Type) *StructProperties { + if prop, ok := propertiesMap[t]; ok { + if collectStats { + stats.Chit++ + } + return prop + } + if collectStats { + stats.Cmiss++ + } + + prop := new(StructProperties) + // in case of recursive protos, fill this in now. + propertiesMap[t] = prop + + // build properties + prop.extendable = reflect.PtrTo(t).Implements(extendableProtoType) + prop.unrecField = invalidField + prop.Prop = make([]*Properties, t.NumField()) + prop.order = make([]int, t.NumField()) + + for i := 0; i < t.NumField(); i++ { + f := t.Field(i) + p := new(Properties) + name := f.Name + p.init(f.Type, name, f.Tag.Get("protobuf"), &f, false) + + if f.Name == "XXX_extensions" { // special case + if len(f.Tag.Get("protobuf")) > 0 { + p.enc = (*Buffer).enc_ext_slice_byte + p.dec = nil // not needed + p.size = size_ext_slice_byte + } else { + p.enc = (*Buffer).enc_map + p.dec = nil // not needed + p.size = size_map + } + } + if f.Name == "XXX_unrecognized" { // special case + prop.unrecField = toField(&f) + } + prop.Prop[i] = p + prop.order[i] = i + if debug { + print(i, " ", f.Name, " ", t.String(), " ") + if p.Tag > 0 { + print(p.String()) + } + print("\n") + } + if p.enc == nil && !strings.HasPrefix(f.Name, "XXX_") { + fmt.Fprintln(os.Stderr, "proto: no encoder for", f.Name, f.Type.String(), "[GetProperties]") + } + } + + // Re-order prop.order. + sort.Sort(prop) + + // build required counts + // build tags + reqCount := 0 + prop.decoderOrigNames = make(map[string]int) + for i, p := range prop.Prop { + if strings.HasPrefix(p.Name, "XXX_") { + // Internal fields should not appear in tags/origNames maps. + // They are handled specially when encoding and decoding. + continue + } + if p.Required { + reqCount++ + } + prop.decoderTags.put(p.Tag, i) + prop.decoderOrigNames[p.OrigName] = i + } + prop.reqCount = reqCount + + return prop +} + +// Return the Properties object for the x[0]'th field of the structure. +func propByIndex(t reflect.Type, x []int) *Properties { + if len(x) != 1 { + fmt.Fprintf(os.Stderr, "proto: field index dimension %d (not 1) for type %s\n", len(x), t) + return nil + } + prop := GetProperties(t) + return prop.Prop[x[0]] +} + +// Get the address and type of a pointer to a struct from an interface. +func getbase(pb Message) (t reflect.Type, b structPointer, err error) { + if pb == nil { + err = ErrNil + return + } + // get the reflect type of the pointer to the struct. + t = reflect.TypeOf(pb) + // get the address of the struct. + value := reflect.ValueOf(pb) + b = toStructPointer(value) + return +} + +// A global registry of enum types. +// The generated code will register the generated maps by calling RegisterEnum. + +var enumValueMaps = make(map[string]map[string]int32) +var enumStringMaps = make(map[string]map[int32]string) + +// RegisterEnum is called from the generated code to install the enum descriptor +// maps into the global table to aid parsing text format protocol buffers. +func RegisterEnum(typeName string, unusedNameMap map[int32]string, valueMap map[string]int32) { + if _, ok := enumValueMaps[typeName]; ok { + panic("proto: duplicate enum registered: " + typeName) + } + enumValueMaps[typeName] = valueMap + if _, ok := enumStringMaps[typeName]; ok { + panic("proto: duplicate enum registered: " + typeName) + } + enumStringMaps[typeName] = unusedNameMap +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/properties_gogo.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/properties_gogo.go new file mode 100644 index 0000000000000..8daf9f7768c25 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/properties_gogo.go @@ -0,0 +1,64 @@ +// Copyright (c) 2013, Vastech SA (PTY) LTD. All rights reserved. +// http://github.com/gogo/protobuf/gogoproto +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +import ( + "fmt" + "os" + "reflect" +) + +func (p *Properties) setCustomEncAndDec(typ reflect.Type) { + p.ctype = typ + if p.Repeated { + p.enc = (*Buffer).enc_custom_slice_bytes + p.dec = (*Buffer).dec_custom_slice_bytes + p.size = size_custom_slice_bytes + } else if typ.Kind() == reflect.Ptr { + p.enc = (*Buffer).enc_custom_bytes + p.dec = (*Buffer).dec_custom_bytes + p.size = size_custom_bytes + } else { + p.enc = (*Buffer).enc_custom_ref_bytes + p.dec = (*Buffer).dec_custom_ref_bytes + p.size = size_custom_ref_bytes + } +} + +func (p *Properties) setSliceOfNonPointerStructs(typ reflect.Type) { + t2 := typ.Elem() + p.sstype = typ + p.stype = t2 + p.isMarshaler = isMarshaler(t2) + p.isUnmarshaler = isUnmarshaler(t2) + p.enc = (*Buffer).enc_slice_ref_struct_message + p.dec = (*Buffer).dec_slice_ref_struct_message + p.size = size_slice_ref_struct_message + if p.Wire != "bytes" { + fmt.Fprintf(os.Stderr, "proto: no ptr oenc for %T -> %T \n", typ, t2) + } +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/proto3_proto/proto3.pb.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/proto3_proto/proto3.pb.go new file mode 100644 index 0000000000000..2f2da4604485d --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/proto3_proto/proto3.pb.go @@ -0,0 +1,122 @@ +// Code generated by protoc-gen-gogo. +// source: proto3_proto/proto3.proto +// DO NOT EDIT! + +/* +Package proto3_proto is a generated protocol buffer package. + +It is generated from these files: + proto3_proto/proto3.proto + +It has these top-level messages: + Message + Nested + MessageWithMap +*/ +package proto3_proto + +import proto "github.com/gogo/protobuf/proto" +import testdata "github.com/gogo/protobuf/proto/testdata" + +// Reference imports to suppress errors if they are not otherwise used. +var _ = proto.Marshal + +type Message_Humour int32 + +const ( + Message_UNKNOWN Message_Humour = 0 + Message_PUNS Message_Humour = 1 + Message_SLAPSTICK Message_Humour = 2 + Message_BILL_BAILEY Message_Humour = 3 +) + +var Message_Humour_name = map[int32]string{ + 0: "UNKNOWN", + 1: "PUNS", + 2: "SLAPSTICK", + 3: "BILL_BAILEY", +} +var Message_Humour_value = map[string]int32{ + "UNKNOWN": 0, + "PUNS": 1, + "SLAPSTICK": 2, + "BILL_BAILEY": 3, +} + +func (x Message_Humour) String() string { + return proto.EnumName(Message_Humour_name, int32(x)) +} + +type Message struct { + Name string `protobuf:"bytes,1,opt,name=name,proto3" json:"name,omitempty"` + Hilarity Message_Humour `protobuf:"varint,2,opt,name=hilarity,proto3,enum=proto3_proto.Message_Humour" json:"hilarity,omitempty"` + HeightInCm uint32 `protobuf:"varint,3,opt,name=height_in_cm,proto3" json:"height_in_cm,omitempty"` + Data []byte `protobuf:"bytes,4,opt,name=data,proto3" json:"data,omitempty"` + ResultCount int64 `protobuf:"varint,7,opt,name=result_count,proto3" json:"result_count,omitempty"` + TrueScotsman bool `protobuf:"varint,8,opt,name=true_scotsman,proto3" json:"true_scotsman,omitempty"` + Score float32 `protobuf:"fixed32,9,opt,name=score,proto3" json:"score,omitempty"` + Key []uint64 `protobuf:"varint,5,rep,name=key" json:"key,omitempty"` + Nested *Nested `protobuf:"bytes,6,opt,name=nested" json:"nested,omitempty"` + Terrain map[string]*Nested `protobuf:"bytes,10,rep,name=terrain" json:"terrain,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value"` + Proto2Field *testdata.SubDefaults `protobuf:"bytes,11,opt,name=proto2_field" json:"proto2_field,omitempty"` + Proto2Value map[string]*testdata.SubDefaults `protobuf:"bytes,13,rep,name=proto2_value" json:"proto2_value,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value"` +} + +func (m *Message) Reset() { *m = Message{} } +func (m *Message) String() string { return proto.CompactTextString(m) } +func (*Message) ProtoMessage() {} + +func (m *Message) GetNested() *Nested { + if m != nil { + return m.Nested + } + return nil +} + +func (m *Message) GetTerrain() map[string]*Nested { + if m != nil { + return m.Terrain + } + return nil +} + +func (m *Message) GetProto2Field() *testdata.SubDefaults { + if m != nil { + return m.Proto2Field + } + return nil +} + +func (m *Message) GetProto2Value() map[string]*testdata.SubDefaults { + if m != nil { + return m.Proto2Value + } + return nil +} + +type Nested struct { + Bunny string `protobuf:"bytes,1,opt,name=bunny,proto3" json:"bunny,omitempty"` +} + +func (m *Nested) Reset() { *m = Nested{} } +func (m *Nested) String() string { return proto.CompactTextString(m) } +func (*Nested) ProtoMessage() {} + +type MessageWithMap struct { + ByteMapping map[bool][]byte `protobuf:"bytes,1,rep,name=byte_mapping" json:"byte_mapping,omitempty" protobuf_key:"varint,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` +} + +func (m *MessageWithMap) Reset() { *m = MessageWithMap{} } +func (m *MessageWithMap) String() string { return proto.CompactTextString(m) } +func (*MessageWithMap) ProtoMessage() {} + +func (m *MessageWithMap) GetByteMapping() map[bool][]byte { + if m != nil { + return m.ByteMapping + } + return nil +} + +func init() { + proto.RegisterEnum("proto3_proto.Message_Humour", Message_Humour_name, Message_Humour_value) +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/proto3_proto/proto3.proto b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/proto3_proto/proto3.proto new file mode 100644 index 0000000000000..ca670015a2636 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/proto3_proto/proto3.proto @@ -0,0 +1,68 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2014 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +syntax = "proto3"; + +package proto3_proto; + +import "github.com/gogo/protobuf/proto/testdata/test.proto"; + +message Message { + enum Humour { + UNKNOWN = 0; + PUNS = 1; + SLAPSTICK = 2; + BILL_BAILEY = 3; + } + + string name = 1; + Humour hilarity = 2; + uint32 height_in_cm = 3; + bytes data = 4; + int64 result_count = 7; + bool true_scotsman = 8; + float score = 9; + + repeated uint64 key = 5; + Nested nested = 6; + + map terrain = 10; + testdata.SubDefaults proto2_field = 11; + map proto2_value = 13; +} + +message Nested { + string bunny = 1; +} + +message MessageWithMap { + map byte_mapping = 1; +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/proto3_test.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/proto3_test.go new file mode 100644 index 0000000000000..6f9cddc3fa9a8 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/proto3_test.go @@ -0,0 +1,125 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2014 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "testing" + + "github.com/gogo/protobuf/proto" + pb "github.com/gogo/protobuf/proto/proto3_proto" + tpb "github.com/gogo/protobuf/proto/testdata" +) + +func TestProto3ZeroValues(t *testing.T) { + tests := []struct { + desc string + m proto.Message + }{ + {"zero message", &pb.Message{}}, + {"empty bytes field", &pb.Message{Data: []byte{}}}, + } + for _, test := range tests { + b, err := proto.Marshal(test.m) + if err != nil { + t.Errorf("%s: proto.Marshal: %v", test.desc, err) + continue + } + if len(b) > 0 { + t.Errorf("%s: Encoding is non-empty: %q", test.desc, b) + } + } +} + +func TestRoundTripProto3(t *testing.T) { + m := &pb.Message{ + Name: "David", // (2 | 1<<3): 0x0a 0x05 "David" + Hilarity: pb.Message_PUNS, // (0 | 2<<3): 0x10 0x01 + HeightInCm: 178, // (0 | 3<<3): 0x18 0xb2 0x01 + Data: []byte("roboto"), // (2 | 4<<3): 0x20 0x06 "roboto" + ResultCount: 47, // (0 | 7<<3): 0x38 0x2f + TrueScotsman: true, // (0 | 8<<3): 0x40 0x01 + Score: 8.1, // (5 | 9<<3): 0x4d <8.1> + + Key: []uint64{1, 0xdeadbeef}, + Nested: &pb.Nested{ + Bunny: "Monty", + }, + } + t.Logf(" m: %v", m) + + b, err := proto.Marshal(m) + if err != nil { + t.Fatalf("proto.Marshal: %v", err) + } + t.Logf(" b: %q", b) + + m2 := new(pb.Message) + if err := proto.Unmarshal(b, m2); err != nil { + t.Fatalf("proto.Unmarshal: %v", err) + } + t.Logf("m2: %v", m2) + + if !proto.Equal(m, m2) { + t.Errorf("proto.Equal returned false:\n m: %v\nm2: %v", m, m2) + } +} + +func TestProto3SetDefaults(t *testing.T) { + in := &pb.Message{ + Terrain: map[string]*pb.Nested{ + "meadow": new(pb.Nested), + }, + Proto2Field: new(tpb.SubDefaults), + Proto2Value: map[string]*tpb.SubDefaults{ + "badlands": new(tpb.SubDefaults), + }, + } + + got := proto.Clone(in).(*pb.Message) + proto.SetDefaults(got) + + // There are no defaults in proto3. Everything should be the zero value, but + // we need to remember to set defaults for nested proto2 messages. + want := &pb.Message{ + Terrain: map[string]*pb.Nested{ + "meadow": new(pb.Nested), + }, + Proto2Field: &tpb.SubDefaults{N: proto.Int64(7)}, + Proto2Value: map[string]*tpb.SubDefaults{ + "badlands": {N: proto.Int64(7)}, + }, + } + + if !proto.Equal(got, want) { + t.Errorf("with in = %v\nproto.SetDefaults(in) =>\ngot %v\nwant %v", in, got, want) + } +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/size2_test.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/size2_test.go new file mode 100644 index 0000000000000..a2729c39a1b83 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/size2_test.go @@ -0,0 +1,63 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2012 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +import ( + "testing" +) + +// This is a separate file and package from size_test.go because that one uses +// generated messages and thus may not be in package proto without having a circular +// dependency, whereas this file tests unexported details of size.go. + +func TestVarintSize(t *testing.T) { + // Check the edge cases carefully. + testCases := []struct { + n uint64 + size int + }{ + {0, 1}, + {1, 1}, + {127, 1}, + {128, 2}, + {16383, 2}, + {16384, 3}, + {1<<63 - 1, 9}, + {1 << 63, 10}, + } + for _, tc := range testCases { + size := sizeVarint(tc.n) + if size != tc.size { + t.Errorf("sizeVarint(%d) = %d, want %d", tc.n, size, tc.size) + } + } +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/size_test.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/size_test.go new file mode 100644 index 0000000000000..457a479ebb41b --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/size_test.go @@ -0,0 +1,142 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2012 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "log" + "strings" + "testing" + + . "github.com/gogo/protobuf/proto" + proto3pb "github.com/gogo/protobuf/proto/proto3_proto" + pb "github.com/gogo/protobuf/proto/testdata" +) + +var messageWithExtension1 = &pb.MyMessage{Count: Int32(7)} + +// messageWithExtension2 is in equal_test.go. +var messageWithExtension3 = &pb.MyMessage{Count: Int32(8)} + +func init() { + if err := SetExtension(messageWithExtension1, pb.E_Ext_More, &pb.Ext{Data: String("Abbott")}); err != nil { + log.Panicf("SetExtension: %v", err) + } + if err := SetExtension(messageWithExtension3, pb.E_Ext_More, &pb.Ext{Data: String("Costello")}); err != nil { + log.Panicf("SetExtension: %v", err) + } + + // Force messageWithExtension3 to have the extension encoded. + Marshal(messageWithExtension3) + +} + +var SizeTests = []struct { + desc string + pb Message +}{ + {"empty", &pb.OtherMessage{}}, + // Basic types. + {"bool", &pb.Defaults{F_Bool: Bool(true)}}, + {"int32", &pb.Defaults{F_Int32: Int32(12)}}, + {"negative int32", &pb.Defaults{F_Int32: Int32(-1)}}, + {"small int64", &pb.Defaults{F_Int64: Int64(1)}}, + {"big int64", &pb.Defaults{F_Int64: Int64(1 << 20)}}, + {"negative int64", &pb.Defaults{F_Int64: Int64(-1)}}, + {"fixed32", &pb.Defaults{F_Fixed32: Uint32(71)}}, + {"fixed64", &pb.Defaults{F_Fixed64: Uint64(72)}}, + {"uint32", &pb.Defaults{F_Uint32: Uint32(123)}}, + {"uint64", &pb.Defaults{F_Uint64: Uint64(124)}}, + {"float", &pb.Defaults{F_Float: Float32(12.6)}}, + {"double", &pb.Defaults{F_Double: Float64(13.9)}}, + {"string", &pb.Defaults{F_String: String("niles")}}, + {"bytes", &pb.Defaults{F_Bytes: []byte("wowsa")}}, + {"bytes, empty", &pb.Defaults{F_Bytes: []byte{}}}, + {"sint32", &pb.Defaults{F_Sint32: Int32(65)}}, + {"sint64", &pb.Defaults{F_Sint64: Int64(67)}}, + {"enum", &pb.Defaults{F_Enum: pb.Defaults_BLUE.Enum()}}, + // Repeated. + {"empty repeated bool", &pb.MoreRepeated{Bools: []bool{}}}, + {"repeated bool", &pb.MoreRepeated{Bools: []bool{false, true, true, false}}}, + {"packed repeated bool", &pb.MoreRepeated{BoolsPacked: []bool{false, true, true, false, true, true, true}}}, + {"repeated int32", &pb.MoreRepeated{Ints: []int32{1, 12203, 1729, -1}}}, + {"repeated int32 packed", &pb.MoreRepeated{IntsPacked: []int32{1, 12203, 1729}}}, + {"repeated int64 packed", &pb.MoreRepeated{Int64SPacked: []int64{ + // Need enough large numbers to verify that the header is counting the number of bytes + // for the field, not the number of elements. + 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, + 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, + }}}, + {"repeated string", &pb.MoreRepeated{Strings: []string{"r", "ken", "gri"}}}, + {"repeated fixed", &pb.MoreRepeated{Fixeds: []uint32{1, 2, 3, 4}}}, + // Nested. + {"nested", &pb.OldMessage{Nested: &pb.OldMessage_Nested{Name: String("whatever")}}}, + {"group", &pb.GroupOld{G: &pb.GroupOld_G{X: Int32(12345)}}}, + // Other things. + {"unrecognized", &pb.MoreRepeated{XXX_unrecognized: []byte{13<<3 | 0, 4}}}, + {"extension (unencoded)", messageWithExtension1}, + {"extension (encoded)", messageWithExtension3}, + // proto3 message + {"proto3 empty", &proto3pb.Message{}}, + {"proto3 bool", &proto3pb.Message{TrueScotsman: true}}, + {"proto3 int64", &proto3pb.Message{ResultCount: 1}}, + {"proto3 uint32", &proto3pb.Message{HeightInCm: 123}}, + {"proto3 float", &proto3pb.Message{Score: 12.6}}, + {"proto3 string", &proto3pb.Message{Name: "Snezana"}}, + {"proto3 bytes", &proto3pb.Message{Data: []byte("wowsa")}}, + {"proto3 bytes, empty", &proto3pb.Message{Data: []byte{}}}, + {"proto3 enum", &proto3pb.Message{Hilarity: proto3pb.Message_PUNS}}, + {"proto3 map field with empty bytes", &proto3pb.MessageWithMap{ByteMapping: map[bool][]byte{false: {}}}}, + + {"map field", &pb.MessageWithMap{NameMapping: map[int32]string{1: "Rob", 7: "Andrew"}}}, + {"map field with message", &pb.MessageWithMap{MsgMapping: map[int64]*pb.FloatingPoint{0x7001: {F: Float64(2.0)}}}}, + {"map field with bytes", &pb.MessageWithMap{ByteMapping: map[bool][]byte{true: []byte("this time for sure")}}}, + {"map field with empty bytes", &pb.MessageWithMap{ByteMapping: map[bool][]byte{true: {}}}}, + + {"map field with big entry", &pb.MessageWithMap{NameMapping: map[int32]string{8: strings.Repeat("x", 125)}}}, + {"map field with big key and val", &pb.MessageWithMap{StrToStr: map[string]string{strings.Repeat("x", 70): strings.Repeat("y", 70)}}}, + {"map field with big numeric key", &pb.MessageWithMap{NameMapping: map[int32]string{0xf00d: "om nom nom"}}}, +} + +func TestSize(t *testing.T) { + for _, tc := range SizeTests { + size := Size(tc.pb) + b, err := Marshal(tc.pb) + if err != nil { + t.Errorf("%v: Marshal failed: %v", tc.desc, err) + continue + } + if size != len(b) { + t.Errorf("%v: Size(%v) = %d, want %d", tc.desc, tc.pb, size, len(b)) + t.Logf("%v: bytes: %#v", tc.desc, b) + } + } +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/skip_gogo.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/skip_gogo.go new file mode 100644 index 0000000000000..4fe7e0815c9f3 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/skip_gogo.go @@ -0,0 +1,117 @@ +// Copyright (c) 2013, Vastech SA (PTY) LTD. All rights reserved. +// http://github.com/gogo/protobuf/gogoproto +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +import ( + "fmt" + "io" +) + +func Skip(data []byte) (n int, err error) { + l := len(data) + index := 0 + for index < l { + var wire uint64 + for shift := uint(0); ; shift += 7 { + if index >= l { + return 0, io.ErrUnexpectedEOF + } + b := data[index] + index++ + wire |= (uint64(b) & 0x7F) << shift + if b < 0x80 { + break + } + } + wireType := int(wire & 0x7) + switch wireType { + case 0: + for { + if index >= l { + return 0, io.ErrUnexpectedEOF + } + index++ + if data[index-1] < 0x80 { + break + } + } + return index, nil + case 1: + index += 8 + return index, nil + case 2: + var length int + for shift := uint(0); ; shift += 7 { + if index >= l { + return 0, io.ErrUnexpectedEOF + } + b := data[index] + index++ + length |= (int(b) & 0x7F) << shift + if b < 0x80 { + break + } + } + index += length + return index, nil + case 3: + for { + var innerWire uint64 + var start int = index + for shift := uint(0); ; shift += 7 { + if index >= l { + return 0, io.ErrUnexpectedEOF + } + b := data[index] + index++ + innerWire |= (uint64(b) & 0x7F) << shift + if b < 0x80 { + break + } + } + innerWireType := int(innerWire & 0x7) + if innerWireType == 4 { + break + } + next, err := Skip(data[start:]) + if err != nil { + return 0, err + } + index = start + next + } + return index, nil + case 4: + return index, nil + case 5: + index += 4 + return index, nil + default: + return 0, fmt.Errorf("proto: illegal wireType %d", wireType) + } + } + panic("unreachable") +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/Makefile b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/Makefile new file mode 100644 index 0000000000000..1e676c37f8457 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/Makefile @@ -0,0 +1,37 @@ +# Go support for Protocol Buffers - Google's data interchange format +# +# Copyright 2010 The Go Authors. All rights reserved. +# https://github.com/golang/protobuf +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are +# met: +# +# * Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +# * Redistributions in binary form must reproduce the above +# copyright notice, this list of conditions and the following disclaimer +# in the documentation and/or other materials provided with the +# distribution. +# * Neither the name of Google Inc. nor the names of its +# contributors may be used to endorse or promote products derived from +# this software without specific prior written permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +all: regenerate + +regenerate: + go install github.com/gogo/protobuf/protoc-gen-gogo/version/protoc-min-version + protoc-min-version --version="3.0.0" --gogo_out=. test.proto + diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/golden_test.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/golden_test.go new file mode 100644 index 0000000000000..8e84515377a96 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/golden_test.go @@ -0,0 +1,86 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2012 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// Verify that the compiler output for test.proto is unchanged. + +package testdata + +import ( + "crypto/sha1" + "fmt" + "io/ioutil" + "os" + "os/exec" + "path/filepath" + "testing" +) + +// sum returns in string form (for easy comparison) the SHA-1 hash of the named file. +func sum(t *testing.T, name string) string { + data, err := ioutil.ReadFile(name) + if err != nil { + t.Fatal(err) + } + t.Logf("sum(%q): length is %d", name, len(data)) + hash := sha1.New() + _, err = hash.Write(data) + if err != nil { + t.Fatal(err) + } + return fmt.Sprintf("% x", hash.Sum(nil)) +} + +func run(t *testing.T, name string, args ...string) { + cmd := exec.Command(name, args...) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + err := cmd.Run() + if err != nil { + t.Fatal(err) + } +} + +func TestGolden(t *testing.T) { + // Compute the original checksum. + goldenSum := sum(t, "test.pb.go") + // Run the proto compiler. + run(t, "protoc", "--gogo_out="+os.TempDir(), "test.proto") + newFile := filepath.Join(os.TempDir(), "test.pb.go") + defer os.Remove(newFile) + // Compute the new checksum. + newSum := sum(t, newFile) + // Verify + if newSum != goldenSum { + run(t, "diff", "-u", "test.pb.go", newFile) + t.Fatal("Code generated by protoc-gen-go has changed; update test.pb.go") + } +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/test.pb.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/test.pb.go new file mode 100644 index 0000000000000..8bc688c2f927a --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/test.pb.go @@ -0,0 +1,2746 @@ +// Code generated by protoc-gen-gogo. +// source: test.proto +// DO NOT EDIT! + +/* +Package testdata is a generated protocol buffer package. + +It is generated from these files: + test.proto + +It has these top-level messages: + GoEnum + GoTestField + GoTest + GoSkipTest + NonPackedTest + PackedTest + MaxTag + OldMessage + NewMessage + InnerMessage + OtherMessage + MyMessage + Ext + DefaultsMessage + MyMessageSet + Empty + MessageList + Strings + Defaults + SubDefaults + RepeatedEnum + MoreRepeated + GroupOld + GroupNew + FloatingPoint + MessageWithMap +*/ +package testdata + +import proto "github.com/gogo/protobuf/proto" +import math "math" + +// Reference imports to suppress errors if they are not otherwise used. +var _ = proto.Marshal +var _ = math.Inf + +type FOO int32 + +const ( + FOO_FOO1 FOO = 1 +) + +var FOO_name = map[int32]string{ + 1: "FOO1", +} +var FOO_value = map[string]int32{ + "FOO1": 1, +} + +func (x FOO) Enum() *FOO { + p := new(FOO) + *p = x + return p +} +func (x FOO) String() string { + return proto.EnumName(FOO_name, int32(x)) +} +func (x *FOO) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(FOO_value, data, "FOO") + if err != nil { + return err + } + *x = FOO(value) + return nil +} + +// An enum, for completeness. +type GoTest_KIND int32 + +const ( + GoTest_VOID GoTest_KIND = 0 + // Basic types + GoTest_BOOL GoTest_KIND = 1 + GoTest_BYTES GoTest_KIND = 2 + GoTest_FINGERPRINT GoTest_KIND = 3 + GoTest_FLOAT GoTest_KIND = 4 + GoTest_INT GoTest_KIND = 5 + GoTest_STRING GoTest_KIND = 6 + GoTest_TIME GoTest_KIND = 7 + // Groupings + GoTest_TUPLE GoTest_KIND = 8 + GoTest_ARRAY GoTest_KIND = 9 + GoTest_MAP GoTest_KIND = 10 + // Table types + GoTest_TABLE GoTest_KIND = 11 + // Functions + GoTest_FUNCTION GoTest_KIND = 12 +) + +var GoTest_KIND_name = map[int32]string{ + 0: "VOID", + 1: "BOOL", + 2: "BYTES", + 3: "FINGERPRINT", + 4: "FLOAT", + 5: "INT", + 6: "STRING", + 7: "TIME", + 8: "TUPLE", + 9: "ARRAY", + 10: "MAP", + 11: "TABLE", + 12: "FUNCTION", +} +var GoTest_KIND_value = map[string]int32{ + "VOID": 0, + "BOOL": 1, + "BYTES": 2, + "FINGERPRINT": 3, + "FLOAT": 4, + "INT": 5, + "STRING": 6, + "TIME": 7, + "TUPLE": 8, + "ARRAY": 9, + "MAP": 10, + "TABLE": 11, + "FUNCTION": 12, +} + +func (x GoTest_KIND) Enum() *GoTest_KIND { + p := new(GoTest_KIND) + *p = x + return p +} +func (x GoTest_KIND) String() string { + return proto.EnumName(GoTest_KIND_name, int32(x)) +} +func (x *GoTest_KIND) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(GoTest_KIND_value, data, "GoTest_KIND") + if err != nil { + return err + } + *x = GoTest_KIND(value) + return nil +} + +type MyMessage_Color int32 + +const ( + MyMessage_RED MyMessage_Color = 0 + MyMessage_GREEN MyMessage_Color = 1 + MyMessage_BLUE MyMessage_Color = 2 +) + +var MyMessage_Color_name = map[int32]string{ + 0: "RED", + 1: "GREEN", + 2: "BLUE", +} +var MyMessage_Color_value = map[string]int32{ + "RED": 0, + "GREEN": 1, + "BLUE": 2, +} + +func (x MyMessage_Color) Enum() *MyMessage_Color { + p := new(MyMessage_Color) + *p = x + return p +} +func (x MyMessage_Color) String() string { + return proto.EnumName(MyMessage_Color_name, int32(x)) +} +func (x *MyMessage_Color) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(MyMessage_Color_value, data, "MyMessage_Color") + if err != nil { + return err + } + *x = MyMessage_Color(value) + return nil +} + +type DefaultsMessage_DefaultsEnum int32 + +const ( + DefaultsMessage_ZERO DefaultsMessage_DefaultsEnum = 0 + DefaultsMessage_ONE DefaultsMessage_DefaultsEnum = 1 + DefaultsMessage_TWO DefaultsMessage_DefaultsEnum = 2 +) + +var DefaultsMessage_DefaultsEnum_name = map[int32]string{ + 0: "ZERO", + 1: "ONE", + 2: "TWO", +} +var DefaultsMessage_DefaultsEnum_value = map[string]int32{ + "ZERO": 0, + "ONE": 1, + "TWO": 2, +} + +func (x DefaultsMessage_DefaultsEnum) Enum() *DefaultsMessage_DefaultsEnum { + p := new(DefaultsMessage_DefaultsEnum) + *p = x + return p +} +func (x DefaultsMessage_DefaultsEnum) String() string { + return proto.EnumName(DefaultsMessage_DefaultsEnum_name, int32(x)) +} +func (x *DefaultsMessage_DefaultsEnum) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(DefaultsMessage_DefaultsEnum_value, data, "DefaultsMessage_DefaultsEnum") + if err != nil { + return err + } + *x = DefaultsMessage_DefaultsEnum(value) + return nil +} + +type Defaults_Color int32 + +const ( + Defaults_RED Defaults_Color = 0 + Defaults_GREEN Defaults_Color = 1 + Defaults_BLUE Defaults_Color = 2 +) + +var Defaults_Color_name = map[int32]string{ + 0: "RED", + 1: "GREEN", + 2: "BLUE", +} +var Defaults_Color_value = map[string]int32{ + "RED": 0, + "GREEN": 1, + "BLUE": 2, +} + +func (x Defaults_Color) Enum() *Defaults_Color { + p := new(Defaults_Color) + *p = x + return p +} +func (x Defaults_Color) String() string { + return proto.EnumName(Defaults_Color_name, int32(x)) +} +func (x *Defaults_Color) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(Defaults_Color_value, data, "Defaults_Color") + if err != nil { + return err + } + *x = Defaults_Color(value) + return nil +} + +type RepeatedEnum_Color int32 + +const ( + RepeatedEnum_RED RepeatedEnum_Color = 1 +) + +var RepeatedEnum_Color_name = map[int32]string{ + 1: "RED", +} +var RepeatedEnum_Color_value = map[string]int32{ + "RED": 1, +} + +func (x RepeatedEnum_Color) Enum() *RepeatedEnum_Color { + p := new(RepeatedEnum_Color) + *p = x + return p +} +func (x RepeatedEnum_Color) String() string { + return proto.EnumName(RepeatedEnum_Color_name, int32(x)) +} +func (x *RepeatedEnum_Color) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(RepeatedEnum_Color_value, data, "RepeatedEnum_Color") + if err != nil { + return err + } + *x = RepeatedEnum_Color(value) + return nil +} + +type GoEnum struct { + Foo *FOO `protobuf:"varint,1,req,name=foo,enum=testdata.FOO" json:"foo,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoEnum) Reset() { *m = GoEnum{} } +func (m *GoEnum) String() string { return proto.CompactTextString(m) } +func (*GoEnum) ProtoMessage() {} + +func (m *GoEnum) GetFoo() FOO { + if m != nil && m.Foo != nil { + return *m.Foo + } + return FOO_FOO1 +} + +type GoTestField struct { + Label *string `protobuf:"bytes,1,req" json:"Label,omitempty"` + Type *string `protobuf:"bytes,2,req" json:"Type,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTestField) Reset() { *m = GoTestField{} } +func (m *GoTestField) String() string { return proto.CompactTextString(m) } +func (*GoTestField) ProtoMessage() {} + +func (m *GoTestField) GetLabel() string { + if m != nil && m.Label != nil { + return *m.Label + } + return "" +} + +func (m *GoTestField) GetType() string { + if m != nil && m.Type != nil { + return *m.Type + } + return "" +} + +type GoTest struct { + // Some typical parameters + Kind *GoTest_KIND `protobuf:"varint,1,req,enum=testdata.GoTest_KIND" json:"Kind,omitempty"` + Table *string `protobuf:"bytes,2,opt" json:"Table,omitempty"` + Param *int32 `protobuf:"varint,3,opt" json:"Param,omitempty"` + // Required, repeated and optional foreign fields. + RequiredField *GoTestField `protobuf:"bytes,4,req" json:"RequiredField,omitempty"` + RepeatedField []*GoTestField `protobuf:"bytes,5,rep" json:"RepeatedField,omitempty"` + OptionalField *GoTestField `protobuf:"bytes,6,opt" json:"OptionalField,omitempty"` + // Required fields of all basic types + F_BoolRequired *bool `protobuf:"varint,10,req,name=F_Bool_required" json:"F_Bool_required,omitempty"` + F_Int32Required *int32 `protobuf:"varint,11,req,name=F_Int32_required" json:"F_Int32_required,omitempty"` + F_Int64Required *int64 `protobuf:"varint,12,req,name=F_Int64_required" json:"F_Int64_required,omitempty"` + F_Fixed32Required *uint32 `protobuf:"fixed32,13,req,name=F_Fixed32_required" json:"F_Fixed32_required,omitempty"` + F_Fixed64Required *uint64 `protobuf:"fixed64,14,req,name=F_Fixed64_required" json:"F_Fixed64_required,omitempty"` + F_Uint32Required *uint32 `protobuf:"varint,15,req,name=F_Uint32_required" json:"F_Uint32_required,omitempty"` + F_Uint64Required *uint64 `protobuf:"varint,16,req,name=F_Uint64_required" json:"F_Uint64_required,omitempty"` + F_FloatRequired *float32 `protobuf:"fixed32,17,req,name=F_Float_required" json:"F_Float_required,omitempty"` + F_DoubleRequired *float64 `protobuf:"fixed64,18,req,name=F_Double_required" json:"F_Double_required,omitempty"` + F_StringRequired *string `protobuf:"bytes,19,req,name=F_String_required" json:"F_String_required,omitempty"` + F_BytesRequired []byte `protobuf:"bytes,101,req,name=F_Bytes_required" json:"F_Bytes_required,omitempty"` + F_Sint32Required *int32 `protobuf:"zigzag32,102,req,name=F_Sint32_required" json:"F_Sint32_required,omitempty"` + F_Sint64Required *int64 `protobuf:"zigzag64,103,req,name=F_Sint64_required" json:"F_Sint64_required,omitempty"` + // Repeated fields of all basic types + F_BoolRepeated []bool `protobuf:"varint,20,rep,name=F_Bool_repeated" json:"F_Bool_repeated,omitempty"` + F_Int32Repeated []int32 `protobuf:"varint,21,rep,name=F_Int32_repeated" json:"F_Int32_repeated,omitempty"` + F_Int64Repeated []int64 `protobuf:"varint,22,rep,name=F_Int64_repeated" json:"F_Int64_repeated,omitempty"` + F_Fixed32Repeated []uint32 `protobuf:"fixed32,23,rep,name=F_Fixed32_repeated" json:"F_Fixed32_repeated,omitempty"` + F_Fixed64Repeated []uint64 `protobuf:"fixed64,24,rep,name=F_Fixed64_repeated" json:"F_Fixed64_repeated,omitempty"` + F_Uint32Repeated []uint32 `protobuf:"varint,25,rep,name=F_Uint32_repeated" json:"F_Uint32_repeated,omitempty"` + F_Uint64Repeated []uint64 `protobuf:"varint,26,rep,name=F_Uint64_repeated" json:"F_Uint64_repeated,omitempty"` + F_FloatRepeated []float32 `protobuf:"fixed32,27,rep,name=F_Float_repeated" json:"F_Float_repeated,omitempty"` + F_DoubleRepeated []float64 `protobuf:"fixed64,28,rep,name=F_Double_repeated" json:"F_Double_repeated,omitempty"` + F_StringRepeated []string `protobuf:"bytes,29,rep,name=F_String_repeated" json:"F_String_repeated,omitempty"` + F_BytesRepeated [][]byte `protobuf:"bytes,201,rep,name=F_Bytes_repeated" json:"F_Bytes_repeated,omitempty"` + F_Sint32Repeated []int32 `protobuf:"zigzag32,202,rep,name=F_Sint32_repeated" json:"F_Sint32_repeated,omitempty"` + F_Sint64Repeated []int64 `protobuf:"zigzag64,203,rep,name=F_Sint64_repeated" json:"F_Sint64_repeated,omitempty"` + // Optional fields of all basic types + F_BoolOptional *bool `protobuf:"varint,30,opt,name=F_Bool_optional" json:"F_Bool_optional,omitempty"` + F_Int32Optional *int32 `protobuf:"varint,31,opt,name=F_Int32_optional" json:"F_Int32_optional,omitempty"` + F_Int64Optional *int64 `protobuf:"varint,32,opt,name=F_Int64_optional" json:"F_Int64_optional,omitempty"` + F_Fixed32Optional *uint32 `protobuf:"fixed32,33,opt,name=F_Fixed32_optional" json:"F_Fixed32_optional,omitempty"` + F_Fixed64Optional *uint64 `protobuf:"fixed64,34,opt,name=F_Fixed64_optional" json:"F_Fixed64_optional,omitempty"` + F_Uint32Optional *uint32 `protobuf:"varint,35,opt,name=F_Uint32_optional" json:"F_Uint32_optional,omitempty"` + F_Uint64Optional *uint64 `protobuf:"varint,36,opt,name=F_Uint64_optional" json:"F_Uint64_optional,omitempty"` + F_FloatOptional *float32 `protobuf:"fixed32,37,opt,name=F_Float_optional" json:"F_Float_optional,omitempty"` + F_DoubleOptional *float64 `protobuf:"fixed64,38,opt,name=F_Double_optional" json:"F_Double_optional,omitempty"` + F_StringOptional *string `protobuf:"bytes,39,opt,name=F_String_optional" json:"F_String_optional,omitempty"` + F_BytesOptional []byte `protobuf:"bytes,301,opt,name=F_Bytes_optional" json:"F_Bytes_optional,omitempty"` + F_Sint32Optional *int32 `protobuf:"zigzag32,302,opt,name=F_Sint32_optional" json:"F_Sint32_optional,omitempty"` + F_Sint64Optional *int64 `protobuf:"zigzag64,303,opt,name=F_Sint64_optional" json:"F_Sint64_optional,omitempty"` + // Default-valued fields of all basic types + F_BoolDefaulted *bool `protobuf:"varint,40,opt,name=F_Bool_defaulted,def=1" json:"F_Bool_defaulted,omitempty"` + F_Int32Defaulted *int32 `protobuf:"varint,41,opt,name=F_Int32_defaulted,def=32" json:"F_Int32_defaulted,omitempty"` + F_Int64Defaulted *int64 `protobuf:"varint,42,opt,name=F_Int64_defaulted,def=64" json:"F_Int64_defaulted,omitempty"` + F_Fixed32Defaulted *uint32 `protobuf:"fixed32,43,opt,name=F_Fixed32_defaulted,def=320" json:"F_Fixed32_defaulted,omitempty"` + F_Fixed64Defaulted *uint64 `protobuf:"fixed64,44,opt,name=F_Fixed64_defaulted,def=640" json:"F_Fixed64_defaulted,omitempty"` + F_Uint32Defaulted *uint32 `protobuf:"varint,45,opt,name=F_Uint32_defaulted,def=3200" json:"F_Uint32_defaulted,omitempty"` + F_Uint64Defaulted *uint64 `protobuf:"varint,46,opt,name=F_Uint64_defaulted,def=6400" json:"F_Uint64_defaulted,omitempty"` + F_FloatDefaulted *float32 `protobuf:"fixed32,47,opt,name=F_Float_defaulted,def=314159" json:"F_Float_defaulted,omitempty"` + F_DoubleDefaulted *float64 `protobuf:"fixed64,48,opt,name=F_Double_defaulted,def=271828" json:"F_Double_defaulted,omitempty"` + F_StringDefaulted *string `protobuf:"bytes,49,opt,name=F_String_defaulted,def=hello, \"world!\"\n" json:"F_String_defaulted,omitempty"` + F_BytesDefaulted []byte `protobuf:"bytes,401,opt,name=F_Bytes_defaulted,def=Bignose" json:"F_Bytes_defaulted,omitempty"` + F_Sint32Defaulted *int32 `protobuf:"zigzag32,402,opt,name=F_Sint32_defaulted,def=-32" json:"F_Sint32_defaulted,omitempty"` + F_Sint64Defaulted *int64 `protobuf:"zigzag64,403,opt,name=F_Sint64_defaulted,def=-64" json:"F_Sint64_defaulted,omitempty"` + // Packed repeated fields (no string or bytes). + F_BoolRepeatedPacked []bool `protobuf:"varint,50,rep,packed,name=F_Bool_repeated_packed" json:"F_Bool_repeated_packed,omitempty"` + F_Int32RepeatedPacked []int32 `protobuf:"varint,51,rep,packed,name=F_Int32_repeated_packed" json:"F_Int32_repeated_packed,omitempty"` + F_Int64RepeatedPacked []int64 `protobuf:"varint,52,rep,packed,name=F_Int64_repeated_packed" json:"F_Int64_repeated_packed,omitempty"` + F_Fixed32RepeatedPacked []uint32 `protobuf:"fixed32,53,rep,packed,name=F_Fixed32_repeated_packed" json:"F_Fixed32_repeated_packed,omitempty"` + F_Fixed64RepeatedPacked []uint64 `protobuf:"fixed64,54,rep,packed,name=F_Fixed64_repeated_packed" json:"F_Fixed64_repeated_packed,omitempty"` + F_Uint32RepeatedPacked []uint32 `protobuf:"varint,55,rep,packed,name=F_Uint32_repeated_packed" json:"F_Uint32_repeated_packed,omitempty"` + F_Uint64RepeatedPacked []uint64 `protobuf:"varint,56,rep,packed,name=F_Uint64_repeated_packed" json:"F_Uint64_repeated_packed,omitempty"` + F_FloatRepeatedPacked []float32 `protobuf:"fixed32,57,rep,packed,name=F_Float_repeated_packed" json:"F_Float_repeated_packed,omitempty"` + F_DoubleRepeatedPacked []float64 `protobuf:"fixed64,58,rep,packed,name=F_Double_repeated_packed" json:"F_Double_repeated_packed,omitempty"` + F_Sint32RepeatedPacked []int32 `protobuf:"zigzag32,502,rep,packed,name=F_Sint32_repeated_packed" json:"F_Sint32_repeated_packed,omitempty"` + F_Sint64RepeatedPacked []int64 `protobuf:"zigzag64,503,rep,packed,name=F_Sint64_repeated_packed" json:"F_Sint64_repeated_packed,omitempty"` + Requiredgroup *GoTest_RequiredGroup `protobuf:"group,70,req,name=RequiredGroup" json:"requiredgroup,omitempty"` + Repeatedgroup []*GoTest_RepeatedGroup `protobuf:"group,80,rep,name=RepeatedGroup" json:"repeatedgroup,omitempty"` + Optionalgroup *GoTest_OptionalGroup `protobuf:"group,90,opt,name=OptionalGroup" json:"optionalgroup,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTest) Reset() { *m = GoTest{} } +func (m *GoTest) String() string { return proto.CompactTextString(m) } +func (*GoTest) ProtoMessage() {} + +const Default_GoTest_F_BoolDefaulted bool = true +const Default_GoTest_F_Int32Defaulted int32 = 32 +const Default_GoTest_F_Int64Defaulted int64 = 64 +const Default_GoTest_F_Fixed32Defaulted uint32 = 320 +const Default_GoTest_F_Fixed64Defaulted uint64 = 640 +const Default_GoTest_F_Uint32Defaulted uint32 = 3200 +const Default_GoTest_F_Uint64Defaulted uint64 = 6400 +const Default_GoTest_F_FloatDefaulted float32 = 314159 +const Default_GoTest_F_DoubleDefaulted float64 = 271828 +const Default_GoTest_F_StringDefaulted string = "hello, \"world!\"\n" + +var Default_GoTest_F_BytesDefaulted []byte = []byte("Bignose") + +const Default_GoTest_F_Sint32Defaulted int32 = -32 +const Default_GoTest_F_Sint64Defaulted int64 = -64 + +func (m *GoTest) GetKind() GoTest_KIND { + if m != nil && m.Kind != nil { + return *m.Kind + } + return GoTest_VOID +} + +func (m *GoTest) GetTable() string { + if m != nil && m.Table != nil { + return *m.Table + } + return "" +} + +func (m *GoTest) GetParam() int32 { + if m != nil && m.Param != nil { + return *m.Param + } + return 0 +} + +func (m *GoTest) GetRequiredField() *GoTestField { + if m != nil { + return m.RequiredField + } + return nil +} + +func (m *GoTest) GetRepeatedField() []*GoTestField { + if m != nil { + return m.RepeatedField + } + return nil +} + +func (m *GoTest) GetOptionalField() *GoTestField { + if m != nil { + return m.OptionalField + } + return nil +} + +func (m *GoTest) GetF_BoolRequired() bool { + if m != nil && m.F_BoolRequired != nil { + return *m.F_BoolRequired + } + return false +} + +func (m *GoTest) GetF_Int32Required() int32 { + if m != nil && m.F_Int32Required != nil { + return *m.F_Int32Required + } + return 0 +} + +func (m *GoTest) GetF_Int64Required() int64 { + if m != nil && m.F_Int64Required != nil { + return *m.F_Int64Required + } + return 0 +} + +func (m *GoTest) GetF_Fixed32Required() uint32 { + if m != nil && m.F_Fixed32Required != nil { + return *m.F_Fixed32Required + } + return 0 +} + +func (m *GoTest) GetF_Fixed64Required() uint64 { + if m != nil && m.F_Fixed64Required != nil { + return *m.F_Fixed64Required + } + return 0 +} + +func (m *GoTest) GetF_Uint32Required() uint32 { + if m != nil && m.F_Uint32Required != nil { + return *m.F_Uint32Required + } + return 0 +} + +func (m *GoTest) GetF_Uint64Required() uint64 { + if m != nil && m.F_Uint64Required != nil { + return *m.F_Uint64Required + } + return 0 +} + +func (m *GoTest) GetF_FloatRequired() float32 { + if m != nil && m.F_FloatRequired != nil { + return *m.F_FloatRequired + } + return 0 +} + +func (m *GoTest) GetF_DoubleRequired() float64 { + if m != nil && m.F_DoubleRequired != nil { + return *m.F_DoubleRequired + } + return 0 +} + +func (m *GoTest) GetF_StringRequired() string { + if m != nil && m.F_StringRequired != nil { + return *m.F_StringRequired + } + return "" +} + +func (m *GoTest) GetF_BytesRequired() []byte { + if m != nil { + return m.F_BytesRequired + } + return nil +} + +func (m *GoTest) GetF_Sint32Required() int32 { + if m != nil && m.F_Sint32Required != nil { + return *m.F_Sint32Required + } + return 0 +} + +func (m *GoTest) GetF_Sint64Required() int64 { + if m != nil && m.F_Sint64Required != nil { + return *m.F_Sint64Required + } + return 0 +} + +func (m *GoTest) GetF_BoolRepeated() []bool { + if m != nil { + return m.F_BoolRepeated + } + return nil +} + +func (m *GoTest) GetF_Int32Repeated() []int32 { + if m != nil { + return m.F_Int32Repeated + } + return nil +} + +func (m *GoTest) GetF_Int64Repeated() []int64 { + if m != nil { + return m.F_Int64Repeated + } + return nil +} + +func (m *GoTest) GetF_Fixed32Repeated() []uint32 { + if m != nil { + return m.F_Fixed32Repeated + } + return nil +} + +func (m *GoTest) GetF_Fixed64Repeated() []uint64 { + if m != nil { + return m.F_Fixed64Repeated + } + return nil +} + +func (m *GoTest) GetF_Uint32Repeated() []uint32 { + if m != nil { + return m.F_Uint32Repeated + } + return nil +} + +func (m *GoTest) GetF_Uint64Repeated() []uint64 { + if m != nil { + return m.F_Uint64Repeated + } + return nil +} + +func (m *GoTest) GetF_FloatRepeated() []float32 { + if m != nil { + return m.F_FloatRepeated + } + return nil +} + +func (m *GoTest) GetF_DoubleRepeated() []float64 { + if m != nil { + return m.F_DoubleRepeated + } + return nil +} + +func (m *GoTest) GetF_StringRepeated() []string { + if m != nil { + return m.F_StringRepeated + } + return nil +} + +func (m *GoTest) GetF_BytesRepeated() [][]byte { + if m != nil { + return m.F_BytesRepeated + } + return nil +} + +func (m *GoTest) GetF_Sint32Repeated() []int32 { + if m != nil { + return m.F_Sint32Repeated + } + return nil +} + +func (m *GoTest) GetF_Sint64Repeated() []int64 { + if m != nil { + return m.F_Sint64Repeated + } + return nil +} + +func (m *GoTest) GetF_BoolOptional() bool { + if m != nil && m.F_BoolOptional != nil { + return *m.F_BoolOptional + } + return false +} + +func (m *GoTest) GetF_Int32Optional() int32 { + if m != nil && m.F_Int32Optional != nil { + return *m.F_Int32Optional + } + return 0 +} + +func (m *GoTest) GetF_Int64Optional() int64 { + if m != nil && m.F_Int64Optional != nil { + return *m.F_Int64Optional + } + return 0 +} + +func (m *GoTest) GetF_Fixed32Optional() uint32 { + if m != nil && m.F_Fixed32Optional != nil { + return *m.F_Fixed32Optional + } + return 0 +} + +func (m *GoTest) GetF_Fixed64Optional() uint64 { + if m != nil && m.F_Fixed64Optional != nil { + return *m.F_Fixed64Optional + } + return 0 +} + +func (m *GoTest) GetF_Uint32Optional() uint32 { + if m != nil && m.F_Uint32Optional != nil { + return *m.F_Uint32Optional + } + return 0 +} + +func (m *GoTest) GetF_Uint64Optional() uint64 { + if m != nil && m.F_Uint64Optional != nil { + return *m.F_Uint64Optional + } + return 0 +} + +func (m *GoTest) GetF_FloatOptional() float32 { + if m != nil && m.F_FloatOptional != nil { + return *m.F_FloatOptional + } + return 0 +} + +func (m *GoTest) GetF_DoubleOptional() float64 { + if m != nil && m.F_DoubleOptional != nil { + return *m.F_DoubleOptional + } + return 0 +} + +func (m *GoTest) GetF_StringOptional() string { + if m != nil && m.F_StringOptional != nil { + return *m.F_StringOptional + } + return "" +} + +func (m *GoTest) GetF_BytesOptional() []byte { + if m != nil { + return m.F_BytesOptional + } + return nil +} + +func (m *GoTest) GetF_Sint32Optional() int32 { + if m != nil && m.F_Sint32Optional != nil { + return *m.F_Sint32Optional + } + return 0 +} + +func (m *GoTest) GetF_Sint64Optional() int64 { + if m != nil && m.F_Sint64Optional != nil { + return *m.F_Sint64Optional + } + return 0 +} + +func (m *GoTest) GetF_BoolDefaulted() bool { + if m != nil && m.F_BoolDefaulted != nil { + return *m.F_BoolDefaulted + } + return Default_GoTest_F_BoolDefaulted +} + +func (m *GoTest) GetF_Int32Defaulted() int32 { + if m != nil && m.F_Int32Defaulted != nil { + return *m.F_Int32Defaulted + } + return Default_GoTest_F_Int32Defaulted +} + +func (m *GoTest) GetF_Int64Defaulted() int64 { + if m != nil && m.F_Int64Defaulted != nil { + return *m.F_Int64Defaulted + } + return Default_GoTest_F_Int64Defaulted +} + +func (m *GoTest) GetF_Fixed32Defaulted() uint32 { + if m != nil && m.F_Fixed32Defaulted != nil { + return *m.F_Fixed32Defaulted + } + return Default_GoTest_F_Fixed32Defaulted +} + +func (m *GoTest) GetF_Fixed64Defaulted() uint64 { + if m != nil && m.F_Fixed64Defaulted != nil { + return *m.F_Fixed64Defaulted + } + return Default_GoTest_F_Fixed64Defaulted +} + +func (m *GoTest) GetF_Uint32Defaulted() uint32 { + if m != nil && m.F_Uint32Defaulted != nil { + return *m.F_Uint32Defaulted + } + return Default_GoTest_F_Uint32Defaulted +} + +func (m *GoTest) GetF_Uint64Defaulted() uint64 { + if m != nil && m.F_Uint64Defaulted != nil { + return *m.F_Uint64Defaulted + } + return Default_GoTest_F_Uint64Defaulted +} + +func (m *GoTest) GetF_FloatDefaulted() float32 { + if m != nil && m.F_FloatDefaulted != nil { + return *m.F_FloatDefaulted + } + return Default_GoTest_F_FloatDefaulted +} + +func (m *GoTest) GetF_DoubleDefaulted() float64 { + if m != nil && m.F_DoubleDefaulted != nil { + return *m.F_DoubleDefaulted + } + return Default_GoTest_F_DoubleDefaulted +} + +func (m *GoTest) GetF_StringDefaulted() string { + if m != nil && m.F_StringDefaulted != nil { + return *m.F_StringDefaulted + } + return Default_GoTest_F_StringDefaulted +} + +func (m *GoTest) GetF_BytesDefaulted() []byte { + if m != nil && m.F_BytesDefaulted != nil { + return m.F_BytesDefaulted + } + return append([]byte(nil), Default_GoTest_F_BytesDefaulted...) +} + +func (m *GoTest) GetF_Sint32Defaulted() int32 { + if m != nil && m.F_Sint32Defaulted != nil { + return *m.F_Sint32Defaulted + } + return Default_GoTest_F_Sint32Defaulted +} + +func (m *GoTest) GetF_Sint64Defaulted() int64 { + if m != nil && m.F_Sint64Defaulted != nil { + return *m.F_Sint64Defaulted + } + return Default_GoTest_F_Sint64Defaulted +} + +func (m *GoTest) GetF_BoolRepeatedPacked() []bool { + if m != nil { + return m.F_BoolRepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Int32RepeatedPacked() []int32 { + if m != nil { + return m.F_Int32RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Int64RepeatedPacked() []int64 { + if m != nil { + return m.F_Int64RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Fixed32RepeatedPacked() []uint32 { + if m != nil { + return m.F_Fixed32RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Fixed64RepeatedPacked() []uint64 { + if m != nil { + return m.F_Fixed64RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Uint32RepeatedPacked() []uint32 { + if m != nil { + return m.F_Uint32RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Uint64RepeatedPacked() []uint64 { + if m != nil { + return m.F_Uint64RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_FloatRepeatedPacked() []float32 { + if m != nil { + return m.F_FloatRepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_DoubleRepeatedPacked() []float64 { + if m != nil { + return m.F_DoubleRepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Sint32RepeatedPacked() []int32 { + if m != nil { + return m.F_Sint32RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Sint64RepeatedPacked() []int64 { + if m != nil { + return m.F_Sint64RepeatedPacked + } + return nil +} + +func (m *GoTest) GetRequiredgroup() *GoTest_RequiredGroup { + if m != nil { + return m.Requiredgroup + } + return nil +} + +func (m *GoTest) GetRepeatedgroup() []*GoTest_RepeatedGroup { + if m != nil { + return m.Repeatedgroup + } + return nil +} + +func (m *GoTest) GetOptionalgroup() *GoTest_OptionalGroup { + if m != nil { + return m.Optionalgroup + } + return nil +} + +// Required, repeated, and optional groups. +type GoTest_RequiredGroup struct { + RequiredField *string `protobuf:"bytes,71,req" json:"RequiredField,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTest_RequiredGroup) Reset() { *m = GoTest_RequiredGroup{} } +func (m *GoTest_RequiredGroup) String() string { return proto.CompactTextString(m) } +func (*GoTest_RequiredGroup) ProtoMessage() {} + +func (m *GoTest_RequiredGroup) GetRequiredField() string { + if m != nil && m.RequiredField != nil { + return *m.RequiredField + } + return "" +} + +type GoTest_RepeatedGroup struct { + RequiredField *string `protobuf:"bytes,81,req" json:"RequiredField,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTest_RepeatedGroup) Reset() { *m = GoTest_RepeatedGroup{} } +func (m *GoTest_RepeatedGroup) String() string { return proto.CompactTextString(m) } +func (*GoTest_RepeatedGroup) ProtoMessage() {} + +func (m *GoTest_RepeatedGroup) GetRequiredField() string { + if m != nil && m.RequiredField != nil { + return *m.RequiredField + } + return "" +} + +type GoTest_OptionalGroup struct { + RequiredField *string `protobuf:"bytes,91,req" json:"RequiredField,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTest_OptionalGroup) Reset() { *m = GoTest_OptionalGroup{} } +func (m *GoTest_OptionalGroup) String() string { return proto.CompactTextString(m) } +func (*GoTest_OptionalGroup) ProtoMessage() {} + +func (m *GoTest_OptionalGroup) GetRequiredField() string { + if m != nil && m.RequiredField != nil { + return *m.RequiredField + } + return "" +} + +// For testing skipping of unrecognized fields. +// Numbers are all big, larger than tag numbers in GoTestField, +// the message used in the corresponding test. +type GoSkipTest struct { + SkipInt32 *int32 `protobuf:"varint,11,req,name=skip_int32" json:"skip_int32,omitempty"` + SkipFixed32 *uint32 `protobuf:"fixed32,12,req,name=skip_fixed32" json:"skip_fixed32,omitempty"` + SkipFixed64 *uint64 `protobuf:"fixed64,13,req,name=skip_fixed64" json:"skip_fixed64,omitempty"` + SkipString *string `protobuf:"bytes,14,req,name=skip_string" json:"skip_string,omitempty"` + Skipgroup *GoSkipTest_SkipGroup `protobuf:"group,15,req,name=SkipGroup" json:"skipgroup,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoSkipTest) Reset() { *m = GoSkipTest{} } +func (m *GoSkipTest) String() string { return proto.CompactTextString(m) } +func (*GoSkipTest) ProtoMessage() {} + +func (m *GoSkipTest) GetSkipInt32() int32 { + if m != nil && m.SkipInt32 != nil { + return *m.SkipInt32 + } + return 0 +} + +func (m *GoSkipTest) GetSkipFixed32() uint32 { + if m != nil && m.SkipFixed32 != nil { + return *m.SkipFixed32 + } + return 0 +} + +func (m *GoSkipTest) GetSkipFixed64() uint64 { + if m != nil && m.SkipFixed64 != nil { + return *m.SkipFixed64 + } + return 0 +} + +func (m *GoSkipTest) GetSkipString() string { + if m != nil && m.SkipString != nil { + return *m.SkipString + } + return "" +} + +func (m *GoSkipTest) GetSkipgroup() *GoSkipTest_SkipGroup { + if m != nil { + return m.Skipgroup + } + return nil +} + +type GoSkipTest_SkipGroup struct { + GroupInt32 *int32 `protobuf:"varint,16,req,name=group_int32" json:"group_int32,omitempty"` + GroupString *string `protobuf:"bytes,17,req,name=group_string" json:"group_string,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoSkipTest_SkipGroup) Reset() { *m = GoSkipTest_SkipGroup{} } +func (m *GoSkipTest_SkipGroup) String() string { return proto.CompactTextString(m) } +func (*GoSkipTest_SkipGroup) ProtoMessage() {} + +func (m *GoSkipTest_SkipGroup) GetGroupInt32() int32 { + if m != nil && m.GroupInt32 != nil { + return *m.GroupInt32 + } + return 0 +} + +func (m *GoSkipTest_SkipGroup) GetGroupString() string { + if m != nil && m.GroupString != nil { + return *m.GroupString + } + return "" +} + +// For testing packed/non-packed decoder switching. +// A serialized instance of one should be deserializable as the other. +type NonPackedTest struct { + A []int32 `protobuf:"varint,1,rep,name=a" json:"a,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *NonPackedTest) Reset() { *m = NonPackedTest{} } +func (m *NonPackedTest) String() string { return proto.CompactTextString(m) } +func (*NonPackedTest) ProtoMessage() {} + +func (m *NonPackedTest) GetA() []int32 { + if m != nil { + return m.A + } + return nil +} + +type PackedTest struct { + B []int32 `protobuf:"varint,1,rep,packed,name=b" json:"b,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *PackedTest) Reset() { *m = PackedTest{} } +func (m *PackedTest) String() string { return proto.CompactTextString(m) } +func (*PackedTest) ProtoMessage() {} + +func (m *PackedTest) GetB() []int32 { + if m != nil { + return m.B + } + return nil +} + +type MaxTag struct { + // Maximum possible tag number. + LastField *string `protobuf:"bytes,536870911,opt,name=last_field" json:"last_field,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MaxTag) Reset() { *m = MaxTag{} } +func (m *MaxTag) String() string { return proto.CompactTextString(m) } +func (*MaxTag) ProtoMessage() {} + +func (m *MaxTag) GetLastField() string { + if m != nil && m.LastField != nil { + return *m.LastField + } + return "" +} + +type OldMessage struct { + Nested *OldMessage_Nested `protobuf:"bytes,1,opt,name=nested" json:"nested,omitempty"` + Num *int32 `protobuf:"varint,2,opt,name=num" json:"num,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *OldMessage) Reset() { *m = OldMessage{} } +func (m *OldMessage) String() string { return proto.CompactTextString(m) } +func (*OldMessage) ProtoMessage() {} + +func (m *OldMessage) GetNested() *OldMessage_Nested { + if m != nil { + return m.Nested + } + return nil +} + +func (m *OldMessage) GetNum() int32 { + if m != nil && m.Num != nil { + return *m.Num + } + return 0 +} + +type OldMessage_Nested struct { + Name *string `protobuf:"bytes,1,opt,name=name" json:"name,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *OldMessage_Nested) Reset() { *m = OldMessage_Nested{} } +func (m *OldMessage_Nested) String() string { return proto.CompactTextString(m) } +func (*OldMessage_Nested) ProtoMessage() {} + +func (m *OldMessage_Nested) GetName() string { + if m != nil && m.Name != nil { + return *m.Name + } + return "" +} + +// NewMessage is wire compatible with OldMessage; +// imagine it as a future version. +type NewMessage struct { + Nested *NewMessage_Nested `protobuf:"bytes,1,opt,name=nested" json:"nested,omitempty"` + // This is an int32 in OldMessage. + Num *int64 `protobuf:"varint,2,opt,name=num" json:"num,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *NewMessage) Reset() { *m = NewMessage{} } +func (m *NewMessage) String() string { return proto.CompactTextString(m) } +func (*NewMessage) ProtoMessage() {} + +func (m *NewMessage) GetNested() *NewMessage_Nested { + if m != nil { + return m.Nested + } + return nil +} + +func (m *NewMessage) GetNum() int64 { + if m != nil && m.Num != nil { + return *m.Num + } + return 0 +} + +type NewMessage_Nested struct { + Name *string `protobuf:"bytes,1,opt,name=name" json:"name,omitempty"` + FoodGroup *string `protobuf:"bytes,2,opt,name=food_group" json:"food_group,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *NewMessage_Nested) Reset() { *m = NewMessage_Nested{} } +func (m *NewMessage_Nested) String() string { return proto.CompactTextString(m) } +func (*NewMessage_Nested) ProtoMessage() {} + +func (m *NewMessage_Nested) GetName() string { + if m != nil && m.Name != nil { + return *m.Name + } + return "" +} + +func (m *NewMessage_Nested) GetFoodGroup() string { + if m != nil && m.FoodGroup != nil { + return *m.FoodGroup + } + return "" +} + +type InnerMessage struct { + Host *string `protobuf:"bytes,1,req,name=host" json:"host,omitempty"` + Port *int32 `protobuf:"varint,2,opt,name=port,def=4000" json:"port,omitempty"` + Connected *bool `protobuf:"varint,3,opt,name=connected" json:"connected,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *InnerMessage) Reset() { *m = InnerMessage{} } +func (m *InnerMessage) String() string { return proto.CompactTextString(m) } +func (*InnerMessage) ProtoMessage() {} + +const Default_InnerMessage_Port int32 = 4000 + +func (m *InnerMessage) GetHost() string { + if m != nil && m.Host != nil { + return *m.Host + } + return "" +} + +func (m *InnerMessage) GetPort() int32 { + if m != nil && m.Port != nil { + return *m.Port + } + return Default_InnerMessage_Port +} + +func (m *InnerMessage) GetConnected() bool { + if m != nil && m.Connected != nil { + return *m.Connected + } + return false +} + +type OtherMessage struct { + Key *int64 `protobuf:"varint,1,opt,name=key" json:"key,omitempty"` + Value []byte `protobuf:"bytes,2,opt,name=value" json:"value,omitempty"` + Weight *float32 `protobuf:"fixed32,3,opt,name=weight" json:"weight,omitempty"` + Inner *InnerMessage `protobuf:"bytes,4,opt,name=inner" json:"inner,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *OtherMessage) Reset() { *m = OtherMessage{} } +func (m *OtherMessage) String() string { return proto.CompactTextString(m) } +func (*OtherMessage) ProtoMessage() {} + +func (m *OtherMessage) GetKey() int64 { + if m != nil && m.Key != nil { + return *m.Key + } + return 0 +} + +func (m *OtherMessage) GetValue() []byte { + if m != nil { + return m.Value + } + return nil +} + +func (m *OtherMessage) GetWeight() float32 { + if m != nil && m.Weight != nil { + return *m.Weight + } + return 0 +} + +func (m *OtherMessage) GetInner() *InnerMessage { + if m != nil { + return m.Inner + } + return nil +} + +type MyMessage struct { + Count *int32 `protobuf:"varint,1,req,name=count" json:"count,omitempty"` + Name *string `protobuf:"bytes,2,opt,name=name" json:"name,omitempty"` + Quote *string `protobuf:"bytes,3,opt,name=quote" json:"quote,omitempty"` + Pet []string `protobuf:"bytes,4,rep,name=pet" json:"pet,omitempty"` + Inner *InnerMessage `protobuf:"bytes,5,opt,name=inner" json:"inner,omitempty"` + Others []*OtherMessage `protobuf:"bytes,6,rep,name=others" json:"others,omitempty"` + RepInner []*InnerMessage `protobuf:"bytes,12,rep,name=rep_inner" json:"rep_inner,omitempty"` + Bikeshed *MyMessage_Color `protobuf:"varint,7,opt,name=bikeshed,enum=testdata.MyMessage_Color" json:"bikeshed,omitempty"` + Somegroup *MyMessage_SomeGroup `protobuf:"group,8,opt,name=SomeGroup" json:"somegroup,omitempty"` + // This field becomes [][]byte in the generated code. + RepBytes [][]byte `protobuf:"bytes,10,rep,name=rep_bytes" json:"rep_bytes,omitempty"` + Bigfloat *float64 `protobuf:"fixed64,11,opt,name=bigfloat" json:"bigfloat,omitempty"` + XXX_extensions map[int32]proto.Extension `json:"-"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MyMessage) Reset() { *m = MyMessage{} } +func (m *MyMessage) String() string { return proto.CompactTextString(m) } +func (*MyMessage) ProtoMessage() {} + +var extRange_MyMessage = []proto.ExtensionRange{ + {100, 536870911}, +} + +func (*MyMessage) ExtensionRangeArray() []proto.ExtensionRange { + return extRange_MyMessage +} +func (m *MyMessage) ExtensionMap() map[int32]proto.Extension { + if m.XXX_extensions == nil { + m.XXX_extensions = make(map[int32]proto.Extension) + } + return m.XXX_extensions +} + +func (m *MyMessage) GetCount() int32 { + if m != nil && m.Count != nil { + return *m.Count + } + return 0 +} + +func (m *MyMessage) GetName() string { + if m != nil && m.Name != nil { + return *m.Name + } + return "" +} + +func (m *MyMessage) GetQuote() string { + if m != nil && m.Quote != nil { + return *m.Quote + } + return "" +} + +func (m *MyMessage) GetPet() []string { + if m != nil { + return m.Pet + } + return nil +} + +func (m *MyMessage) GetInner() *InnerMessage { + if m != nil { + return m.Inner + } + return nil +} + +func (m *MyMessage) GetOthers() []*OtherMessage { + if m != nil { + return m.Others + } + return nil +} + +func (m *MyMessage) GetRepInner() []*InnerMessage { + if m != nil { + return m.RepInner + } + return nil +} + +func (m *MyMessage) GetBikeshed() MyMessage_Color { + if m != nil && m.Bikeshed != nil { + return *m.Bikeshed + } + return MyMessage_RED +} + +func (m *MyMessage) GetSomegroup() *MyMessage_SomeGroup { + if m != nil { + return m.Somegroup + } + return nil +} + +func (m *MyMessage) GetRepBytes() [][]byte { + if m != nil { + return m.RepBytes + } + return nil +} + +func (m *MyMessage) GetBigfloat() float64 { + if m != nil && m.Bigfloat != nil { + return *m.Bigfloat + } + return 0 +} + +type MyMessage_SomeGroup struct { + GroupField *int32 `protobuf:"varint,9,opt,name=group_field" json:"group_field,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MyMessage_SomeGroup) Reset() { *m = MyMessage_SomeGroup{} } +func (m *MyMessage_SomeGroup) String() string { return proto.CompactTextString(m) } +func (*MyMessage_SomeGroup) ProtoMessage() {} + +func (m *MyMessage_SomeGroup) GetGroupField() int32 { + if m != nil && m.GroupField != nil { + return *m.GroupField + } + return 0 +} + +type Ext struct { + Data *string `protobuf:"bytes,1,opt,name=data" json:"data,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Ext) Reset() { *m = Ext{} } +func (m *Ext) String() string { return proto.CompactTextString(m) } +func (*Ext) ProtoMessage() {} + +func (m *Ext) GetData() string { + if m != nil && m.Data != nil { + return *m.Data + } + return "" +} + +var E_Ext_More = &proto.ExtensionDesc{ + ExtendedType: (*MyMessage)(nil), + ExtensionType: (*Ext)(nil), + Field: 103, + Name: "testdata.Ext.more", + Tag: "bytes,103,opt,name=more", +} + +var E_Ext_Text = &proto.ExtensionDesc{ + ExtendedType: (*MyMessage)(nil), + ExtensionType: (*string)(nil), + Field: 104, + Name: "testdata.Ext.text", + Tag: "bytes,104,opt,name=text", +} + +var E_Ext_Number = &proto.ExtensionDesc{ + ExtendedType: (*MyMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 105, + Name: "testdata.Ext.number", + Tag: "varint,105,opt,name=number", +} + +type DefaultsMessage struct { + XXX_extensions map[int32]proto.Extension `json:"-"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *DefaultsMessage) Reset() { *m = DefaultsMessage{} } +func (m *DefaultsMessage) String() string { return proto.CompactTextString(m) } +func (*DefaultsMessage) ProtoMessage() {} + +var extRange_DefaultsMessage = []proto.ExtensionRange{ + {100, 536870911}, +} + +func (*DefaultsMessage) ExtensionRangeArray() []proto.ExtensionRange { + return extRange_DefaultsMessage +} +func (m *DefaultsMessage) ExtensionMap() map[int32]proto.Extension { + if m.XXX_extensions == nil { + m.XXX_extensions = make(map[int32]proto.Extension) + } + return m.XXX_extensions +} + +type MyMessageSet struct { + XXX_extensions map[int32]proto.Extension `json:"-"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MyMessageSet) Reset() { *m = MyMessageSet{} } +func (m *MyMessageSet) String() string { return proto.CompactTextString(m) } +func (*MyMessageSet) ProtoMessage() {} + +func (m *MyMessageSet) Marshal() ([]byte, error) { + return proto.MarshalMessageSet(m.ExtensionMap()) +} +func (m *MyMessageSet) Unmarshal(buf []byte) error { + return proto.UnmarshalMessageSet(buf, m.ExtensionMap()) +} +func (m *MyMessageSet) MarshalJSON() ([]byte, error) { + return proto.MarshalMessageSetJSON(m.XXX_extensions) +} +func (m *MyMessageSet) UnmarshalJSON(buf []byte) error { + return proto.UnmarshalMessageSetJSON(buf, m.XXX_extensions) +} + +// ensure MyMessageSet satisfies proto.Marshaler and proto.Unmarshaler +var _ proto.Marshaler = (*MyMessageSet)(nil) +var _ proto.Unmarshaler = (*MyMessageSet)(nil) + +var extRange_MyMessageSet = []proto.ExtensionRange{ + {100, 2147483646}, +} + +func (*MyMessageSet) ExtensionRangeArray() []proto.ExtensionRange { + return extRange_MyMessageSet +} +func (m *MyMessageSet) ExtensionMap() map[int32]proto.Extension { + if m.XXX_extensions == nil { + m.XXX_extensions = make(map[int32]proto.Extension) + } + return m.XXX_extensions +} + +type Empty struct { + XXX_unrecognized []byte `json:"-"` +} + +func (m *Empty) Reset() { *m = Empty{} } +func (m *Empty) String() string { return proto.CompactTextString(m) } +func (*Empty) ProtoMessage() {} + +type MessageList struct { + Message []*MessageList_Message `protobuf:"group,1,rep" json:"message,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MessageList) Reset() { *m = MessageList{} } +func (m *MessageList) String() string { return proto.CompactTextString(m) } +func (*MessageList) ProtoMessage() {} + +func (m *MessageList) GetMessage() []*MessageList_Message { + if m != nil { + return m.Message + } + return nil +} + +type MessageList_Message struct { + Name *string `protobuf:"bytes,2,req,name=name" json:"name,omitempty"` + Count *int32 `protobuf:"varint,3,req,name=count" json:"count,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MessageList_Message) Reset() { *m = MessageList_Message{} } +func (m *MessageList_Message) String() string { return proto.CompactTextString(m) } +func (*MessageList_Message) ProtoMessage() {} + +func (m *MessageList_Message) GetName() string { + if m != nil && m.Name != nil { + return *m.Name + } + return "" +} + +func (m *MessageList_Message) GetCount() int32 { + if m != nil && m.Count != nil { + return *m.Count + } + return 0 +} + +type Strings struct { + StringField *string `protobuf:"bytes,1,opt,name=string_field" json:"string_field,omitempty"` + BytesField []byte `protobuf:"bytes,2,opt,name=bytes_field" json:"bytes_field,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Strings) Reset() { *m = Strings{} } +func (m *Strings) String() string { return proto.CompactTextString(m) } +func (*Strings) ProtoMessage() {} + +func (m *Strings) GetStringField() string { + if m != nil && m.StringField != nil { + return *m.StringField + } + return "" +} + +func (m *Strings) GetBytesField() []byte { + if m != nil { + return m.BytesField + } + return nil +} + +type Defaults struct { + // Default-valued fields of all basic types. + // Same as GoTest, but copied here to make testing easier. + F_Bool *bool `protobuf:"varint,1,opt,def=1" json:"F_Bool,omitempty"` + F_Int32 *int32 `protobuf:"varint,2,opt,def=32" json:"F_Int32,omitempty"` + F_Int64 *int64 `protobuf:"varint,3,opt,def=64" json:"F_Int64,omitempty"` + F_Fixed32 *uint32 `protobuf:"fixed32,4,opt,def=320" json:"F_Fixed32,omitempty"` + F_Fixed64 *uint64 `protobuf:"fixed64,5,opt,def=640" json:"F_Fixed64,omitempty"` + F_Uint32 *uint32 `protobuf:"varint,6,opt,def=3200" json:"F_Uint32,omitempty"` + F_Uint64 *uint64 `protobuf:"varint,7,opt,def=6400" json:"F_Uint64,omitempty"` + F_Float *float32 `protobuf:"fixed32,8,opt,def=314159" json:"F_Float,omitempty"` + F_Double *float64 `protobuf:"fixed64,9,opt,def=271828" json:"F_Double,omitempty"` + F_String *string `protobuf:"bytes,10,opt,def=hello, \"world!\"\n" json:"F_String,omitempty"` + F_Bytes []byte `protobuf:"bytes,11,opt,def=Bignose" json:"F_Bytes,omitempty"` + F_Sint32 *int32 `protobuf:"zigzag32,12,opt,def=-32" json:"F_Sint32,omitempty"` + F_Sint64 *int64 `protobuf:"zigzag64,13,opt,def=-64" json:"F_Sint64,omitempty"` + F_Enum *Defaults_Color `protobuf:"varint,14,opt,enum=testdata.Defaults_Color,def=1" json:"F_Enum,omitempty"` + // More fields with crazy defaults. + F_Pinf *float32 `protobuf:"fixed32,15,opt,def=inf" json:"F_Pinf,omitempty"` + F_Ninf *float32 `protobuf:"fixed32,16,opt,def=-inf" json:"F_Ninf,omitempty"` + F_Nan *float32 `protobuf:"fixed32,17,opt,def=nan" json:"F_Nan,omitempty"` + // Sub-message. + Sub *SubDefaults `protobuf:"bytes,18,opt,name=sub" json:"sub,omitempty"` + // Redundant but explicit defaults. + StrZero *string `protobuf:"bytes,19,opt,name=str_zero,def=" json:"str_zero,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Defaults) Reset() { *m = Defaults{} } +func (m *Defaults) String() string { return proto.CompactTextString(m) } +func (*Defaults) ProtoMessage() {} + +const Default_Defaults_F_Bool bool = true +const Default_Defaults_F_Int32 int32 = 32 +const Default_Defaults_F_Int64 int64 = 64 +const Default_Defaults_F_Fixed32 uint32 = 320 +const Default_Defaults_F_Fixed64 uint64 = 640 +const Default_Defaults_F_Uint32 uint32 = 3200 +const Default_Defaults_F_Uint64 uint64 = 6400 +const Default_Defaults_F_Float float32 = 314159 +const Default_Defaults_F_Double float64 = 271828 +const Default_Defaults_F_String string = "hello, \"world!\"\n" + +var Default_Defaults_F_Bytes []byte = []byte("Bignose") + +const Default_Defaults_F_Sint32 int32 = -32 +const Default_Defaults_F_Sint64 int64 = -64 +const Default_Defaults_F_Enum Defaults_Color = Defaults_GREEN + +var Default_Defaults_F_Pinf float32 = float32(math.Inf(1)) +var Default_Defaults_F_Ninf float32 = float32(math.Inf(-1)) +var Default_Defaults_F_Nan float32 = float32(math.NaN()) + +func (m *Defaults) GetF_Bool() bool { + if m != nil && m.F_Bool != nil { + return *m.F_Bool + } + return Default_Defaults_F_Bool +} + +func (m *Defaults) GetF_Int32() int32 { + if m != nil && m.F_Int32 != nil { + return *m.F_Int32 + } + return Default_Defaults_F_Int32 +} + +func (m *Defaults) GetF_Int64() int64 { + if m != nil && m.F_Int64 != nil { + return *m.F_Int64 + } + return Default_Defaults_F_Int64 +} + +func (m *Defaults) GetF_Fixed32() uint32 { + if m != nil && m.F_Fixed32 != nil { + return *m.F_Fixed32 + } + return Default_Defaults_F_Fixed32 +} + +func (m *Defaults) GetF_Fixed64() uint64 { + if m != nil && m.F_Fixed64 != nil { + return *m.F_Fixed64 + } + return Default_Defaults_F_Fixed64 +} + +func (m *Defaults) GetF_Uint32() uint32 { + if m != nil && m.F_Uint32 != nil { + return *m.F_Uint32 + } + return Default_Defaults_F_Uint32 +} + +func (m *Defaults) GetF_Uint64() uint64 { + if m != nil && m.F_Uint64 != nil { + return *m.F_Uint64 + } + return Default_Defaults_F_Uint64 +} + +func (m *Defaults) GetF_Float() float32 { + if m != nil && m.F_Float != nil { + return *m.F_Float + } + return Default_Defaults_F_Float +} + +func (m *Defaults) GetF_Double() float64 { + if m != nil && m.F_Double != nil { + return *m.F_Double + } + return Default_Defaults_F_Double +} + +func (m *Defaults) GetF_String() string { + if m != nil && m.F_String != nil { + return *m.F_String + } + return Default_Defaults_F_String +} + +func (m *Defaults) GetF_Bytes() []byte { + if m != nil && m.F_Bytes != nil { + return m.F_Bytes + } + return append([]byte(nil), Default_Defaults_F_Bytes...) +} + +func (m *Defaults) GetF_Sint32() int32 { + if m != nil && m.F_Sint32 != nil { + return *m.F_Sint32 + } + return Default_Defaults_F_Sint32 +} + +func (m *Defaults) GetF_Sint64() int64 { + if m != nil && m.F_Sint64 != nil { + return *m.F_Sint64 + } + return Default_Defaults_F_Sint64 +} + +func (m *Defaults) GetF_Enum() Defaults_Color { + if m != nil && m.F_Enum != nil { + return *m.F_Enum + } + return Default_Defaults_F_Enum +} + +func (m *Defaults) GetF_Pinf() float32 { + if m != nil && m.F_Pinf != nil { + return *m.F_Pinf + } + return Default_Defaults_F_Pinf +} + +func (m *Defaults) GetF_Ninf() float32 { + if m != nil && m.F_Ninf != nil { + return *m.F_Ninf + } + return Default_Defaults_F_Ninf +} + +func (m *Defaults) GetF_Nan() float32 { + if m != nil && m.F_Nan != nil { + return *m.F_Nan + } + return Default_Defaults_F_Nan +} + +func (m *Defaults) GetSub() *SubDefaults { + if m != nil { + return m.Sub + } + return nil +} + +func (m *Defaults) GetStrZero() string { + if m != nil && m.StrZero != nil { + return *m.StrZero + } + return "" +} + +type SubDefaults struct { + N *int64 `protobuf:"varint,1,opt,name=n,def=7" json:"n,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *SubDefaults) Reset() { *m = SubDefaults{} } +func (m *SubDefaults) String() string { return proto.CompactTextString(m) } +func (*SubDefaults) ProtoMessage() {} + +const Default_SubDefaults_N int64 = 7 + +func (m *SubDefaults) GetN() int64 { + if m != nil && m.N != nil { + return *m.N + } + return Default_SubDefaults_N +} + +type RepeatedEnum struct { + Color []RepeatedEnum_Color `protobuf:"varint,1,rep,name=color,enum=testdata.RepeatedEnum_Color" json:"color,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *RepeatedEnum) Reset() { *m = RepeatedEnum{} } +func (m *RepeatedEnum) String() string { return proto.CompactTextString(m) } +func (*RepeatedEnum) ProtoMessage() {} + +func (m *RepeatedEnum) GetColor() []RepeatedEnum_Color { + if m != nil { + return m.Color + } + return nil +} + +type MoreRepeated struct { + Bools []bool `protobuf:"varint,1,rep,name=bools" json:"bools,omitempty"` + BoolsPacked []bool `protobuf:"varint,2,rep,packed,name=bools_packed" json:"bools_packed,omitempty"` + Ints []int32 `protobuf:"varint,3,rep,name=ints" json:"ints,omitempty"` + IntsPacked []int32 `protobuf:"varint,4,rep,packed,name=ints_packed" json:"ints_packed,omitempty"` + Int64SPacked []int64 `protobuf:"varint,7,rep,packed,name=int64s_packed" json:"int64s_packed,omitempty"` + Strings []string `protobuf:"bytes,5,rep,name=strings" json:"strings,omitempty"` + Fixeds []uint32 `protobuf:"fixed32,6,rep,name=fixeds" json:"fixeds,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MoreRepeated) Reset() { *m = MoreRepeated{} } +func (m *MoreRepeated) String() string { return proto.CompactTextString(m) } +func (*MoreRepeated) ProtoMessage() {} + +func (m *MoreRepeated) GetBools() []bool { + if m != nil { + return m.Bools + } + return nil +} + +func (m *MoreRepeated) GetBoolsPacked() []bool { + if m != nil { + return m.BoolsPacked + } + return nil +} + +func (m *MoreRepeated) GetInts() []int32 { + if m != nil { + return m.Ints + } + return nil +} + +func (m *MoreRepeated) GetIntsPacked() []int32 { + if m != nil { + return m.IntsPacked + } + return nil +} + +func (m *MoreRepeated) GetInt64SPacked() []int64 { + if m != nil { + return m.Int64SPacked + } + return nil +} + +func (m *MoreRepeated) GetStrings() []string { + if m != nil { + return m.Strings + } + return nil +} + +func (m *MoreRepeated) GetFixeds() []uint32 { + if m != nil { + return m.Fixeds + } + return nil +} + +type GroupOld struct { + G *GroupOld_G `protobuf:"group,101,opt" json:"g,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GroupOld) Reset() { *m = GroupOld{} } +func (m *GroupOld) String() string { return proto.CompactTextString(m) } +func (*GroupOld) ProtoMessage() {} + +func (m *GroupOld) GetG() *GroupOld_G { + if m != nil { + return m.G + } + return nil +} + +type GroupOld_G struct { + X *int32 `protobuf:"varint,2,opt,name=x" json:"x,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GroupOld_G) Reset() { *m = GroupOld_G{} } +func (m *GroupOld_G) String() string { return proto.CompactTextString(m) } +func (*GroupOld_G) ProtoMessage() {} + +func (m *GroupOld_G) GetX() int32 { + if m != nil && m.X != nil { + return *m.X + } + return 0 +} + +type GroupNew struct { + G *GroupNew_G `protobuf:"group,101,opt" json:"g,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GroupNew) Reset() { *m = GroupNew{} } +func (m *GroupNew) String() string { return proto.CompactTextString(m) } +func (*GroupNew) ProtoMessage() {} + +func (m *GroupNew) GetG() *GroupNew_G { + if m != nil { + return m.G + } + return nil +} + +type GroupNew_G struct { + X *int32 `protobuf:"varint,2,opt,name=x" json:"x,omitempty"` + Y *int32 `protobuf:"varint,3,opt,name=y" json:"y,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GroupNew_G) Reset() { *m = GroupNew_G{} } +func (m *GroupNew_G) String() string { return proto.CompactTextString(m) } +func (*GroupNew_G) ProtoMessage() {} + +func (m *GroupNew_G) GetX() int32 { + if m != nil && m.X != nil { + return *m.X + } + return 0 +} + +func (m *GroupNew_G) GetY() int32 { + if m != nil && m.Y != nil { + return *m.Y + } + return 0 +} + +type FloatingPoint struct { + F *float64 `protobuf:"fixed64,1,req,name=f" json:"f,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *FloatingPoint) Reset() { *m = FloatingPoint{} } +func (m *FloatingPoint) String() string { return proto.CompactTextString(m) } +func (*FloatingPoint) ProtoMessage() {} + +func (m *FloatingPoint) GetF() float64 { + if m != nil && m.F != nil { + return *m.F + } + return 0 +} + +type MessageWithMap struct { + NameMapping map[int32]string `protobuf:"bytes,1,rep,name=name_mapping" json:"name_mapping,omitempty" protobuf_key:"varint,1,opt,name=key" protobuf_val:"bytes,2,opt,name=value"` + MsgMapping map[int64]*FloatingPoint `protobuf:"bytes,2,rep,name=msg_mapping" json:"msg_mapping,omitempty" protobuf_key:"zigzag64,1,opt,name=key" protobuf_val:"bytes,2,opt,name=value"` + ByteMapping map[bool][]byte `protobuf:"bytes,3,rep,name=byte_mapping" json:"byte_mapping,omitempty" protobuf_key:"varint,1,opt,name=key" protobuf_val:"bytes,2,opt,name=value"` + StrToStr map[string]string `protobuf:"bytes,4,rep,name=str_to_str" json:"str_to_str,omitempty" protobuf_key:"bytes,1,opt,name=key" protobuf_val:"bytes,2,opt,name=value"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MessageWithMap) Reset() { *m = MessageWithMap{} } +func (m *MessageWithMap) String() string { return proto.CompactTextString(m) } +func (*MessageWithMap) ProtoMessage() {} + +func (m *MessageWithMap) GetNameMapping() map[int32]string { + if m != nil { + return m.NameMapping + } + return nil +} + +func (m *MessageWithMap) GetMsgMapping() map[int64]*FloatingPoint { + if m != nil { + return m.MsgMapping + } + return nil +} + +func (m *MessageWithMap) GetByteMapping() map[bool][]byte { + if m != nil { + return m.ByteMapping + } + return nil +} + +func (m *MessageWithMap) GetStrToStr() map[string]string { + if m != nil { + return m.StrToStr + } + return nil +} + +var E_Greeting = &proto.ExtensionDesc{ + ExtendedType: (*MyMessage)(nil), + ExtensionType: ([]string)(nil), + Field: 106, + Name: "testdata.greeting", + Tag: "bytes,106,rep,name=greeting", +} + +var E_NoDefaultDouble = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*float64)(nil), + Field: 101, + Name: "testdata.no_default_double", + Tag: "fixed64,101,opt,name=no_default_double", +} + +var E_NoDefaultFloat = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*float32)(nil), + Field: 102, + Name: "testdata.no_default_float", + Tag: "fixed32,102,opt,name=no_default_float", +} + +var E_NoDefaultInt32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 103, + Name: "testdata.no_default_int32", + Tag: "varint,103,opt,name=no_default_int32", +} + +var E_NoDefaultInt64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int64)(nil), + Field: 104, + Name: "testdata.no_default_int64", + Tag: "varint,104,opt,name=no_default_int64", +} + +var E_NoDefaultUint32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint32)(nil), + Field: 105, + Name: "testdata.no_default_uint32", + Tag: "varint,105,opt,name=no_default_uint32", +} + +var E_NoDefaultUint64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint64)(nil), + Field: 106, + Name: "testdata.no_default_uint64", + Tag: "varint,106,opt,name=no_default_uint64", +} + +var E_NoDefaultSint32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 107, + Name: "testdata.no_default_sint32", + Tag: "zigzag32,107,opt,name=no_default_sint32", +} + +var E_NoDefaultSint64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int64)(nil), + Field: 108, + Name: "testdata.no_default_sint64", + Tag: "zigzag64,108,opt,name=no_default_sint64", +} + +var E_NoDefaultFixed32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint32)(nil), + Field: 109, + Name: "testdata.no_default_fixed32", + Tag: "fixed32,109,opt,name=no_default_fixed32", +} + +var E_NoDefaultFixed64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint64)(nil), + Field: 110, + Name: "testdata.no_default_fixed64", + Tag: "fixed64,110,opt,name=no_default_fixed64", +} + +var E_NoDefaultSfixed32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 111, + Name: "testdata.no_default_sfixed32", + Tag: "fixed32,111,opt,name=no_default_sfixed32", +} + +var E_NoDefaultSfixed64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int64)(nil), + Field: 112, + Name: "testdata.no_default_sfixed64", + Tag: "fixed64,112,opt,name=no_default_sfixed64", +} + +var E_NoDefaultBool = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*bool)(nil), + Field: 113, + Name: "testdata.no_default_bool", + Tag: "varint,113,opt,name=no_default_bool", +} + +var E_NoDefaultString = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*string)(nil), + Field: 114, + Name: "testdata.no_default_string", + Tag: "bytes,114,opt,name=no_default_string", +} + +var E_NoDefaultBytes = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: ([]byte)(nil), + Field: 115, + Name: "testdata.no_default_bytes", + Tag: "bytes,115,opt,name=no_default_bytes", +} + +var E_NoDefaultEnum = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*DefaultsMessage_DefaultsEnum)(nil), + Field: 116, + Name: "testdata.no_default_enum", + Tag: "varint,116,opt,name=no_default_enum,enum=testdata.DefaultsMessage_DefaultsEnum", +} + +var E_DefaultDouble = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*float64)(nil), + Field: 201, + Name: "testdata.default_double", + Tag: "fixed64,201,opt,name=default_double,def=3.1415", +} + +var E_DefaultFloat = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*float32)(nil), + Field: 202, + Name: "testdata.default_float", + Tag: "fixed32,202,opt,name=default_float,def=3.14", +} + +var E_DefaultInt32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 203, + Name: "testdata.default_int32", + Tag: "varint,203,opt,name=default_int32,def=42", +} + +var E_DefaultInt64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int64)(nil), + Field: 204, + Name: "testdata.default_int64", + Tag: "varint,204,opt,name=default_int64,def=43", +} + +var E_DefaultUint32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint32)(nil), + Field: 205, + Name: "testdata.default_uint32", + Tag: "varint,205,opt,name=default_uint32,def=44", +} + +var E_DefaultUint64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint64)(nil), + Field: 206, + Name: "testdata.default_uint64", + Tag: "varint,206,opt,name=default_uint64,def=45", +} + +var E_DefaultSint32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 207, + Name: "testdata.default_sint32", + Tag: "zigzag32,207,opt,name=default_sint32,def=46", +} + +var E_DefaultSint64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int64)(nil), + Field: 208, + Name: "testdata.default_sint64", + Tag: "zigzag64,208,opt,name=default_sint64,def=47", +} + +var E_DefaultFixed32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint32)(nil), + Field: 209, + Name: "testdata.default_fixed32", + Tag: "fixed32,209,opt,name=default_fixed32,def=48", +} + +var E_DefaultFixed64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint64)(nil), + Field: 210, + Name: "testdata.default_fixed64", + Tag: "fixed64,210,opt,name=default_fixed64,def=49", +} + +var E_DefaultSfixed32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 211, + Name: "testdata.default_sfixed32", + Tag: "fixed32,211,opt,name=default_sfixed32,def=50", +} + +var E_DefaultSfixed64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int64)(nil), + Field: 212, + Name: "testdata.default_sfixed64", + Tag: "fixed64,212,opt,name=default_sfixed64,def=51", +} + +var E_DefaultBool = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*bool)(nil), + Field: 213, + Name: "testdata.default_bool", + Tag: "varint,213,opt,name=default_bool,def=1", +} + +var E_DefaultString = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*string)(nil), + Field: 214, + Name: "testdata.default_string", + Tag: "bytes,214,opt,name=default_string,def=Hello, string", +} + +var E_DefaultBytes = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: ([]byte)(nil), + Field: 215, + Name: "testdata.default_bytes", + Tag: "bytes,215,opt,name=default_bytes,def=Hello, bytes", +} + +var E_DefaultEnum = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*DefaultsMessage_DefaultsEnum)(nil), + Field: 216, + Name: "testdata.default_enum", + Tag: "varint,216,opt,name=default_enum,enum=testdata.DefaultsMessage_DefaultsEnum,def=1", +} + +var E_X201 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 201, + Name: "testdata.x201", + Tag: "bytes,201,opt,name=x201", +} + +var E_X202 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 202, + Name: "testdata.x202", + Tag: "bytes,202,opt,name=x202", +} + +var E_X203 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 203, + Name: "testdata.x203", + Tag: "bytes,203,opt,name=x203", +} + +var E_X204 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 204, + Name: "testdata.x204", + Tag: "bytes,204,opt,name=x204", +} + +var E_X205 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 205, + Name: "testdata.x205", + Tag: "bytes,205,opt,name=x205", +} + +var E_X206 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 206, + Name: "testdata.x206", + Tag: "bytes,206,opt,name=x206", +} + +var E_X207 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 207, + Name: "testdata.x207", + Tag: "bytes,207,opt,name=x207", +} + +var E_X208 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 208, + Name: "testdata.x208", + Tag: "bytes,208,opt,name=x208", +} + +var E_X209 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 209, + Name: "testdata.x209", + Tag: "bytes,209,opt,name=x209", +} + +var E_X210 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 210, + Name: "testdata.x210", + Tag: "bytes,210,opt,name=x210", +} + +var E_X211 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 211, + Name: "testdata.x211", + Tag: "bytes,211,opt,name=x211", +} + +var E_X212 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 212, + Name: "testdata.x212", + Tag: "bytes,212,opt,name=x212", +} + +var E_X213 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 213, + Name: "testdata.x213", + Tag: "bytes,213,opt,name=x213", +} + +var E_X214 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 214, + Name: "testdata.x214", + Tag: "bytes,214,opt,name=x214", +} + +var E_X215 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 215, + Name: "testdata.x215", + Tag: "bytes,215,opt,name=x215", +} + +var E_X216 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 216, + Name: "testdata.x216", + Tag: "bytes,216,opt,name=x216", +} + +var E_X217 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 217, + Name: "testdata.x217", + Tag: "bytes,217,opt,name=x217", +} + +var E_X218 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 218, + Name: "testdata.x218", + Tag: "bytes,218,opt,name=x218", +} + +var E_X219 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 219, + Name: "testdata.x219", + Tag: "bytes,219,opt,name=x219", +} + +var E_X220 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 220, + Name: "testdata.x220", + Tag: "bytes,220,opt,name=x220", +} + +var E_X221 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 221, + Name: "testdata.x221", + Tag: "bytes,221,opt,name=x221", +} + +var E_X222 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 222, + Name: "testdata.x222", + Tag: "bytes,222,opt,name=x222", +} + +var E_X223 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 223, + Name: "testdata.x223", + Tag: "bytes,223,opt,name=x223", +} + +var E_X224 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 224, + Name: "testdata.x224", + Tag: "bytes,224,opt,name=x224", +} + +var E_X225 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 225, + Name: "testdata.x225", + Tag: "bytes,225,opt,name=x225", +} + +var E_X226 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 226, + Name: "testdata.x226", + Tag: "bytes,226,opt,name=x226", +} + +var E_X227 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 227, + Name: "testdata.x227", + Tag: "bytes,227,opt,name=x227", +} + +var E_X228 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 228, + Name: "testdata.x228", + Tag: "bytes,228,opt,name=x228", +} + +var E_X229 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 229, + Name: "testdata.x229", + Tag: "bytes,229,opt,name=x229", +} + +var E_X230 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 230, + Name: "testdata.x230", + Tag: "bytes,230,opt,name=x230", +} + +var E_X231 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 231, + Name: "testdata.x231", + Tag: "bytes,231,opt,name=x231", +} + +var E_X232 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 232, + Name: "testdata.x232", + Tag: "bytes,232,opt,name=x232", +} + +var E_X233 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 233, + Name: "testdata.x233", + Tag: "bytes,233,opt,name=x233", +} + +var E_X234 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 234, + Name: "testdata.x234", + Tag: "bytes,234,opt,name=x234", +} + +var E_X235 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 235, + Name: "testdata.x235", + Tag: "bytes,235,opt,name=x235", +} + +var E_X236 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 236, + Name: "testdata.x236", + Tag: "bytes,236,opt,name=x236", +} + +var E_X237 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 237, + Name: "testdata.x237", + Tag: "bytes,237,opt,name=x237", +} + +var E_X238 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 238, + Name: "testdata.x238", + Tag: "bytes,238,opt,name=x238", +} + +var E_X239 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 239, + Name: "testdata.x239", + Tag: "bytes,239,opt,name=x239", +} + +var E_X240 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 240, + Name: "testdata.x240", + Tag: "bytes,240,opt,name=x240", +} + +var E_X241 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 241, + Name: "testdata.x241", + Tag: "bytes,241,opt,name=x241", +} + +var E_X242 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 242, + Name: "testdata.x242", + Tag: "bytes,242,opt,name=x242", +} + +var E_X243 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 243, + Name: "testdata.x243", + Tag: "bytes,243,opt,name=x243", +} + +var E_X244 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 244, + Name: "testdata.x244", + Tag: "bytes,244,opt,name=x244", +} + +var E_X245 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 245, + Name: "testdata.x245", + Tag: "bytes,245,opt,name=x245", +} + +var E_X246 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 246, + Name: "testdata.x246", + Tag: "bytes,246,opt,name=x246", +} + +var E_X247 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 247, + Name: "testdata.x247", + Tag: "bytes,247,opt,name=x247", +} + +var E_X248 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 248, + Name: "testdata.x248", + Tag: "bytes,248,opt,name=x248", +} + +var E_X249 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 249, + Name: "testdata.x249", + Tag: "bytes,249,opt,name=x249", +} + +var E_X250 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 250, + Name: "testdata.x250", + Tag: "bytes,250,opt,name=x250", +} + +func init() { + proto.RegisterEnum("testdata.FOO", FOO_name, FOO_value) + proto.RegisterEnum("testdata.GoTest_KIND", GoTest_KIND_name, GoTest_KIND_value) + proto.RegisterEnum("testdata.MyMessage_Color", MyMessage_Color_name, MyMessage_Color_value) + proto.RegisterEnum("testdata.DefaultsMessage_DefaultsEnum", DefaultsMessage_DefaultsEnum_name, DefaultsMessage_DefaultsEnum_value) + proto.RegisterEnum("testdata.Defaults_Color", Defaults_Color_name, Defaults_Color_value) + proto.RegisterEnum("testdata.RepeatedEnum_Color", RepeatedEnum_Color_name, RepeatedEnum_Color_value) + proto.RegisterExtension(E_Ext_More) + proto.RegisterExtension(E_Ext_Text) + proto.RegisterExtension(E_Ext_Number) + proto.RegisterExtension(E_Greeting) + proto.RegisterExtension(E_NoDefaultDouble) + proto.RegisterExtension(E_NoDefaultFloat) + proto.RegisterExtension(E_NoDefaultInt32) + proto.RegisterExtension(E_NoDefaultInt64) + proto.RegisterExtension(E_NoDefaultUint32) + proto.RegisterExtension(E_NoDefaultUint64) + proto.RegisterExtension(E_NoDefaultSint32) + proto.RegisterExtension(E_NoDefaultSint64) + proto.RegisterExtension(E_NoDefaultFixed32) + proto.RegisterExtension(E_NoDefaultFixed64) + proto.RegisterExtension(E_NoDefaultSfixed32) + proto.RegisterExtension(E_NoDefaultSfixed64) + proto.RegisterExtension(E_NoDefaultBool) + proto.RegisterExtension(E_NoDefaultString) + proto.RegisterExtension(E_NoDefaultBytes) + proto.RegisterExtension(E_NoDefaultEnum) + proto.RegisterExtension(E_DefaultDouble) + proto.RegisterExtension(E_DefaultFloat) + proto.RegisterExtension(E_DefaultInt32) + proto.RegisterExtension(E_DefaultInt64) + proto.RegisterExtension(E_DefaultUint32) + proto.RegisterExtension(E_DefaultUint64) + proto.RegisterExtension(E_DefaultSint32) + proto.RegisterExtension(E_DefaultSint64) + proto.RegisterExtension(E_DefaultFixed32) + proto.RegisterExtension(E_DefaultFixed64) + proto.RegisterExtension(E_DefaultSfixed32) + proto.RegisterExtension(E_DefaultSfixed64) + proto.RegisterExtension(E_DefaultBool) + proto.RegisterExtension(E_DefaultString) + proto.RegisterExtension(E_DefaultBytes) + proto.RegisterExtension(E_DefaultEnum) + proto.RegisterExtension(E_X201) + proto.RegisterExtension(E_X202) + proto.RegisterExtension(E_X203) + proto.RegisterExtension(E_X204) + proto.RegisterExtension(E_X205) + proto.RegisterExtension(E_X206) + proto.RegisterExtension(E_X207) + proto.RegisterExtension(E_X208) + proto.RegisterExtension(E_X209) + proto.RegisterExtension(E_X210) + proto.RegisterExtension(E_X211) + proto.RegisterExtension(E_X212) + proto.RegisterExtension(E_X213) + proto.RegisterExtension(E_X214) + proto.RegisterExtension(E_X215) + proto.RegisterExtension(E_X216) + proto.RegisterExtension(E_X217) + proto.RegisterExtension(E_X218) + proto.RegisterExtension(E_X219) + proto.RegisterExtension(E_X220) + proto.RegisterExtension(E_X221) + proto.RegisterExtension(E_X222) + proto.RegisterExtension(E_X223) + proto.RegisterExtension(E_X224) + proto.RegisterExtension(E_X225) + proto.RegisterExtension(E_X226) + proto.RegisterExtension(E_X227) + proto.RegisterExtension(E_X228) + proto.RegisterExtension(E_X229) + proto.RegisterExtension(E_X230) + proto.RegisterExtension(E_X231) + proto.RegisterExtension(E_X232) + proto.RegisterExtension(E_X233) + proto.RegisterExtension(E_X234) + proto.RegisterExtension(E_X235) + proto.RegisterExtension(E_X236) + proto.RegisterExtension(E_X237) + proto.RegisterExtension(E_X238) + proto.RegisterExtension(E_X239) + proto.RegisterExtension(E_X240) + proto.RegisterExtension(E_X241) + proto.RegisterExtension(E_X242) + proto.RegisterExtension(E_X243) + proto.RegisterExtension(E_X244) + proto.RegisterExtension(E_X245) + proto.RegisterExtension(E_X246) + proto.RegisterExtension(E_X247) + proto.RegisterExtension(E_X248) + proto.RegisterExtension(E_X249) + proto.RegisterExtension(E_X250) +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/test.pb.go.golden b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/test.pb.go.golden new file mode 100644 index 0000000000000..0387853d56c61 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/test.pb.go.golden @@ -0,0 +1,1737 @@ +// Code generated by protoc-gen-gogo. +// source: test.proto +// DO NOT EDIT! + +package testdata + +import proto "github.com/gogo/protobuf/proto" +import json "encoding/json" +import math "math" + +import () + +// Reference proto, json, and math imports to suppress error if they are not otherwise used. +var _ = proto.Marshal +var _ = &json.SyntaxError{} +var _ = math.Inf + +type FOO int32 + +const ( + FOO_FOO1 FOO = 1 +) + +var FOO_name = map[int32]string{ + 1: "FOO1", +} +var FOO_value = map[string]int32{ + "FOO1": 1, +} + +func (x FOO) Enum() *FOO { + p := new(FOO) + *p = x + return p +} +func (x FOO) String() string { + return proto.EnumName(FOO_name, int32(x)) +} +func (x FOO) MarshalJSON() ([]byte, error) { + return json.Marshal(x.String()) +} +func (x *FOO) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(FOO_value, data, "FOO") + if err != nil { + return err + } + *x = FOO(value) + return nil +} + +type GoTest_KIND int32 + +const ( + GoTest_VOID GoTest_KIND = 0 + GoTest_BOOL GoTest_KIND = 1 + GoTest_BYTES GoTest_KIND = 2 + GoTest_FINGERPRINT GoTest_KIND = 3 + GoTest_FLOAT GoTest_KIND = 4 + GoTest_INT GoTest_KIND = 5 + GoTest_STRING GoTest_KIND = 6 + GoTest_TIME GoTest_KIND = 7 + GoTest_TUPLE GoTest_KIND = 8 + GoTest_ARRAY GoTest_KIND = 9 + GoTest_MAP GoTest_KIND = 10 + GoTest_TABLE GoTest_KIND = 11 + GoTest_FUNCTION GoTest_KIND = 12 +) + +var GoTest_KIND_name = map[int32]string{ + 0: "VOID", + 1: "BOOL", + 2: "BYTES", + 3: "FINGERPRINT", + 4: "FLOAT", + 5: "INT", + 6: "STRING", + 7: "TIME", + 8: "TUPLE", + 9: "ARRAY", + 10: "MAP", + 11: "TABLE", + 12: "FUNCTION", +} +var GoTest_KIND_value = map[string]int32{ + "VOID": 0, + "BOOL": 1, + "BYTES": 2, + "FINGERPRINT": 3, + "FLOAT": 4, + "INT": 5, + "STRING": 6, + "TIME": 7, + "TUPLE": 8, + "ARRAY": 9, + "MAP": 10, + "TABLE": 11, + "FUNCTION": 12, +} + +func (x GoTest_KIND) Enum() *GoTest_KIND { + p := new(GoTest_KIND) + *p = x + return p +} +func (x GoTest_KIND) String() string { + return proto.EnumName(GoTest_KIND_name, int32(x)) +} +func (x GoTest_KIND) MarshalJSON() ([]byte, error) { + return json.Marshal(x.String()) +} +func (x *GoTest_KIND) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(GoTest_KIND_value, data, "GoTest_KIND") + if err != nil { + return err + } + *x = GoTest_KIND(value) + return nil +} + +type MyMessage_Color int32 + +const ( + MyMessage_RED MyMessage_Color = 0 + MyMessage_GREEN MyMessage_Color = 1 + MyMessage_BLUE MyMessage_Color = 2 +) + +var MyMessage_Color_name = map[int32]string{ + 0: "RED", + 1: "GREEN", + 2: "BLUE", +} +var MyMessage_Color_value = map[string]int32{ + "RED": 0, + "GREEN": 1, + "BLUE": 2, +} + +func (x MyMessage_Color) Enum() *MyMessage_Color { + p := new(MyMessage_Color) + *p = x + return p +} +func (x MyMessage_Color) String() string { + return proto.EnumName(MyMessage_Color_name, int32(x)) +} +func (x MyMessage_Color) MarshalJSON() ([]byte, error) { + return json.Marshal(x.String()) +} +func (x *MyMessage_Color) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(MyMessage_Color_value, data, "MyMessage_Color") + if err != nil { + return err + } + *x = MyMessage_Color(value) + return nil +} + +type Defaults_Color int32 + +const ( + Defaults_RED Defaults_Color = 0 + Defaults_GREEN Defaults_Color = 1 + Defaults_BLUE Defaults_Color = 2 +) + +var Defaults_Color_name = map[int32]string{ + 0: "RED", + 1: "GREEN", + 2: "BLUE", +} +var Defaults_Color_value = map[string]int32{ + "RED": 0, + "GREEN": 1, + "BLUE": 2, +} + +func (x Defaults_Color) Enum() *Defaults_Color { + p := new(Defaults_Color) + *p = x + return p +} +func (x Defaults_Color) String() string { + return proto.EnumName(Defaults_Color_name, int32(x)) +} +func (x Defaults_Color) MarshalJSON() ([]byte, error) { + return json.Marshal(x.String()) +} +func (x *Defaults_Color) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(Defaults_Color_value, data, "Defaults_Color") + if err != nil { + return err + } + *x = Defaults_Color(value) + return nil +} + +type RepeatedEnum_Color int32 + +const ( + RepeatedEnum_RED RepeatedEnum_Color = 1 +) + +var RepeatedEnum_Color_name = map[int32]string{ + 1: "RED", +} +var RepeatedEnum_Color_value = map[string]int32{ + "RED": 1, +} + +func (x RepeatedEnum_Color) Enum() *RepeatedEnum_Color { + p := new(RepeatedEnum_Color) + *p = x + return p +} +func (x RepeatedEnum_Color) String() string { + return proto.EnumName(RepeatedEnum_Color_name, int32(x)) +} +func (x RepeatedEnum_Color) MarshalJSON() ([]byte, error) { + return json.Marshal(x.String()) +} +func (x *RepeatedEnum_Color) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(RepeatedEnum_Color_value, data, "RepeatedEnum_Color") + if err != nil { + return err + } + *x = RepeatedEnum_Color(value) + return nil +} + +type GoEnum struct { + Foo *FOO `protobuf:"varint,1,req,name=foo,enum=testdata.FOO" json:"foo,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoEnum) Reset() { *m = GoEnum{} } +func (m *GoEnum) String() string { return proto.CompactTextString(m) } +func (*GoEnum) ProtoMessage() {} + +func (m *GoEnum) GetFoo() FOO { + if m != nil && m.Foo != nil { + return *m.Foo + } + return 0 +} + +type GoTestField struct { + Label *string `protobuf:"bytes,1,req" json:"Label,omitempty"` + Type *string `protobuf:"bytes,2,req" json:"Type,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTestField) Reset() { *m = GoTestField{} } +func (m *GoTestField) String() string { return proto.CompactTextString(m) } +func (*GoTestField) ProtoMessage() {} + +func (m *GoTestField) GetLabel() string { + if m != nil && m.Label != nil { + return *m.Label + } + return "" +} + +func (m *GoTestField) GetType() string { + if m != nil && m.Type != nil { + return *m.Type + } + return "" +} + +type GoTest struct { + Kind *GoTest_KIND `protobuf:"varint,1,req,enum=testdata.GoTest_KIND" json:"Kind,omitempty"` + Table *string `protobuf:"bytes,2,opt" json:"Table,omitempty"` + Param *int32 `protobuf:"varint,3,opt" json:"Param,omitempty"` + RequiredField *GoTestField `protobuf:"bytes,4,req" json:"RequiredField,omitempty"` + RepeatedField []*GoTestField `protobuf:"bytes,5,rep" json:"RepeatedField,omitempty"` + OptionalField *GoTestField `protobuf:"bytes,6,opt" json:"OptionalField,omitempty"` + F_BoolRequired *bool `protobuf:"varint,10,req,name=F_Bool_required" json:"F_Bool_required,omitempty"` + F_Int32Required *int32 `protobuf:"varint,11,req,name=F_Int32_required" json:"F_Int32_required,omitempty"` + F_Int64Required *int64 `protobuf:"varint,12,req,name=F_Int64_required" json:"F_Int64_required,omitempty"` + F_Fixed32Required *uint32 `protobuf:"fixed32,13,req,name=F_Fixed32_required" json:"F_Fixed32_required,omitempty"` + F_Fixed64Required *uint64 `protobuf:"fixed64,14,req,name=F_Fixed64_required" json:"F_Fixed64_required,omitempty"` + F_Uint32Required *uint32 `protobuf:"varint,15,req,name=F_Uint32_required" json:"F_Uint32_required,omitempty"` + F_Uint64Required *uint64 `protobuf:"varint,16,req,name=F_Uint64_required" json:"F_Uint64_required,omitempty"` + F_FloatRequired *float32 `protobuf:"fixed32,17,req,name=F_Float_required" json:"F_Float_required,omitempty"` + F_DoubleRequired *float64 `protobuf:"fixed64,18,req,name=F_Double_required" json:"F_Double_required,omitempty"` + F_StringRequired *string `protobuf:"bytes,19,req,name=F_String_required" json:"F_String_required,omitempty"` + F_BytesRequired []byte `protobuf:"bytes,101,req,name=F_Bytes_required" json:"F_Bytes_required,omitempty"` + F_Sint32Required *int32 `protobuf:"zigzag32,102,req,name=F_Sint32_required" json:"F_Sint32_required,omitempty"` + F_Sint64Required *int64 `protobuf:"zigzag64,103,req,name=F_Sint64_required" json:"F_Sint64_required,omitempty"` + F_BoolRepeated []bool `protobuf:"varint,20,rep,name=F_Bool_repeated" json:"F_Bool_repeated,omitempty"` + F_Int32Repeated []int32 `protobuf:"varint,21,rep,name=F_Int32_repeated" json:"F_Int32_repeated,omitempty"` + F_Int64Repeated []int64 `protobuf:"varint,22,rep,name=F_Int64_repeated" json:"F_Int64_repeated,omitempty"` + F_Fixed32Repeated []uint32 `protobuf:"fixed32,23,rep,name=F_Fixed32_repeated" json:"F_Fixed32_repeated,omitempty"` + F_Fixed64Repeated []uint64 `protobuf:"fixed64,24,rep,name=F_Fixed64_repeated" json:"F_Fixed64_repeated,omitempty"` + F_Uint32Repeated []uint32 `protobuf:"varint,25,rep,name=F_Uint32_repeated" json:"F_Uint32_repeated,omitempty"` + F_Uint64Repeated []uint64 `protobuf:"varint,26,rep,name=F_Uint64_repeated" json:"F_Uint64_repeated,omitempty"` + F_FloatRepeated []float32 `protobuf:"fixed32,27,rep,name=F_Float_repeated" json:"F_Float_repeated,omitempty"` + F_DoubleRepeated []float64 `protobuf:"fixed64,28,rep,name=F_Double_repeated" json:"F_Double_repeated,omitempty"` + F_StringRepeated []string `protobuf:"bytes,29,rep,name=F_String_repeated" json:"F_String_repeated,omitempty"` + F_BytesRepeated [][]byte `protobuf:"bytes,201,rep,name=F_Bytes_repeated" json:"F_Bytes_repeated,omitempty"` + F_Sint32Repeated []int32 `protobuf:"zigzag32,202,rep,name=F_Sint32_repeated" json:"F_Sint32_repeated,omitempty"` + F_Sint64Repeated []int64 `protobuf:"zigzag64,203,rep,name=F_Sint64_repeated" json:"F_Sint64_repeated,omitempty"` + F_BoolOptional *bool `protobuf:"varint,30,opt,name=F_Bool_optional" json:"F_Bool_optional,omitempty"` + F_Int32Optional *int32 `protobuf:"varint,31,opt,name=F_Int32_optional" json:"F_Int32_optional,omitempty"` + F_Int64Optional *int64 `protobuf:"varint,32,opt,name=F_Int64_optional" json:"F_Int64_optional,omitempty"` + F_Fixed32Optional *uint32 `protobuf:"fixed32,33,opt,name=F_Fixed32_optional" json:"F_Fixed32_optional,omitempty"` + F_Fixed64Optional *uint64 `protobuf:"fixed64,34,opt,name=F_Fixed64_optional" json:"F_Fixed64_optional,omitempty"` + F_Uint32Optional *uint32 `protobuf:"varint,35,opt,name=F_Uint32_optional" json:"F_Uint32_optional,omitempty"` + F_Uint64Optional *uint64 `protobuf:"varint,36,opt,name=F_Uint64_optional" json:"F_Uint64_optional,omitempty"` + F_FloatOptional *float32 `protobuf:"fixed32,37,opt,name=F_Float_optional" json:"F_Float_optional,omitempty"` + F_DoubleOptional *float64 `protobuf:"fixed64,38,opt,name=F_Double_optional" json:"F_Double_optional,omitempty"` + F_StringOptional *string `protobuf:"bytes,39,opt,name=F_String_optional" json:"F_String_optional,omitempty"` + F_BytesOptional []byte `protobuf:"bytes,301,opt,name=F_Bytes_optional" json:"F_Bytes_optional,omitempty"` + F_Sint32Optional *int32 `protobuf:"zigzag32,302,opt,name=F_Sint32_optional" json:"F_Sint32_optional,omitempty"` + F_Sint64Optional *int64 `protobuf:"zigzag64,303,opt,name=F_Sint64_optional" json:"F_Sint64_optional,omitempty"` + F_BoolDefaulted *bool `protobuf:"varint,40,opt,name=F_Bool_defaulted,def=1" json:"F_Bool_defaulted,omitempty"` + F_Int32Defaulted *int32 `protobuf:"varint,41,opt,name=F_Int32_defaulted,def=32" json:"F_Int32_defaulted,omitempty"` + F_Int64Defaulted *int64 `protobuf:"varint,42,opt,name=F_Int64_defaulted,def=64" json:"F_Int64_defaulted,omitempty"` + F_Fixed32Defaulted *uint32 `protobuf:"fixed32,43,opt,name=F_Fixed32_defaulted,def=320" json:"F_Fixed32_defaulted,omitempty"` + F_Fixed64Defaulted *uint64 `protobuf:"fixed64,44,opt,name=F_Fixed64_defaulted,def=640" json:"F_Fixed64_defaulted,omitempty"` + F_Uint32Defaulted *uint32 `protobuf:"varint,45,opt,name=F_Uint32_defaulted,def=3200" json:"F_Uint32_defaulted,omitempty"` + F_Uint64Defaulted *uint64 `protobuf:"varint,46,opt,name=F_Uint64_defaulted,def=6400" json:"F_Uint64_defaulted,omitempty"` + F_FloatDefaulted *float32 `protobuf:"fixed32,47,opt,name=F_Float_defaulted,def=314159" json:"F_Float_defaulted,omitempty"` + F_DoubleDefaulted *float64 `protobuf:"fixed64,48,opt,name=F_Double_defaulted,def=271828" json:"F_Double_defaulted,omitempty"` + F_StringDefaulted *string `protobuf:"bytes,49,opt,name=F_String_defaulted,def=hello, \"world!\"\n" json:"F_String_defaulted,omitempty"` + F_BytesDefaulted []byte `protobuf:"bytes,401,opt,name=F_Bytes_defaulted,def=Bignose" json:"F_Bytes_defaulted,omitempty"` + F_Sint32Defaulted *int32 `protobuf:"zigzag32,402,opt,name=F_Sint32_defaulted,def=-32" json:"F_Sint32_defaulted,omitempty"` + F_Sint64Defaulted *int64 `protobuf:"zigzag64,403,opt,name=F_Sint64_defaulted,def=-64" json:"F_Sint64_defaulted,omitempty"` + F_BoolRepeatedPacked []bool `protobuf:"varint,50,rep,packed,name=F_Bool_repeated_packed" json:"F_Bool_repeated_packed,omitempty"` + F_Int32RepeatedPacked []int32 `protobuf:"varint,51,rep,packed,name=F_Int32_repeated_packed" json:"F_Int32_repeated_packed,omitempty"` + F_Int64RepeatedPacked []int64 `protobuf:"varint,52,rep,packed,name=F_Int64_repeated_packed" json:"F_Int64_repeated_packed,omitempty"` + F_Fixed32RepeatedPacked []uint32 `protobuf:"fixed32,53,rep,packed,name=F_Fixed32_repeated_packed" json:"F_Fixed32_repeated_packed,omitempty"` + F_Fixed64RepeatedPacked []uint64 `protobuf:"fixed64,54,rep,packed,name=F_Fixed64_repeated_packed" json:"F_Fixed64_repeated_packed,omitempty"` + F_Uint32RepeatedPacked []uint32 `protobuf:"varint,55,rep,packed,name=F_Uint32_repeated_packed" json:"F_Uint32_repeated_packed,omitempty"` + F_Uint64RepeatedPacked []uint64 `protobuf:"varint,56,rep,packed,name=F_Uint64_repeated_packed" json:"F_Uint64_repeated_packed,omitempty"` + F_FloatRepeatedPacked []float32 `protobuf:"fixed32,57,rep,packed,name=F_Float_repeated_packed" json:"F_Float_repeated_packed,omitempty"` + F_DoubleRepeatedPacked []float64 `protobuf:"fixed64,58,rep,packed,name=F_Double_repeated_packed" json:"F_Double_repeated_packed,omitempty"` + F_Sint32RepeatedPacked []int32 `protobuf:"zigzag32,502,rep,packed,name=F_Sint32_repeated_packed" json:"F_Sint32_repeated_packed,omitempty"` + F_Sint64RepeatedPacked []int64 `protobuf:"zigzag64,503,rep,packed,name=F_Sint64_repeated_packed" json:"F_Sint64_repeated_packed,omitempty"` + Requiredgroup *GoTest_RequiredGroup `protobuf:"group,70,req,name=RequiredGroup" json:"requiredgroup,omitempty"` + Repeatedgroup []*GoTest_RepeatedGroup `protobuf:"group,80,rep,name=RepeatedGroup" json:"repeatedgroup,omitempty"` + Optionalgroup *GoTest_OptionalGroup `protobuf:"group,90,opt,name=OptionalGroup" json:"optionalgroup,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTest) Reset() { *m = GoTest{} } +func (m *GoTest) String() string { return proto.CompactTextString(m) } +func (*GoTest) ProtoMessage() {} + +const Default_GoTest_F_BoolDefaulted bool = true +const Default_GoTest_F_Int32Defaulted int32 = 32 +const Default_GoTest_F_Int64Defaulted int64 = 64 +const Default_GoTest_F_Fixed32Defaulted uint32 = 320 +const Default_GoTest_F_Fixed64Defaulted uint64 = 640 +const Default_GoTest_F_Uint32Defaulted uint32 = 3200 +const Default_GoTest_F_Uint64Defaulted uint64 = 6400 +const Default_GoTest_F_FloatDefaulted float32 = 314159 +const Default_GoTest_F_DoubleDefaulted float64 = 271828 +const Default_GoTest_F_StringDefaulted string = "hello, \"world!\"\n" + +var Default_GoTest_F_BytesDefaulted []byte = []byte("Bignose") + +const Default_GoTest_F_Sint32Defaulted int32 = -32 +const Default_GoTest_F_Sint64Defaulted int64 = -64 + +func (m *GoTest) GetKind() GoTest_KIND { + if m != nil && m.Kind != nil { + return *m.Kind + } + return 0 +} + +func (m *GoTest) GetTable() string { + if m != nil && m.Table != nil { + return *m.Table + } + return "" +} + +func (m *GoTest) GetParam() int32 { + if m != nil && m.Param != nil { + return *m.Param + } + return 0 +} + +func (m *GoTest) GetRequiredField() *GoTestField { + if m != nil { + return m.RequiredField + } + return nil +} + +func (m *GoTest) GetRepeatedField() []*GoTestField { + if m != nil { + return m.RepeatedField + } + return nil +} + +func (m *GoTest) GetOptionalField() *GoTestField { + if m != nil { + return m.OptionalField + } + return nil +} + +func (m *GoTest) GetF_BoolRequired() bool { + if m != nil && m.F_BoolRequired != nil { + return *m.F_BoolRequired + } + return false +} + +func (m *GoTest) GetF_Int32Required() int32 { + if m != nil && m.F_Int32Required != nil { + return *m.F_Int32Required + } + return 0 +} + +func (m *GoTest) GetF_Int64Required() int64 { + if m != nil && m.F_Int64Required != nil { + return *m.F_Int64Required + } + return 0 +} + +func (m *GoTest) GetF_Fixed32Required() uint32 { + if m != nil && m.F_Fixed32Required != nil { + return *m.F_Fixed32Required + } + return 0 +} + +func (m *GoTest) GetF_Fixed64Required() uint64 { + if m != nil && m.F_Fixed64Required != nil { + return *m.F_Fixed64Required + } + return 0 +} + +func (m *GoTest) GetF_Uint32Required() uint32 { + if m != nil && m.F_Uint32Required != nil { + return *m.F_Uint32Required + } + return 0 +} + +func (m *GoTest) GetF_Uint64Required() uint64 { + if m != nil && m.F_Uint64Required != nil { + return *m.F_Uint64Required + } + return 0 +} + +func (m *GoTest) GetF_FloatRequired() float32 { + if m != nil && m.F_FloatRequired != nil { + return *m.F_FloatRequired + } + return 0 +} + +func (m *GoTest) GetF_DoubleRequired() float64 { + if m != nil && m.F_DoubleRequired != nil { + return *m.F_DoubleRequired + } + return 0 +} + +func (m *GoTest) GetF_StringRequired() string { + if m != nil && m.F_StringRequired != nil { + return *m.F_StringRequired + } + return "" +} + +func (m *GoTest) GetF_BytesRequired() []byte { + if m != nil { + return m.F_BytesRequired + } + return nil +} + +func (m *GoTest) GetF_Sint32Required() int32 { + if m != nil && m.F_Sint32Required != nil { + return *m.F_Sint32Required + } + return 0 +} + +func (m *GoTest) GetF_Sint64Required() int64 { + if m != nil && m.F_Sint64Required != nil { + return *m.F_Sint64Required + } + return 0 +} + +func (m *GoTest) GetF_BoolRepeated() []bool { + if m != nil { + return m.F_BoolRepeated + } + return nil +} + +func (m *GoTest) GetF_Int32Repeated() []int32 { + if m != nil { + return m.F_Int32Repeated + } + return nil +} + +func (m *GoTest) GetF_Int64Repeated() []int64 { + if m != nil { + return m.F_Int64Repeated + } + return nil +} + +func (m *GoTest) GetF_Fixed32Repeated() []uint32 { + if m != nil { + return m.F_Fixed32Repeated + } + return nil +} + +func (m *GoTest) GetF_Fixed64Repeated() []uint64 { + if m != nil { + return m.F_Fixed64Repeated + } + return nil +} + +func (m *GoTest) GetF_Uint32Repeated() []uint32 { + if m != nil { + return m.F_Uint32Repeated + } + return nil +} + +func (m *GoTest) GetF_Uint64Repeated() []uint64 { + if m != nil { + return m.F_Uint64Repeated + } + return nil +} + +func (m *GoTest) GetF_FloatRepeated() []float32 { + if m != nil { + return m.F_FloatRepeated + } + return nil +} + +func (m *GoTest) GetF_DoubleRepeated() []float64 { + if m != nil { + return m.F_DoubleRepeated + } + return nil +} + +func (m *GoTest) GetF_StringRepeated() []string { + if m != nil { + return m.F_StringRepeated + } + return nil +} + +func (m *GoTest) GetF_BytesRepeated() [][]byte { + if m != nil { + return m.F_BytesRepeated + } + return nil +} + +func (m *GoTest) GetF_Sint32Repeated() []int32 { + if m != nil { + return m.F_Sint32Repeated + } + return nil +} + +func (m *GoTest) GetF_Sint64Repeated() []int64 { + if m != nil { + return m.F_Sint64Repeated + } + return nil +} + +func (m *GoTest) GetF_BoolOptional() bool { + if m != nil && m.F_BoolOptional != nil { + return *m.F_BoolOptional + } + return false +} + +func (m *GoTest) GetF_Int32Optional() int32 { + if m != nil && m.F_Int32Optional != nil { + return *m.F_Int32Optional + } + return 0 +} + +func (m *GoTest) GetF_Int64Optional() int64 { + if m != nil && m.F_Int64Optional != nil { + return *m.F_Int64Optional + } + return 0 +} + +func (m *GoTest) GetF_Fixed32Optional() uint32 { + if m != nil && m.F_Fixed32Optional != nil { + return *m.F_Fixed32Optional + } + return 0 +} + +func (m *GoTest) GetF_Fixed64Optional() uint64 { + if m != nil && m.F_Fixed64Optional != nil { + return *m.F_Fixed64Optional + } + return 0 +} + +func (m *GoTest) GetF_Uint32Optional() uint32 { + if m != nil && m.F_Uint32Optional != nil { + return *m.F_Uint32Optional + } + return 0 +} + +func (m *GoTest) GetF_Uint64Optional() uint64 { + if m != nil && m.F_Uint64Optional != nil { + return *m.F_Uint64Optional + } + return 0 +} + +func (m *GoTest) GetF_FloatOptional() float32 { + if m != nil && m.F_FloatOptional != nil { + return *m.F_FloatOptional + } + return 0 +} + +func (m *GoTest) GetF_DoubleOptional() float64 { + if m != nil && m.F_DoubleOptional != nil { + return *m.F_DoubleOptional + } + return 0 +} + +func (m *GoTest) GetF_StringOptional() string { + if m != nil && m.F_StringOptional != nil { + return *m.F_StringOptional + } + return "" +} + +func (m *GoTest) GetF_BytesOptional() []byte { + if m != nil { + return m.F_BytesOptional + } + return nil +} + +func (m *GoTest) GetF_Sint32Optional() int32 { + if m != nil && m.F_Sint32Optional != nil { + return *m.F_Sint32Optional + } + return 0 +} + +func (m *GoTest) GetF_Sint64Optional() int64 { + if m != nil && m.F_Sint64Optional != nil { + return *m.F_Sint64Optional + } + return 0 +} + +func (m *GoTest) GetF_BoolDefaulted() bool { + if m != nil && m.F_BoolDefaulted != nil { + return *m.F_BoolDefaulted + } + return Default_GoTest_F_BoolDefaulted +} + +func (m *GoTest) GetF_Int32Defaulted() int32 { + if m != nil && m.F_Int32Defaulted != nil { + return *m.F_Int32Defaulted + } + return Default_GoTest_F_Int32Defaulted +} + +func (m *GoTest) GetF_Int64Defaulted() int64 { + if m != nil && m.F_Int64Defaulted != nil { + return *m.F_Int64Defaulted + } + return Default_GoTest_F_Int64Defaulted +} + +func (m *GoTest) GetF_Fixed32Defaulted() uint32 { + if m != nil && m.F_Fixed32Defaulted != nil { + return *m.F_Fixed32Defaulted + } + return Default_GoTest_F_Fixed32Defaulted +} + +func (m *GoTest) GetF_Fixed64Defaulted() uint64 { + if m != nil && m.F_Fixed64Defaulted != nil { + return *m.F_Fixed64Defaulted + } + return Default_GoTest_F_Fixed64Defaulted +} + +func (m *GoTest) GetF_Uint32Defaulted() uint32 { + if m != nil && m.F_Uint32Defaulted != nil { + return *m.F_Uint32Defaulted + } + return Default_GoTest_F_Uint32Defaulted +} + +func (m *GoTest) GetF_Uint64Defaulted() uint64 { + if m != nil && m.F_Uint64Defaulted != nil { + return *m.F_Uint64Defaulted + } + return Default_GoTest_F_Uint64Defaulted +} + +func (m *GoTest) GetF_FloatDefaulted() float32 { + if m != nil && m.F_FloatDefaulted != nil { + return *m.F_FloatDefaulted + } + return Default_GoTest_F_FloatDefaulted +} + +func (m *GoTest) GetF_DoubleDefaulted() float64 { + if m != nil && m.F_DoubleDefaulted != nil { + return *m.F_DoubleDefaulted + } + return Default_GoTest_F_DoubleDefaulted +} + +func (m *GoTest) GetF_StringDefaulted() string { + if m != nil && m.F_StringDefaulted != nil { + return *m.F_StringDefaulted + } + return Default_GoTest_F_StringDefaulted +} + +func (m *GoTest) GetF_BytesDefaulted() []byte { + if m != nil && m.F_BytesDefaulted != nil { + return m.F_BytesDefaulted + } + return append([]byte(nil), Default_GoTest_F_BytesDefaulted...) +} + +func (m *GoTest) GetF_Sint32Defaulted() int32 { + if m != nil && m.F_Sint32Defaulted != nil { + return *m.F_Sint32Defaulted + } + return Default_GoTest_F_Sint32Defaulted +} + +func (m *GoTest) GetF_Sint64Defaulted() int64 { + if m != nil && m.F_Sint64Defaulted != nil { + return *m.F_Sint64Defaulted + } + return Default_GoTest_F_Sint64Defaulted +} + +func (m *GoTest) GetF_BoolRepeatedPacked() []bool { + if m != nil { + return m.F_BoolRepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Int32RepeatedPacked() []int32 { + if m != nil { + return m.F_Int32RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Int64RepeatedPacked() []int64 { + if m != nil { + return m.F_Int64RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Fixed32RepeatedPacked() []uint32 { + if m != nil { + return m.F_Fixed32RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Fixed64RepeatedPacked() []uint64 { + if m != nil { + return m.F_Fixed64RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Uint32RepeatedPacked() []uint32 { + if m != nil { + return m.F_Uint32RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Uint64RepeatedPacked() []uint64 { + if m != nil { + return m.F_Uint64RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_FloatRepeatedPacked() []float32 { + if m != nil { + return m.F_FloatRepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_DoubleRepeatedPacked() []float64 { + if m != nil { + return m.F_DoubleRepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Sint32RepeatedPacked() []int32 { + if m != nil { + return m.F_Sint32RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Sint64RepeatedPacked() []int64 { + if m != nil { + return m.F_Sint64RepeatedPacked + } + return nil +} + +func (m *GoTest) GetRequiredgroup() *GoTest_RequiredGroup { + if m != nil { + return m.Requiredgroup + } + return nil +} + +func (m *GoTest) GetRepeatedgroup() []*GoTest_RepeatedGroup { + if m != nil { + return m.Repeatedgroup + } + return nil +} + +func (m *GoTest) GetOptionalgroup() *GoTest_OptionalGroup { + if m != nil { + return m.Optionalgroup + } + return nil +} + +type GoTest_RequiredGroup struct { + RequiredField *string `protobuf:"bytes,71,req" json:"RequiredField,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTest_RequiredGroup) Reset() { *m = GoTest_RequiredGroup{} } + +func (m *GoTest_RequiredGroup) GetRequiredField() string { + if m != nil && m.RequiredField != nil { + return *m.RequiredField + } + return "" +} + +type GoTest_RepeatedGroup struct { + RequiredField *string `protobuf:"bytes,81,req" json:"RequiredField,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTest_RepeatedGroup) Reset() { *m = GoTest_RepeatedGroup{} } + +func (m *GoTest_RepeatedGroup) GetRequiredField() string { + if m != nil && m.RequiredField != nil { + return *m.RequiredField + } + return "" +} + +type GoTest_OptionalGroup struct { + RequiredField *string `protobuf:"bytes,91,req" json:"RequiredField,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTest_OptionalGroup) Reset() { *m = GoTest_OptionalGroup{} } + +func (m *GoTest_OptionalGroup) GetRequiredField() string { + if m != nil && m.RequiredField != nil { + return *m.RequiredField + } + return "" +} + +type GoSkipTest struct { + SkipInt32 *int32 `protobuf:"varint,11,req,name=skip_int32" json:"skip_int32,omitempty"` + SkipFixed32 *uint32 `protobuf:"fixed32,12,req,name=skip_fixed32" json:"skip_fixed32,omitempty"` + SkipFixed64 *uint64 `protobuf:"fixed64,13,req,name=skip_fixed64" json:"skip_fixed64,omitempty"` + SkipString *string `protobuf:"bytes,14,req,name=skip_string" json:"skip_string,omitempty"` + Skipgroup *GoSkipTest_SkipGroup `protobuf:"group,15,req,name=SkipGroup" json:"skipgroup,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoSkipTest) Reset() { *m = GoSkipTest{} } +func (m *GoSkipTest) String() string { return proto.CompactTextString(m) } +func (*GoSkipTest) ProtoMessage() {} + +func (m *GoSkipTest) GetSkipInt32() int32 { + if m != nil && m.SkipInt32 != nil { + return *m.SkipInt32 + } + return 0 +} + +func (m *GoSkipTest) GetSkipFixed32() uint32 { + if m != nil && m.SkipFixed32 != nil { + return *m.SkipFixed32 + } + return 0 +} + +func (m *GoSkipTest) GetSkipFixed64() uint64 { + if m != nil && m.SkipFixed64 != nil { + return *m.SkipFixed64 + } + return 0 +} + +func (m *GoSkipTest) GetSkipString() string { + if m != nil && m.SkipString != nil { + return *m.SkipString + } + return "" +} + +func (m *GoSkipTest) GetSkipgroup() *GoSkipTest_SkipGroup { + if m != nil { + return m.Skipgroup + } + return nil +} + +type GoSkipTest_SkipGroup struct { + GroupInt32 *int32 `protobuf:"varint,16,req,name=group_int32" json:"group_int32,omitempty"` + GroupString *string `protobuf:"bytes,17,req,name=group_string" json:"group_string,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoSkipTest_SkipGroup) Reset() { *m = GoSkipTest_SkipGroup{} } + +func (m *GoSkipTest_SkipGroup) GetGroupInt32() int32 { + if m != nil && m.GroupInt32 != nil { + return *m.GroupInt32 + } + return 0 +} + +func (m *GoSkipTest_SkipGroup) GetGroupString() string { + if m != nil && m.GroupString != nil { + return *m.GroupString + } + return "" +} + +type NonPackedTest struct { + A []int32 `protobuf:"varint,1,rep,name=a" json:"a,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *NonPackedTest) Reset() { *m = NonPackedTest{} } +func (m *NonPackedTest) String() string { return proto.CompactTextString(m) } +func (*NonPackedTest) ProtoMessage() {} + +func (m *NonPackedTest) GetA() []int32 { + if m != nil { + return m.A + } + return nil +} + +type PackedTest struct { + B []int32 `protobuf:"varint,1,rep,packed,name=b" json:"b,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *PackedTest) Reset() { *m = PackedTest{} } +func (m *PackedTest) String() string { return proto.CompactTextString(m) } +func (*PackedTest) ProtoMessage() {} + +func (m *PackedTest) GetB() []int32 { + if m != nil { + return m.B + } + return nil +} + +type MaxTag struct { + LastField *string `protobuf:"bytes,536870911,opt,name=last_field" json:"last_field,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MaxTag) Reset() { *m = MaxTag{} } +func (m *MaxTag) String() string { return proto.CompactTextString(m) } +func (*MaxTag) ProtoMessage() {} + +func (m *MaxTag) GetLastField() string { + if m != nil && m.LastField != nil { + return *m.LastField + } + return "" +} + +type OldMessage struct { + Nested *OldMessage_Nested `protobuf:"bytes,1,opt,name=nested" json:"nested,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *OldMessage) Reset() { *m = OldMessage{} } +func (m *OldMessage) String() string { return proto.CompactTextString(m) } +func (*OldMessage) ProtoMessage() {} + +func (m *OldMessage) GetNested() *OldMessage_Nested { + if m != nil { + return m.Nested + } + return nil +} + +type OldMessage_Nested struct { + Name *string `protobuf:"bytes,1,opt,name=name" json:"name,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *OldMessage_Nested) Reset() { *m = OldMessage_Nested{} } +func (m *OldMessage_Nested) String() string { return proto.CompactTextString(m) } +func (*OldMessage_Nested) ProtoMessage() {} + +func (m *OldMessage_Nested) GetName() string { + if m != nil && m.Name != nil { + return *m.Name + } + return "" +} + +type NewMessage struct { + Nested *NewMessage_Nested `protobuf:"bytes,1,opt,name=nested" json:"nested,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *NewMessage) Reset() { *m = NewMessage{} } +func (m *NewMessage) String() string { return proto.CompactTextString(m) } +func (*NewMessage) ProtoMessage() {} + +func (m *NewMessage) GetNested() *NewMessage_Nested { + if m != nil { + return m.Nested + } + return nil +} + +type NewMessage_Nested struct { + Name *string `protobuf:"bytes,1,opt,name=name" json:"name,omitempty"` + FoodGroup *string `protobuf:"bytes,2,opt,name=food_group" json:"food_group,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *NewMessage_Nested) Reset() { *m = NewMessage_Nested{} } +func (m *NewMessage_Nested) String() string { return proto.CompactTextString(m) } +func (*NewMessage_Nested) ProtoMessage() {} + +func (m *NewMessage_Nested) GetName() string { + if m != nil && m.Name != nil { + return *m.Name + } + return "" +} + +func (m *NewMessage_Nested) GetFoodGroup() string { + if m != nil && m.FoodGroup != nil { + return *m.FoodGroup + } + return "" +} + +type InnerMessage struct { + Host *string `protobuf:"bytes,1,req,name=host" json:"host,omitempty"` + Port *int32 `protobuf:"varint,2,opt,name=port,def=4000" json:"port,omitempty"` + Connected *bool `protobuf:"varint,3,opt,name=connected" json:"connected,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *InnerMessage) Reset() { *m = InnerMessage{} } +func (m *InnerMessage) String() string { return proto.CompactTextString(m) } +func (*InnerMessage) ProtoMessage() {} + +const Default_InnerMessage_Port int32 = 4000 + +func (m *InnerMessage) GetHost() string { + if m != nil && m.Host != nil { + return *m.Host + } + return "" +} + +func (m *InnerMessage) GetPort() int32 { + if m != nil && m.Port != nil { + return *m.Port + } + return Default_InnerMessage_Port +} + +func (m *InnerMessage) GetConnected() bool { + if m != nil && m.Connected != nil { + return *m.Connected + } + return false +} + +type OtherMessage struct { + Key *int64 `protobuf:"varint,1,opt,name=key" json:"key,omitempty"` + Value []byte `protobuf:"bytes,2,opt,name=value" json:"value,omitempty"` + Weight *float32 `protobuf:"fixed32,3,opt,name=weight" json:"weight,omitempty"` + Inner *InnerMessage `protobuf:"bytes,4,opt,name=inner" json:"inner,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *OtherMessage) Reset() { *m = OtherMessage{} } +func (m *OtherMessage) String() string { return proto.CompactTextString(m) } +func (*OtherMessage) ProtoMessage() {} + +func (m *OtherMessage) GetKey() int64 { + if m != nil && m.Key != nil { + return *m.Key + } + return 0 +} + +func (m *OtherMessage) GetValue() []byte { + if m != nil { + return m.Value + } + return nil +} + +func (m *OtherMessage) GetWeight() float32 { + if m != nil && m.Weight != nil { + return *m.Weight + } + return 0 +} + +func (m *OtherMessage) GetInner() *InnerMessage { + if m != nil { + return m.Inner + } + return nil +} + +type MyMessage struct { + Count *int32 `protobuf:"varint,1,req,name=count" json:"count,omitempty"` + Name *string `protobuf:"bytes,2,opt,name=name" json:"name,omitempty"` + Quote *string `protobuf:"bytes,3,opt,name=quote" json:"quote,omitempty"` + Pet []string `protobuf:"bytes,4,rep,name=pet" json:"pet,omitempty"` + Inner *InnerMessage `protobuf:"bytes,5,opt,name=inner" json:"inner,omitempty"` + Others []*OtherMessage `protobuf:"bytes,6,rep,name=others" json:"others,omitempty"` + Bikeshed *MyMessage_Color `protobuf:"varint,7,opt,name=bikeshed,enum=testdata.MyMessage_Color" json:"bikeshed,omitempty"` + Somegroup *MyMessage_SomeGroup `protobuf:"group,8,opt,name=SomeGroup" json:"somegroup,omitempty"` + RepBytes [][]byte `protobuf:"bytes,10,rep,name=rep_bytes" json:"rep_bytes,omitempty"` + Bigfloat *float64 `protobuf:"fixed64,11,opt,name=bigfloat" json:"bigfloat,omitempty"` + XXX_extensions map[int32]proto.Extension `json:"-"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MyMessage) Reset() { *m = MyMessage{} } +func (m *MyMessage) String() string { return proto.CompactTextString(m) } +func (*MyMessage) ProtoMessage() {} + +var extRange_MyMessage = []proto.ExtensionRange{ + {100, 536870911}, +} + +func (*MyMessage) ExtensionRangeArray() []proto.ExtensionRange { + return extRange_MyMessage +} +func (m *MyMessage) ExtensionMap() map[int32]proto.Extension { + if m.XXX_extensions == nil { + m.XXX_extensions = make(map[int32]proto.Extension) + } + return m.XXX_extensions +} + +func (m *MyMessage) GetCount() int32 { + if m != nil && m.Count != nil { + return *m.Count + } + return 0 +} + +func (m *MyMessage) GetName() string { + if m != nil && m.Name != nil { + return *m.Name + } + return "" +} + +func (m *MyMessage) GetQuote() string { + if m != nil && m.Quote != nil { + return *m.Quote + } + return "" +} + +func (m *MyMessage) GetPet() []string { + if m != nil { + return m.Pet + } + return nil +} + +func (m *MyMessage) GetInner() *InnerMessage { + if m != nil { + return m.Inner + } + return nil +} + +func (m *MyMessage) GetOthers() []*OtherMessage { + if m != nil { + return m.Others + } + return nil +} + +func (m *MyMessage) GetBikeshed() MyMessage_Color { + if m != nil && m.Bikeshed != nil { + return *m.Bikeshed + } + return 0 +} + +func (m *MyMessage) GetSomegroup() *MyMessage_SomeGroup { + if m != nil { + return m.Somegroup + } + return nil +} + +func (m *MyMessage) GetRepBytes() [][]byte { + if m != nil { + return m.RepBytes + } + return nil +} + +func (m *MyMessage) GetBigfloat() float64 { + if m != nil && m.Bigfloat != nil { + return *m.Bigfloat + } + return 0 +} + +type MyMessage_SomeGroup struct { + GroupField *int32 `protobuf:"varint,9,opt,name=group_field" json:"group_field,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MyMessage_SomeGroup) Reset() { *m = MyMessage_SomeGroup{} } + +func (m *MyMessage_SomeGroup) GetGroupField() int32 { + if m != nil && m.GroupField != nil { + return *m.GroupField + } + return 0 +} + +type Ext struct { + Data *string `protobuf:"bytes,1,opt,name=data" json:"data,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Ext) Reset() { *m = Ext{} } +func (m *Ext) String() string { return proto.CompactTextString(m) } +func (*Ext) ProtoMessage() {} + +func (m *Ext) GetData() string { + if m != nil && m.Data != nil { + return *m.Data + } + return "" +} + +var E_Ext_More = &proto.ExtensionDesc{ + ExtendedType: (*MyMessage)(nil), + ExtensionType: (*Ext)(nil), + Field: 103, + Name: "testdata.Ext.more", + Tag: "bytes,103,opt,name=more", +} + +var E_Ext_Text = &proto.ExtensionDesc{ + ExtendedType: (*MyMessage)(nil), + ExtensionType: (*string)(nil), + Field: 104, + Name: "testdata.Ext.text", + Tag: "bytes,104,opt,name=text", +} + +var E_Ext_Number = &proto.ExtensionDesc{ + ExtendedType: (*MyMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 105, + Name: "testdata.Ext.number", + Tag: "varint,105,opt,name=number", +} + +type MessageList struct { + Message []*MessageList_Message `protobuf:"group,1,rep" json:"message,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MessageList) Reset() { *m = MessageList{} } +func (m *MessageList) String() string { return proto.CompactTextString(m) } +func (*MessageList) ProtoMessage() {} + +func (m *MessageList) GetMessage() []*MessageList_Message { + if m != nil { + return m.Message + } + return nil +} + +type MessageList_Message struct { + Name *string `protobuf:"bytes,2,req,name=name" json:"name,omitempty"` + Count *int32 `protobuf:"varint,3,req,name=count" json:"count,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MessageList_Message) Reset() { *m = MessageList_Message{} } + +func (m *MessageList_Message) GetName() string { + if m != nil && m.Name != nil { + return *m.Name + } + return "" +} + +func (m *MessageList_Message) GetCount() int32 { + if m != nil && m.Count != nil { + return *m.Count + } + return 0 +} + +type Strings struct { + StringField *string `protobuf:"bytes,1,opt,name=string_field" json:"string_field,omitempty"` + BytesField []byte `protobuf:"bytes,2,opt,name=bytes_field" json:"bytes_field,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Strings) Reset() { *m = Strings{} } +func (m *Strings) String() string { return proto.CompactTextString(m) } +func (*Strings) ProtoMessage() {} + +func (m *Strings) GetStringField() string { + if m != nil && m.StringField != nil { + return *m.StringField + } + return "" +} + +func (m *Strings) GetBytesField() []byte { + if m != nil { + return m.BytesField + } + return nil +} + +type Defaults struct { + F_Bool *bool `protobuf:"varint,1,opt,def=1" json:"F_Bool,omitempty"` + F_Int32 *int32 `protobuf:"varint,2,opt,def=32" json:"F_Int32,omitempty"` + F_Int64 *int64 `protobuf:"varint,3,opt,def=64" json:"F_Int64,omitempty"` + F_Fixed32 *uint32 `protobuf:"fixed32,4,opt,def=320" json:"F_Fixed32,omitempty"` + F_Fixed64 *uint64 `protobuf:"fixed64,5,opt,def=640" json:"F_Fixed64,omitempty"` + F_Uint32 *uint32 `protobuf:"varint,6,opt,def=3200" json:"F_Uint32,omitempty"` + F_Uint64 *uint64 `protobuf:"varint,7,opt,def=6400" json:"F_Uint64,omitempty"` + F_Float *float32 `protobuf:"fixed32,8,opt,def=314159" json:"F_Float,omitempty"` + F_Double *float64 `protobuf:"fixed64,9,opt,def=271828" json:"F_Double,omitempty"` + F_String *string `protobuf:"bytes,10,opt,def=hello, \"world!\"\n" json:"F_String,omitempty"` + F_Bytes []byte `protobuf:"bytes,11,opt,def=Bignose" json:"F_Bytes,omitempty"` + F_Sint32 *int32 `protobuf:"zigzag32,12,opt,def=-32" json:"F_Sint32,omitempty"` + F_Sint64 *int64 `protobuf:"zigzag64,13,opt,def=-64" json:"F_Sint64,omitempty"` + F_Enum *Defaults_Color `protobuf:"varint,14,opt,enum=testdata.Defaults_Color,def=1" json:"F_Enum,omitempty"` + F_Pinf *float32 `protobuf:"fixed32,15,opt,def=inf" json:"F_Pinf,omitempty"` + F_Ninf *float32 `protobuf:"fixed32,16,opt,def=-inf" json:"F_Ninf,omitempty"` + F_Nan *float32 `protobuf:"fixed32,17,opt,def=nan" json:"F_Nan,omitempty"` + Sub *SubDefaults `protobuf:"bytes,18,opt,name=sub" json:"sub,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Defaults) Reset() { *m = Defaults{} } +func (m *Defaults) String() string { return proto.CompactTextString(m) } +func (*Defaults) ProtoMessage() {} + +const Default_Defaults_F_Bool bool = true +const Default_Defaults_F_Int32 int32 = 32 +const Default_Defaults_F_Int64 int64 = 64 +const Default_Defaults_F_Fixed32 uint32 = 320 +const Default_Defaults_F_Fixed64 uint64 = 640 +const Default_Defaults_F_Uint32 uint32 = 3200 +const Default_Defaults_F_Uint64 uint64 = 6400 +const Default_Defaults_F_Float float32 = 314159 +const Default_Defaults_F_Double float64 = 271828 +const Default_Defaults_F_String string = "hello, \"world!\"\n" + +var Default_Defaults_F_Bytes []byte = []byte("Bignose") + +const Default_Defaults_F_Sint32 int32 = -32 +const Default_Defaults_F_Sint64 int64 = -64 +const Default_Defaults_F_Enum Defaults_Color = Defaults_GREEN + +var Default_Defaults_F_Pinf float32 = float32(math.Inf(1)) +var Default_Defaults_F_Ninf float32 = float32(math.Inf(-1)) +var Default_Defaults_F_Nan float32 = float32(math.NaN()) + +func (m *Defaults) GetF_Bool() bool { + if m != nil && m.F_Bool != nil { + return *m.F_Bool + } + return Default_Defaults_F_Bool +} + +func (m *Defaults) GetF_Int32() int32 { + if m != nil && m.F_Int32 != nil { + return *m.F_Int32 + } + return Default_Defaults_F_Int32 +} + +func (m *Defaults) GetF_Int64() int64 { + if m != nil && m.F_Int64 != nil { + return *m.F_Int64 + } + return Default_Defaults_F_Int64 +} + +func (m *Defaults) GetF_Fixed32() uint32 { + if m != nil && m.F_Fixed32 != nil { + return *m.F_Fixed32 + } + return Default_Defaults_F_Fixed32 +} + +func (m *Defaults) GetF_Fixed64() uint64 { + if m != nil && m.F_Fixed64 != nil { + return *m.F_Fixed64 + } + return Default_Defaults_F_Fixed64 +} + +func (m *Defaults) GetF_Uint32() uint32 { + if m != nil && m.F_Uint32 != nil { + return *m.F_Uint32 + } + return Default_Defaults_F_Uint32 +} + +func (m *Defaults) GetF_Uint64() uint64 { + if m != nil && m.F_Uint64 != nil { + return *m.F_Uint64 + } + return Default_Defaults_F_Uint64 +} + +func (m *Defaults) GetF_Float() float32 { + if m != nil && m.F_Float != nil { + return *m.F_Float + } + return Default_Defaults_F_Float +} + +func (m *Defaults) GetF_Double() float64 { + if m != nil && m.F_Double != nil { + return *m.F_Double + } + return Default_Defaults_F_Double +} + +func (m *Defaults) GetF_String() string { + if m != nil && m.F_String != nil { + return *m.F_String + } + return Default_Defaults_F_String +} + +func (m *Defaults) GetF_Bytes() []byte { + if m != nil && m.F_Bytes != nil { + return m.F_Bytes + } + return append([]byte(nil), Default_Defaults_F_Bytes...) +} + +func (m *Defaults) GetF_Sint32() int32 { + if m != nil && m.F_Sint32 != nil { + return *m.F_Sint32 + } + return Default_Defaults_F_Sint32 +} + +func (m *Defaults) GetF_Sint64() int64 { + if m != nil && m.F_Sint64 != nil { + return *m.F_Sint64 + } + return Default_Defaults_F_Sint64 +} + +func (m *Defaults) GetF_Enum() Defaults_Color { + if m != nil && m.F_Enum != nil { + return *m.F_Enum + } + return Default_Defaults_F_Enum +} + +func (m *Defaults) GetF_Pinf() float32 { + if m != nil && m.F_Pinf != nil { + return *m.F_Pinf + } + return Default_Defaults_F_Pinf +} + +func (m *Defaults) GetF_Ninf() float32 { + if m != nil && m.F_Ninf != nil { + return *m.F_Ninf + } + return Default_Defaults_F_Ninf +} + +func (m *Defaults) GetF_Nan() float32 { + if m != nil && m.F_Nan != nil { + return *m.F_Nan + } + return Default_Defaults_F_Nan +} + +func (m *Defaults) GetSub() *SubDefaults { + if m != nil { + return m.Sub + } + return nil +} + +type SubDefaults struct { + N *int64 `protobuf:"varint,1,opt,name=n,def=7" json:"n,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *SubDefaults) Reset() { *m = SubDefaults{} } +func (m *SubDefaults) String() string { return proto.CompactTextString(m) } +func (*SubDefaults) ProtoMessage() {} + +const Default_SubDefaults_N int64 = 7 + +func (m *SubDefaults) GetN() int64 { + if m != nil && m.N != nil { + return *m.N + } + return Default_SubDefaults_N +} + +type RepeatedEnum struct { + Color []RepeatedEnum_Color `protobuf:"varint,1,rep,name=color,enum=testdata.RepeatedEnum_Color" json:"color,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *RepeatedEnum) Reset() { *m = RepeatedEnum{} } +func (m *RepeatedEnum) String() string { return proto.CompactTextString(m) } +func (*RepeatedEnum) ProtoMessage() {} + +func (m *RepeatedEnum) GetColor() []RepeatedEnum_Color { + if m != nil { + return m.Color + } + return nil +} + +type MoreRepeated struct { + Bools []bool `protobuf:"varint,1,rep,name=bools" json:"bools,omitempty"` + BoolsPacked []bool `protobuf:"varint,2,rep,packed,name=bools_packed" json:"bools_packed,omitempty"` + Ints []int32 `protobuf:"varint,3,rep,name=ints" json:"ints,omitempty"` + IntsPacked []int32 `protobuf:"varint,4,rep,packed,name=ints_packed" json:"ints_packed,omitempty"` + Strings []string `protobuf:"bytes,5,rep,name=strings" json:"strings,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MoreRepeated) Reset() { *m = MoreRepeated{} } +func (m *MoreRepeated) String() string { return proto.CompactTextString(m) } +func (*MoreRepeated) ProtoMessage() {} + +func (m *MoreRepeated) GetBools() []bool { + if m != nil { + return m.Bools + } + return nil +} + +func (m *MoreRepeated) GetBoolsPacked() []bool { + if m != nil { + return m.BoolsPacked + } + return nil +} + +func (m *MoreRepeated) GetInts() []int32 { + if m != nil { + return m.Ints + } + return nil +} + +func (m *MoreRepeated) GetIntsPacked() []int32 { + if m != nil { + return m.IntsPacked + } + return nil +} + +func (m *MoreRepeated) GetStrings() []string { + if m != nil { + return m.Strings + } + return nil +} + +type GroupOld struct { + G *GroupOld_G `protobuf:"group,1,opt" json:"g,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GroupOld) Reset() { *m = GroupOld{} } +func (m *GroupOld) String() string { return proto.CompactTextString(m) } +func (*GroupOld) ProtoMessage() {} + +func (m *GroupOld) GetG() *GroupOld_G { + if m != nil { + return m.G + } + return nil +} + +type GroupOld_G struct { + X *int32 `protobuf:"varint,2,opt,name=x" json:"x,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GroupOld_G) Reset() { *m = GroupOld_G{} } + +func (m *GroupOld_G) GetX() int32 { + if m != nil && m.X != nil { + return *m.X + } + return 0 +} + +type GroupNew struct { + G *GroupNew_G `protobuf:"group,1,opt" json:"g,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GroupNew) Reset() { *m = GroupNew{} } +func (m *GroupNew) String() string { return proto.CompactTextString(m) } +func (*GroupNew) ProtoMessage() {} + +func (m *GroupNew) GetG() *GroupNew_G { + if m != nil { + return m.G + } + return nil +} + +type GroupNew_G struct { + X *int32 `protobuf:"varint,2,opt,name=x" json:"x,omitempty"` + Y *int32 `protobuf:"varint,3,opt,name=y" json:"y,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GroupNew_G) Reset() { *m = GroupNew_G{} } + +func (m *GroupNew_G) GetX() int32 { + if m != nil && m.X != nil { + return *m.X + } + return 0 +} + +func (m *GroupNew_G) GetY() int32 { + if m != nil && m.Y != nil { + return *m.Y + } + return 0 +} + +var E_Greeting = &proto.ExtensionDesc{ + ExtendedType: (*MyMessage)(nil), + ExtensionType: ([]string)(nil), + Field: 106, + Name: "testdata.greeting", + Tag: "bytes,106,rep,name=greeting", +} + +func init() { + proto.RegisterEnum("testdata.FOO", FOO_name, FOO_value) + proto.RegisterEnum("testdata.GoTest_KIND", GoTest_KIND_name, GoTest_KIND_value) + proto.RegisterEnum("testdata.MyMessage_Color", MyMessage_Color_name, MyMessage_Color_value) + proto.RegisterEnum("testdata.Defaults_Color", Defaults_Color_name, Defaults_Color_value) + proto.RegisterEnum("testdata.RepeatedEnum_Color", RepeatedEnum_Color_name, RepeatedEnum_Color_value) + proto.RegisterExtension(E_Ext_More) + proto.RegisterExtension(E_Ext_Text) + proto.RegisterExtension(E_Ext_Number) + proto.RegisterExtension(E_Greeting) +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/test.proto b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/test.proto new file mode 100644 index 0000000000000..440dba38dd3bd --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/testdata/test.proto @@ -0,0 +1,480 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// A feature-rich test file for the protocol compiler and libraries. + +syntax = "proto2"; + +package testdata; + +enum FOO { FOO1 = 1; }; + +message GoEnum { + required FOO foo = 1; +} + +message GoTestField { + required string Label = 1; + required string Type = 2; +} + +message GoTest { + // An enum, for completeness. + enum KIND { + VOID = 0; + + // Basic types + BOOL = 1; + BYTES = 2; + FINGERPRINT = 3; + FLOAT = 4; + INT = 5; + STRING = 6; + TIME = 7; + + // Groupings + TUPLE = 8; + ARRAY = 9; + MAP = 10; + + // Table types + TABLE = 11; + + // Functions + FUNCTION = 12; // last tag + }; + + // Some typical parameters + required KIND Kind = 1; + optional string Table = 2; + optional int32 Param = 3; + + // Required, repeated and optional foreign fields. + required GoTestField RequiredField = 4; + repeated GoTestField RepeatedField = 5; + optional GoTestField OptionalField = 6; + + // Required fields of all basic types + required bool F_Bool_required = 10; + required int32 F_Int32_required = 11; + required int64 F_Int64_required = 12; + required fixed32 F_Fixed32_required = 13; + required fixed64 F_Fixed64_required = 14; + required uint32 F_Uint32_required = 15; + required uint64 F_Uint64_required = 16; + required float F_Float_required = 17; + required double F_Double_required = 18; + required string F_String_required = 19; + required bytes F_Bytes_required = 101; + required sint32 F_Sint32_required = 102; + required sint64 F_Sint64_required = 103; + + // Repeated fields of all basic types + repeated bool F_Bool_repeated = 20; + repeated int32 F_Int32_repeated = 21; + repeated int64 F_Int64_repeated = 22; + repeated fixed32 F_Fixed32_repeated = 23; + repeated fixed64 F_Fixed64_repeated = 24; + repeated uint32 F_Uint32_repeated = 25; + repeated uint64 F_Uint64_repeated = 26; + repeated float F_Float_repeated = 27; + repeated double F_Double_repeated = 28; + repeated string F_String_repeated = 29; + repeated bytes F_Bytes_repeated = 201; + repeated sint32 F_Sint32_repeated = 202; + repeated sint64 F_Sint64_repeated = 203; + + // Optional fields of all basic types + optional bool F_Bool_optional = 30; + optional int32 F_Int32_optional = 31; + optional int64 F_Int64_optional = 32; + optional fixed32 F_Fixed32_optional = 33; + optional fixed64 F_Fixed64_optional = 34; + optional uint32 F_Uint32_optional = 35; + optional uint64 F_Uint64_optional = 36; + optional float F_Float_optional = 37; + optional double F_Double_optional = 38; + optional string F_String_optional = 39; + optional bytes F_Bytes_optional = 301; + optional sint32 F_Sint32_optional = 302; + optional sint64 F_Sint64_optional = 303; + + // Default-valued fields of all basic types + optional bool F_Bool_defaulted = 40 [default=true]; + optional int32 F_Int32_defaulted = 41 [default=32]; + optional int64 F_Int64_defaulted = 42 [default=64]; + optional fixed32 F_Fixed32_defaulted = 43 [default=320]; + optional fixed64 F_Fixed64_defaulted = 44 [default=640]; + optional uint32 F_Uint32_defaulted = 45 [default=3200]; + optional uint64 F_Uint64_defaulted = 46 [default=6400]; + optional float F_Float_defaulted = 47 [default=314159.]; + optional double F_Double_defaulted = 48 [default=271828.]; + optional string F_String_defaulted = 49 [default="hello, \"world!\"\n"]; + optional bytes F_Bytes_defaulted = 401 [default="Bignose"]; + optional sint32 F_Sint32_defaulted = 402 [default = -32]; + optional sint64 F_Sint64_defaulted = 403 [default = -64]; + + // Packed repeated fields (no string or bytes). + repeated bool F_Bool_repeated_packed = 50 [packed=true]; + repeated int32 F_Int32_repeated_packed = 51 [packed=true]; + repeated int64 F_Int64_repeated_packed = 52 [packed=true]; + repeated fixed32 F_Fixed32_repeated_packed = 53 [packed=true]; + repeated fixed64 F_Fixed64_repeated_packed = 54 [packed=true]; + repeated uint32 F_Uint32_repeated_packed = 55 [packed=true]; + repeated uint64 F_Uint64_repeated_packed = 56 [packed=true]; + repeated float F_Float_repeated_packed = 57 [packed=true]; + repeated double F_Double_repeated_packed = 58 [packed=true]; + repeated sint32 F_Sint32_repeated_packed = 502 [packed=true]; + repeated sint64 F_Sint64_repeated_packed = 503 [packed=true]; + + // Required, repeated, and optional groups. + required group RequiredGroup = 70 { + required string RequiredField = 71; + }; + + repeated group RepeatedGroup = 80 { + required string RequiredField = 81; + }; + + optional group OptionalGroup = 90 { + required string RequiredField = 91; + }; +} + +// For testing skipping of unrecognized fields. +// Numbers are all big, larger than tag numbers in GoTestField, +// the message used in the corresponding test. +message GoSkipTest { + required int32 skip_int32 = 11; + required fixed32 skip_fixed32 = 12; + required fixed64 skip_fixed64 = 13; + required string skip_string = 14; + required group SkipGroup = 15 { + required int32 group_int32 = 16; + required string group_string = 17; + } +} + +// For testing packed/non-packed decoder switching. +// A serialized instance of one should be deserializable as the other. +message NonPackedTest { + repeated int32 a = 1; +} + +message PackedTest { + repeated int32 b = 1 [packed=true]; +} + +message MaxTag { + // Maximum possible tag number. + optional string last_field = 536870911; +} + +message OldMessage { + message Nested { + optional string name = 1; + } + optional Nested nested = 1; + + optional int32 num = 2; +} + +// NewMessage is wire compatible with OldMessage; +// imagine it as a future version. +message NewMessage { + message Nested { + optional string name = 1; + optional string food_group = 2; + } + optional Nested nested = 1; + + // This is an int32 in OldMessage. + optional int64 num = 2; +} + +// Smaller tests for ASCII formatting. + +message InnerMessage { + required string host = 1; + optional int32 port = 2 [default=4000]; + optional bool connected = 3; +} + +message OtherMessage { + optional int64 key = 1; + optional bytes value = 2; + optional float weight = 3; + optional InnerMessage inner = 4; +} + +message MyMessage { + required int32 count = 1; + optional string name = 2; + optional string quote = 3; + repeated string pet = 4; + optional InnerMessage inner = 5; + repeated OtherMessage others = 6; + repeated InnerMessage rep_inner = 12; + + enum Color { + RED = 0; + GREEN = 1; + BLUE = 2; + }; + optional Color bikeshed = 7; + + optional group SomeGroup = 8 { + optional int32 group_field = 9; + } + + // This field becomes [][]byte in the generated code. + repeated bytes rep_bytes = 10; + + optional double bigfloat = 11; + + extensions 100 to max; +} + +message Ext { + extend MyMessage { + optional Ext more = 103; + optional string text = 104; + optional int32 number = 105; + } + + optional string data = 1; +} + +extend MyMessage { + repeated string greeting = 106; +} + +message DefaultsMessage { + enum DefaultsEnum { + ZERO = 0; + ONE = 1; + TWO = 2; + }; + extensions 100 to max; +} + +extend DefaultsMessage { + optional double no_default_double = 101; + optional float no_default_float = 102; + optional int32 no_default_int32 = 103; + optional int64 no_default_int64 = 104; + optional uint32 no_default_uint32 = 105; + optional uint64 no_default_uint64 = 106; + optional sint32 no_default_sint32 = 107; + optional sint64 no_default_sint64 = 108; + optional fixed32 no_default_fixed32 = 109; + optional fixed64 no_default_fixed64 = 110; + optional sfixed32 no_default_sfixed32 = 111; + optional sfixed64 no_default_sfixed64 = 112; + optional bool no_default_bool = 113; + optional string no_default_string = 114; + optional bytes no_default_bytes = 115; + optional DefaultsMessage.DefaultsEnum no_default_enum = 116; + + optional double default_double = 201 [default = 3.1415]; + optional float default_float = 202 [default = 3.14]; + optional int32 default_int32 = 203 [default = 42]; + optional int64 default_int64 = 204 [default = 43]; + optional uint32 default_uint32 = 205 [default = 44]; + optional uint64 default_uint64 = 206 [default = 45]; + optional sint32 default_sint32 = 207 [default = 46]; + optional sint64 default_sint64 = 208 [default = 47]; + optional fixed32 default_fixed32 = 209 [default = 48]; + optional fixed64 default_fixed64 = 210 [default = 49]; + optional sfixed32 default_sfixed32 = 211 [default = 50]; + optional sfixed64 default_sfixed64 = 212 [default = 51]; + optional bool default_bool = 213 [default = true]; + optional string default_string = 214 [default = "Hello, string"]; + optional bytes default_bytes = 215 [default = "Hello, bytes"]; + optional DefaultsMessage.DefaultsEnum default_enum = 216 [default = ONE]; +} + +message MyMessageSet { + option message_set_wire_format = true; + extensions 100 to max; +} + +message Empty { +} + +extend MyMessageSet { + optional Empty x201 = 201; + optional Empty x202 = 202; + optional Empty x203 = 203; + optional Empty x204 = 204; + optional Empty x205 = 205; + optional Empty x206 = 206; + optional Empty x207 = 207; + optional Empty x208 = 208; + optional Empty x209 = 209; + optional Empty x210 = 210; + optional Empty x211 = 211; + optional Empty x212 = 212; + optional Empty x213 = 213; + optional Empty x214 = 214; + optional Empty x215 = 215; + optional Empty x216 = 216; + optional Empty x217 = 217; + optional Empty x218 = 218; + optional Empty x219 = 219; + optional Empty x220 = 220; + optional Empty x221 = 221; + optional Empty x222 = 222; + optional Empty x223 = 223; + optional Empty x224 = 224; + optional Empty x225 = 225; + optional Empty x226 = 226; + optional Empty x227 = 227; + optional Empty x228 = 228; + optional Empty x229 = 229; + optional Empty x230 = 230; + optional Empty x231 = 231; + optional Empty x232 = 232; + optional Empty x233 = 233; + optional Empty x234 = 234; + optional Empty x235 = 235; + optional Empty x236 = 236; + optional Empty x237 = 237; + optional Empty x238 = 238; + optional Empty x239 = 239; + optional Empty x240 = 240; + optional Empty x241 = 241; + optional Empty x242 = 242; + optional Empty x243 = 243; + optional Empty x244 = 244; + optional Empty x245 = 245; + optional Empty x246 = 246; + optional Empty x247 = 247; + optional Empty x248 = 248; + optional Empty x249 = 249; + optional Empty x250 = 250; +} + +message MessageList { + repeated group Message = 1 { + required string name = 2; + required int32 count = 3; + } +} + +message Strings { + optional string string_field = 1; + optional bytes bytes_field = 2; +} + +message Defaults { + enum Color { + RED = 0; + GREEN = 1; + BLUE = 2; + } + + // Default-valued fields of all basic types. + // Same as GoTest, but copied here to make testing easier. + optional bool F_Bool = 1 [default=true]; + optional int32 F_Int32 = 2 [default=32]; + optional int64 F_Int64 = 3 [default=64]; + optional fixed32 F_Fixed32 = 4 [default=320]; + optional fixed64 F_Fixed64 = 5 [default=640]; + optional uint32 F_Uint32 = 6 [default=3200]; + optional uint64 F_Uint64 = 7 [default=6400]; + optional float F_Float = 8 [default=314159.]; + optional double F_Double = 9 [default=271828.]; + optional string F_String = 10 [default="hello, \"world!\"\n"]; + optional bytes F_Bytes = 11 [default="Bignose"]; + optional sint32 F_Sint32 = 12 [default=-32]; + optional sint64 F_Sint64 = 13 [default=-64]; + optional Color F_Enum = 14 [default=GREEN]; + + // More fields with crazy defaults. + optional float F_Pinf = 15 [default=inf]; + optional float F_Ninf = 16 [default=-inf]; + optional float F_Nan = 17 [default=nan]; + + // Sub-message. + optional SubDefaults sub = 18; + + // Redundant but explicit defaults. + optional string str_zero = 19 [default=""]; +} + +message SubDefaults { + optional int64 n = 1 [default=7]; +} + +message RepeatedEnum { + enum Color { + RED = 1; + } + repeated Color color = 1; +} + +message MoreRepeated { + repeated bool bools = 1; + repeated bool bools_packed = 2 [packed=true]; + repeated int32 ints = 3; + repeated int32 ints_packed = 4 [packed=true]; + repeated int64 int64s_packed = 7 [packed=true]; + repeated string strings = 5; + repeated fixed32 fixeds = 6; +} + +// GroupOld and GroupNew have the same wire format. +// GroupNew has a new field inside a group. + +message GroupOld { + optional group G = 101 { + optional int32 x = 2; + } +} + +message GroupNew { + optional group G = 101 { + optional int32 x = 2; + optional int32 y = 3; + } +} + +message FloatingPoint { + required double f = 1; +} + +message MessageWithMap { + map name_mapping = 1; + map msg_mapping = 2; + map byte_mapping = 3; + map str_to_str = 4; +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text.go new file mode 100644 index 0000000000000..36524244188d9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text.go @@ -0,0 +1,804 @@ +// Extensions for Protocol Buffers to create more go like structures. +// +// Copyright (c) 2013, Vastech SA (PTY) LTD. All rights reserved. +// http://github.com/gogo/protobuf/gogoproto +// +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +// Functions for writing the text protocol buffer format. + +import ( + "bufio" + "bytes" + "encoding" + "fmt" + "io" + "log" + "math" + "reflect" + "sort" + "strings" +) + +var ( + newline = []byte("\n") + spaces = []byte(" ") + gtNewline = []byte(">\n") + endBraceNewline = []byte("}\n") + backslashN = []byte{'\\', 'n'} + backslashR = []byte{'\\', 'r'} + backslashT = []byte{'\\', 't'} + backslashDQ = []byte{'\\', '"'} + backslashBS = []byte{'\\', '\\'} + posInf = []byte("inf") + negInf = []byte("-inf") + nan = []byte("nan") +) + +type writer interface { + io.Writer + WriteByte(byte) error +} + +// textWriter is an io.Writer that tracks its indentation level. +type textWriter struct { + ind int + complete bool // if the current position is a complete line + compact bool // whether to write out as a one-liner + w writer +} + +func (w *textWriter) WriteString(s string) (n int, err error) { + if !strings.Contains(s, "\n") { + if !w.compact && w.complete { + w.writeIndent() + } + w.complete = false + return io.WriteString(w.w, s) + } + // WriteString is typically called without newlines, so this + // codepath and its copy are rare. We copy to avoid + // duplicating all of Write's logic here. + return w.Write([]byte(s)) +} + +func (w *textWriter) Write(p []byte) (n int, err error) { + newlines := bytes.Count(p, newline) + if newlines == 0 { + if !w.compact && w.complete { + w.writeIndent() + } + n, err = w.w.Write(p) + w.complete = false + return n, err + } + + frags := bytes.SplitN(p, newline, newlines+1) + if w.compact { + for i, frag := range frags { + if i > 0 { + if err := w.w.WriteByte(' '); err != nil { + return n, err + } + n++ + } + nn, err := w.w.Write(frag) + n += nn + if err != nil { + return n, err + } + } + return n, nil + } + + for i, frag := range frags { + if w.complete { + w.writeIndent() + } + nn, err := w.w.Write(frag) + n += nn + if err != nil { + return n, err + } + if i+1 < len(frags) { + if err := w.w.WriteByte('\n'); err != nil { + return n, err + } + n++ + } + } + w.complete = len(frags[len(frags)-1]) == 0 + return n, nil +} + +func (w *textWriter) WriteByte(c byte) error { + if w.compact && c == '\n' { + c = ' ' + } + if !w.compact && w.complete { + w.writeIndent() + } + err := w.w.WriteByte(c) + w.complete = c == '\n' + return err +} + +func (w *textWriter) indent() { w.ind++ } + +func (w *textWriter) unindent() { + if w.ind == 0 { + log.Printf("proto: textWriter unindented too far") + return + } + w.ind-- +} + +func writeName(w *textWriter, props *Properties) error { + if _, err := w.WriteString(props.OrigName); err != nil { + return err + } + if props.Wire != "group" { + return w.WriteByte(':') + } + return nil +} + +var ( + messageSetType = reflect.TypeOf((*MessageSet)(nil)).Elem() +) + +// raw is the interface satisfied by RawMessage. +type raw interface { + Bytes() []byte +} + +func writeStruct(w *textWriter, sv reflect.Value) error { + if sv.Type() == messageSetType { + return writeMessageSet(w, sv.Addr().Interface().(*MessageSet)) + } + + st := sv.Type() + sprops := GetProperties(st) + for i := 0; i < sv.NumField(); i++ { + fv := sv.Field(i) + props := sprops.Prop[i] + name := st.Field(i).Name + + if strings.HasPrefix(name, "XXX_") { + // There are two XXX_ fields: + // XXX_unrecognized []byte + // XXX_extensions map[int32]proto.Extension + // The first is handled here; + // the second is handled at the bottom of this function. + if name == "XXX_unrecognized" && !fv.IsNil() { + if err := writeUnknownStruct(w, fv.Interface().([]byte)); err != nil { + return err + } + } + continue + } + if fv.Kind() == reflect.Ptr && fv.IsNil() { + // Field not filled in. This could be an optional field or + // a required field that wasn't filled in. Either way, there + // isn't anything we can show for it. + continue + } + if fv.Kind() == reflect.Slice && fv.IsNil() { + // Repeated field that is empty, or a bytes field that is unused. + continue + } + + if props.Repeated && fv.Kind() == reflect.Slice { + // Repeated field. + for j := 0; j < fv.Len(); j++ { + if err := writeName(w, props); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte(' '); err != nil { + return err + } + } + v := fv.Index(j) + if v.Kind() == reflect.Ptr && v.IsNil() { + // A nil message in a repeated field is not valid, + // but we can handle that more gracefully than panicking. + if _, err := w.Write([]byte("\n")); err != nil { + return err + } + continue + } + if len(props.Enum) > 0 { + if err := writeEnum(w, v, props); err != nil { + return err + } + } else if err := writeAny(w, v, props); err != nil { + return err + } + if err := w.WriteByte('\n'); err != nil { + return err + } + } + continue + } + if fv.Kind() == reflect.Map { + // Map fields are rendered as a repeated struct with key/value fields. + keys := fv.MapKeys() + sort.Sort(mapKeys(keys)) + for _, key := range keys { + val := fv.MapIndex(key) + if err := writeName(w, props); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte(' '); err != nil { + return err + } + } + // open struct + if err := w.WriteByte('<'); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte('\n'); err != nil { + return err + } + } + w.indent() + // key + if _, err := w.WriteString("key:"); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte(' '); err != nil { + return err + } + } + if err := writeAny(w, key, props.mkeyprop); err != nil { + return err + } + if err := w.WriteByte('\n'); err != nil { + return err + } + // nil values aren't legal, but we can avoid panicking because of them. + if val.Kind() != reflect.Ptr || !val.IsNil() { + // value + if _, err := w.WriteString("value:"); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte(' '); err != nil { + return err + } + } + if err := writeAny(w, val, props.mvalprop); err != nil { + return err + } + if err := w.WriteByte('\n'); err != nil { + return err + } + } + // close struct + w.unindent() + if err := w.WriteByte('>'); err != nil { + return err + } + if err := w.WriteByte('\n'); err != nil { + return err + } + } + continue + } + if props.proto3 && fv.Kind() == reflect.Slice && fv.Len() == 0 { + // empty bytes field + continue + } + if props.proto3 && fv.Kind() != reflect.Ptr && fv.Kind() != reflect.Slice { + // proto3 non-repeated scalar field; skip if zero value + if isProto3Zero(fv) { + continue + } + } + + if err := writeName(w, props); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte(' '); err != nil { + return err + } + } + if b, ok := fv.Interface().(raw); ok { + if err := writeRaw(w, b.Bytes()); err != nil { + return err + } + continue + } + + if len(props.Enum) > 0 { + if err := writeEnum(w, fv, props); err != nil { + return err + } + } else if err := writeAny(w, fv, props); err != nil { + return err + } + + if err := w.WriteByte('\n'); err != nil { + return err + } + } + + // Extensions (the XXX_extensions field). + pv := sv.Addr() + if pv.Type().Implements(extendableProtoType) { + if err := writeExtensions(w, pv); err != nil { + return err + } + } + + return nil +} + +// writeRaw writes an uninterpreted raw message. +func writeRaw(w *textWriter, b []byte) error { + if err := w.WriteByte('<'); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte('\n'); err != nil { + return err + } + } + w.indent() + if err := writeUnknownStruct(w, b); err != nil { + return err + } + w.unindent() + if err := w.WriteByte('>'); err != nil { + return err + } + return nil +} + +// writeAny writes an arbitrary field. +func writeAny(w *textWriter, v reflect.Value, props *Properties) error { + v = reflect.Indirect(v) + + if props != nil && len(props.CustomType) > 0 { + var custom Marshaler = v.Interface().(Marshaler) + data, err := custom.Marshal() + if err != nil { + return err + } + if err := writeString(w, string(data)); err != nil { + return err + } + return nil + } + + // Floats have special cases. + if v.Kind() == reflect.Float32 || v.Kind() == reflect.Float64 { + x := v.Float() + var b []byte + switch { + case math.IsInf(x, 1): + b = posInf + case math.IsInf(x, -1): + b = negInf + case math.IsNaN(x): + b = nan + } + if b != nil { + _, err := w.Write(b) + return err + } + // Other values are handled below. + } + + // We don't attempt to serialise every possible value type; only those + // that can occur in protocol buffers. + switch v.Kind() { + case reflect.Slice: + // Should only be a []byte; repeated fields are handled in writeStruct. + if err := writeString(w, string(v.Bytes())); err != nil { + return err + } + case reflect.String: + if err := writeString(w, v.String()); err != nil { + return err + } + case reflect.Struct: + // Required/optional group/message. + var bra, ket byte = '<', '>' + if props != nil && props.Wire == "group" { + bra, ket = '{', '}' + } + if err := w.WriteByte(bra); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte('\n'); err != nil { + return err + } + } + w.indent() + if tm, ok := v.Interface().(encoding.TextMarshaler); ok { + text, err := tm.MarshalText() + if err != nil { + return err + } + if _, err = w.Write(text); err != nil { + return err + } + } else if err := writeStruct(w, v); err != nil { + return err + } + w.unindent() + if err := w.WriteByte(ket); err != nil { + return err + } + default: + _, err := fmt.Fprint(w, v.Interface()) + return err + } + return nil +} + +// equivalent to C's isprint. +func isprint(c byte) bool { + return c >= 0x20 && c < 0x7f +} + +// writeString writes a string in the protocol buffer text format. +// It is similar to strconv.Quote except we don't use Go escape sequences, +// we treat the string as a byte sequence, and we use octal escapes. +// These differences are to maintain interoperability with the other +// languages' implementations of the text format. +func writeString(w *textWriter, s string) error { + // use WriteByte here to get any needed indent + if err := w.WriteByte('"'); err != nil { + return err + } + // Loop over the bytes, not the runes. + for i := 0; i < len(s); i++ { + var err error + // Divergence from C++: we don't escape apostrophes. + // There's no need to escape them, and the C++ parser + // copes with a naked apostrophe. + switch c := s[i]; c { + case '\n': + _, err = w.w.Write(backslashN) + case '\r': + _, err = w.w.Write(backslashR) + case '\t': + _, err = w.w.Write(backslashT) + case '"': + _, err = w.w.Write(backslashDQ) + case '\\': + _, err = w.w.Write(backslashBS) + default: + if isprint(c) { + err = w.w.WriteByte(c) + } else { + _, err = fmt.Fprintf(w.w, "\\%03o", c) + } + } + if err != nil { + return err + } + } + return w.WriteByte('"') +} + +func writeMessageSet(w *textWriter, ms *MessageSet) error { + for _, item := range ms.Item { + id := *item.TypeId + if msd, ok := messageSetMap[id]; ok { + // Known message set type. + if _, err := fmt.Fprintf(w, "[%s]: <\n", msd.name); err != nil { + return err + } + w.indent() + + pb := reflect.New(msd.t.Elem()) + if err := Unmarshal(item.Message, pb.Interface().(Message)); err != nil { + if _, err := fmt.Fprintf(w, "/* bad message: %v */\n", err); err != nil { + return err + } + } else { + if err := writeStruct(w, pb.Elem()); err != nil { + return err + } + } + } else { + // Unknown type. + if _, err := fmt.Fprintf(w, "[%d]: <\n", id); err != nil { + return err + } + w.indent() + if err := writeUnknownStruct(w, item.Message); err != nil { + return err + } + } + w.unindent() + if _, err := w.Write(gtNewline); err != nil { + return err + } + } + return nil +} + +func writeUnknownStruct(w *textWriter, data []byte) error { + if !w.compact { + if _, err := fmt.Fprintf(w, "/* %d unknown bytes */\n", len(data)); err != nil { + return err + } + } + b := NewBuffer(data) + for b.index < len(b.buf) { + x, err := b.DecodeVarint() + if err != nil { + _, err := fmt.Fprintf(w, "/* %v */\n", err) + return err + } + wire, tag := x&7, x>>3 + if wire == WireEndGroup { + w.unindent() + if _, err := w.Write(endBraceNewline); err != nil { + return err + } + continue + } + if _, err := fmt.Fprint(w, tag); err != nil { + return err + } + if wire != WireStartGroup { + if err := w.WriteByte(':'); err != nil { + return err + } + } + if !w.compact || wire == WireStartGroup { + if err := w.WriteByte(' '); err != nil { + return err + } + } + switch wire { + case WireBytes: + buf, e := b.DecodeRawBytes(false) + if e == nil { + _, err = fmt.Fprintf(w, "%q", buf) + } else { + _, err = fmt.Fprintf(w, "/* %v */", e) + } + case WireFixed32: + x, err = b.DecodeFixed32() + err = writeUnknownInt(w, x, err) + case WireFixed64: + x, err = b.DecodeFixed64() + err = writeUnknownInt(w, x, err) + case WireStartGroup: + err = w.WriteByte('{') + w.indent() + case WireVarint: + x, err = b.DecodeVarint() + err = writeUnknownInt(w, x, err) + default: + _, err = fmt.Fprintf(w, "/* unknown wire type %d */", wire) + } + if err != nil { + return err + } + if err := w.WriteByte('\n'); err != nil { + return err + } + } + return nil +} + +func writeUnknownInt(w *textWriter, x uint64, err error) error { + if err == nil { + _, err = fmt.Fprint(w, x) + } else { + _, err = fmt.Fprintf(w, "/* %v */", err) + } + return err +} + +type int32Slice []int32 + +func (s int32Slice) Len() int { return len(s) } +func (s int32Slice) Less(i, j int) bool { return s[i] < s[j] } +func (s int32Slice) Swap(i, j int) { s[i], s[j] = s[j], s[i] } + +// writeExtensions writes all the extensions in pv. +// pv is assumed to be a pointer to a protocol message struct that is extendable. +func writeExtensions(w *textWriter, pv reflect.Value) error { + emap := extensionMaps[pv.Type().Elem()] + ep := pv.Interface().(extendableProto) + + // Order the extensions by ID. + // This isn't strictly necessary, but it will give us + // canonical output, which will also make testing easier. + var m map[int32]Extension + if em, ok := ep.(extensionsMap); ok { + m = em.ExtensionMap() + } else if em, ok := ep.(extensionsBytes); ok { + eb := em.GetExtensions() + var err error + m, err = BytesToExtensionsMap(*eb) + if err != nil { + return err + } + } + + ids := make([]int32, 0, len(m)) + for id := range m { + ids = append(ids, id) + } + sort.Sort(int32Slice(ids)) + + for _, extNum := range ids { + ext := m[extNum] + var desc *ExtensionDesc + if emap != nil { + desc = emap[extNum] + } + if desc == nil { + // Unknown extension. + if err := writeUnknownStruct(w, ext.enc); err != nil { + return err + } + continue + } + + pb, err := GetExtension(ep, desc) + if err != nil { + return fmt.Errorf("failed getting extension: %v", err) + } + + // Repeated extensions will appear as a slice. + if !desc.repeated() { + if err := writeExtension(w, desc.Name, pb); err != nil { + return err + } + } else { + v := reflect.ValueOf(pb) + for i := 0; i < v.Len(); i++ { + if err := writeExtension(w, desc.Name, v.Index(i).Interface()); err != nil { + return err + } + } + } + } + return nil +} + +func writeExtension(w *textWriter, name string, pb interface{}) error { + if _, err := fmt.Fprintf(w, "[%s]:", name); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte(' '); err != nil { + return err + } + } + if err := writeAny(w, reflect.ValueOf(pb), nil); err != nil { + return err + } + if err := w.WriteByte('\n'); err != nil { + return err + } + return nil +} + +func (w *textWriter) writeIndent() { + if !w.complete { + return + } + remain := w.ind * 2 + for remain > 0 { + n := remain + if n > len(spaces) { + n = len(spaces) + } + w.w.Write(spaces[:n]) + remain -= n + } + w.complete = false +} + +func marshalText(w io.Writer, pb Message, compact bool) error { + val := reflect.ValueOf(pb) + if pb == nil || val.IsNil() { + w.Write([]byte("")) + return nil + } + var bw *bufio.Writer + ww, ok := w.(writer) + if !ok { + bw = bufio.NewWriter(w) + ww = bw + } + aw := &textWriter{ + w: ww, + complete: true, + compact: compact, + } + + if tm, ok := pb.(encoding.TextMarshaler); ok { + text, err := tm.MarshalText() + if err != nil { + return err + } + if _, err = aw.Write(text); err != nil { + return err + } + if bw != nil { + return bw.Flush() + } + return nil + } + // Dereference the received pointer so we don't have outer < and >. + v := reflect.Indirect(val) + if err := writeStruct(aw, v); err != nil { + return err + } + if bw != nil { + return bw.Flush() + } + return nil +} + +// MarshalText writes a given protocol buffer in text format. +// The only errors returned are from w. +func MarshalText(w io.Writer, pb Message) error { + return marshalText(w, pb, false) +} + +// MarshalTextString is the same as MarshalText, but returns the string directly. +func MarshalTextString(pb Message) string { + var buf bytes.Buffer + marshalText(&buf, pb, false) + return buf.String() +} + +// CompactText writes a given protocol buffer in compact text format (one line). +func CompactText(w io.Writer, pb Message) error { return marshalText(w, pb, true) } + +// CompactTextString is the same as CompactText, but returns the string directly. +func CompactTextString(pb Message) string { + var buf bytes.Buffer + marshalText(&buf, pb, true) + return buf.String() +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text_gogo.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text_gogo.go new file mode 100644 index 0000000000000..cdb23373c39ba --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text_gogo.go @@ -0,0 +1,55 @@ +// Copyright (c) 2013, Vastech SA (PTY) LTD. All rights reserved. +// http://github.com/gogo/protobuf/gogoproto +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +import ( + "fmt" + "reflect" +) + +func writeEnum(w *textWriter, v reflect.Value, props *Properties) error { + m, ok := enumStringMaps[props.Enum] + if !ok { + if err := writeAny(w, v, props); err != nil { + return err + } + } + key := int32(0) + if v.Kind() == reflect.Ptr { + key = int32(v.Elem().Int()) + } else { + key = int32(v.Int()) + } + s, ok := m[key] + if !ok { + if err := writeAny(w, v, props); err != nil { + return err + } + } + _, err := fmt.Fprint(w, s) + return err +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text_parser.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text_parser.go new file mode 100644 index 0000000000000..9b2fab593572b --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text_parser.go @@ -0,0 +1,815 @@ +// Extensions for Protocol Buffers to create more go like structures. +// +// Copyright (c) 2013, Vastech SA (PTY) LTD. All rights reserved. +// http://github.com/gogo/protobuf/gogoproto +// +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +// Functions for parsing the Text protocol buffer format. +// TODO: message sets. + +import ( + "encoding" + "errors" + "fmt" + "reflect" + "strconv" + "strings" + "unicode/utf8" +) + +type ParseError struct { + Message string + Line int // 1-based line number + Offset int // 0-based byte offset from start of input +} + +func (p *ParseError) Error() string { + if p.Line == 1 { + // show offset only for first line + return fmt.Sprintf("line 1.%d: %v", p.Offset, p.Message) + } + return fmt.Sprintf("line %d: %v", p.Line, p.Message) +} + +type token struct { + value string + err *ParseError + line int // line number + offset int // byte number from start of input, not start of line + unquoted string // the unquoted version of value, if it was a quoted string +} + +func (t *token) String() string { + if t.err == nil { + return fmt.Sprintf("%q (line=%d, offset=%d)", t.value, t.line, t.offset) + } + return fmt.Sprintf("parse error: %v", t.err) +} + +type textParser struct { + s string // remaining input + done bool // whether the parsing is finished (success or error) + backed bool // whether back() was called + offset, line int + cur token +} + +func newTextParser(s string) *textParser { + p := new(textParser) + p.s = s + p.line = 1 + p.cur.line = 1 + return p +} + +func (p *textParser) errorf(format string, a ...interface{}) *ParseError { + pe := &ParseError{fmt.Sprintf(format, a...), p.cur.line, p.cur.offset} + p.cur.err = pe + p.done = true + return pe +} + +// Numbers and identifiers are matched by [-+._A-Za-z0-9] +func isIdentOrNumberChar(c byte) bool { + switch { + case 'A' <= c && c <= 'Z', 'a' <= c && c <= 'z': + return true + case '0' <= c && c <= '9': + return true + } + switch c { + case '-', '+', '.', '_': + return true + } + return false +} + +func isWhitespace(c byte) bool { + switch c { + case ' ', '\t', '\n', '\r': + return true + } + return false +} + +func (p *textParser) skipWhitespace() { + i := 0 + for i < len(p.s) && (isWhitespace(p.s[i]) || p.s[i] == '#') { + if p.s[i] == '#' { + // comment; skip to end of line or input + for i < len(p.s) && p.s[i] != '\n' { + i++ + } + if i == len(p.s) { + break + } + } + if p.s[i] == '\n' { + p.line++ + } + i++ + } + p.offset += i + p.s = p.s[i:len(p.s)] + if len(p.s) == 0 { + p.done = true + } +} + +func (p *textParser) advance() { + // Skip whitespace + p.skipWhitespace() + if p.done { + return + } + + // Start of non-whitespace + p.cur.err = nil + p.cur.offset, p.cur.line = p.offset, p.line + p.cur.unquoted = "" + switch p.s[0] { + case '<', '>', '{', '}', ':', '[', ']', ';', ',': + // Single symbol + p.cur.value, p.s = p.s[0:1], p.s[1:len(p.s)] + case '"', '\'': + // Quoted string + i := 1 + for i < len(p.s) && p.s[i] != p.s[0] && p.s[i] != '\n' { + if p.s[i] == '\\' && i+1 < len(p.s) { + // skip escaped char + i++ + } + i++ + } + if i >= len(p.s) || p.s[i] != p.s[0] { + p.errorf("unmatched quote") + return + } + unq, err := unquoteC(p.s[1:i], rune(p.s[0])) + if err != nil { + p.errorf("invalid quoted string %s: %v", p.s[0:i+1], err) + return + } + p.cur.value, p.s = p.s[0:i+1], p.s[i+1:len(p.s)] + p.cur.unquoted = unq + default: + i := 0 + for i < len(p.s) && isIdentOrNumberChar(p.s[i]) { + i++ + } + if i == 0 { + p.errorf("unexpected byte %#x", p.s[0]) + return + } + p.cur.value, p.s = p.s[0:i], p.s[i:len(p.s)] + } + p.offset += len(p.cur.value) +} + +var ( + errBadUTF8 = errors.New("proto: bad UTF-8") + errBadHex = errors.New("proto: bad hexadecimal") +) + +func unquoteC(s string, quote rune) (string, error) { + // This is based on C++'s tokenizer.cc. + // Despite its name, this is *not* parsing C syntax. + // For instance, "\0" is an invalid quoted string. + + // Avoid allocation in trivial cases. + simple := true + for _, r := range s { + if r == '\\' || r == quote { + simple = false + break + } + } + if simple { + return s, nil + } + + buf := make([]byte, 0, 3*len(s)/2) + for len(s) > 0 { + r, n := utf8.DecodeRuneInString(s) + if r == utf8.RuneError && n == 1 { + return "", errBadUTF8 + } + s = s[n:] + if r != '\\' { + if r < utf8.RuneSelf { + buf = append(buf, byte(r)) + } else { + buf = append(buf, string(r)...) + } + continue + } + + ch, tail, err := unescape(s) + if err != nil { + return "", err + } + buf = append(buf, ch...) + s = tail + } + return string(buf), nil +} + +func unescape(s string) (ch string, tail string, err error) { + r, n := utf8.DecodeRuneInString(s) + if r == utf8.RuneError && n == 1 { + return "", "", errBadUTF8 + } + s = s[n:] + switch r { + case 'a': + return "\a", s, nil + case 'b': + return "\b", s, nil + case 'f': + return "\f", s, nil + case 'n': + return "\n", s, nil + case 'r': + return "\r", s, nil + case 't': + return "\t", s, nil + case 'v': + return "\v", s, nil + case '?': + return "?", s, nil // trigraph workaround + case '\'', '"', '\\': + return string(r), s, nil + case '0', '1', '2', '3', '4', '5', '6', '7', 'x', 'X': + if len(s) < 2 { + return "", "", fmt.Errorf(`\%c requires 2 following digits`, r) + } + base := 8 + ss := s[:2] + s = s[2:] + if r == 'x' || r == 'X' { + base = 16 + } else { + ss = string(r) + ss + } + i, err := strconv.ParseUint(ss, base, 8) + if err != nil { + return "", "", err + } + return string([]byte{byte(i)}), s, nil + case 'u', 'U': + n := 4 + if r == 'U' { + n = 8 + } + if len(s) < n { + return "", "", fmt.Errorf(`\%c requires %d digits`, r, n) + } + + bs := make([]byte, n/2) + for i := 0; i < n; i += 2 { + a, ok1 := unhex(s[i]) + b, ok2 := unhex(s[i+1]) + if !ok1 || !ok2 { + return "", "", errBadHex + } + bs[i/2] = a<<4 | b + } + s = s[n:] + return string(bs), s, nil + } + return "", "", fmt.Errorf(`unknown escape \%c`, r) +} + +// Adapted from src/pkg/strconv/quote.go. +func unhex(b byte) (v byte, ok bool) { + switch { + case '0' <= b && b <= '9': + return b - '0', true + case 'a' <= b && b <= 'f': + return b - 'a' + 10, true + case 'A' <= b && b <= 'F': + return b - 'A' + 10, true + } + return 0, false +} + +// Back off the parser by one token. Can only be done between calls to next(). +// It makes the next advance() a no-op. +func (p *textParser) back() { p.backed = true } + +// Advances the parser and returns the new current token. +func (p *textParser) next() *token { + if p.backed || p.done { + p.backed = false + return &p.cur + } + p.advance() + if p.done { + p.cur.value = "" + } else if len(p.cur.value) > 0 && p.cur.value[0] == '"' { + // Look for multiple quoted strings separated by whitespace, + // and concatenate them. + cat := p.cur + for { + p.skipWhitespace() + if p.done || p.s[0] != '"' { + break + } + p.advance() + if p.cur.err != nil { + return &p.cur + } + cat.value += " " + p.cur.value + cat.unquoted += p.cur.unquoted + } + p.done = false // parser may have seen EOF, but we want to return cat + p.cur = cat + } + return &p.cur +} + +func (p *textParser) consumeToken(s string) error { + tok := p.next() + if tok.err != nil { + return tok.err + } + if tok.value != s { + p.back() + return p.errorf("expected %q, found %q", s, tok.value) + } + return nil +} + +// Return a RequiredNotSetError indicating which required field was not set. +func (p *textParser) missingRequiredFieldError(sv reflect.Value) *RequiredNotSetError { + st := sv.Type() + sprops := GetProperties(st) + for i := 0; i < st.NumField(); i++ { + if !isNil(sv.Field(i)) { + continue + } + + props := sprops.Prop[i] + if props.Required { + return &RequiredNotSetError{fmt.Sprintf("%v.%v", st, props.OrigName)} + } + } + return &RequiredNotSetError{fmt.Sprintf("%v.", st)} // should not happen +} + +// Returns the index in the struct for the named field, as well as the parsed tag properties. +func structFieldByName(st reflect.Type, name string) (int, *Properties, bool) { + sprops := GetProperties(st) + i, ok := sprops.decoderOrigNames[name] + if ok { + return i, sprops.Prop[i], true + } + return -1, nil, false +} + +// Consume a ':' from the input stream (if the next token is a colon), +// returning an error if a colon is needed but not present. +func (p *textParser) checkForColon(props *Properties, typ reflect.Type) *ParseError { + tok := p.next() + if tok.err != nil { + return tok.err + } + if tok.value != ":" { + // Colon is optional when the field is a group or message. + needColon := true + switch props.Wire { + case "group": + needColon = false + case "bytes": + // A "bytes" field is either a message, a string, or a repeated field; + // those three become *T, *string and []T respectively, so we can check for + // this field being a pointer to a non-string. + if typ.Kind() == reflect.Ptr { + // *T or *string + if typ.Elem().Kind() == reflect.String { + break + } + } else if typ.Kind() == reflect.Slice { + // []T or []*T + if typ.Elem().Kind() != reflect.Ptr { + break + } + } else if typ.Kind() == reflect.String { + // The proto3 exception is for a string field, + // which requires a colon. + break + } + needColon = false + } + if needColon { + return p.errorf("expected ':', found %q", tok.value) + } + p.back() + } + return nil +} + +func (p *textParser) readStruct(sv reflect.Value, terminator string) error { + st := sv.Type() + reqCount := GetProperties(st).reqCount + var reqFieldErr error + fieldSet := make(map[string]bool) + // A struct is a sequence of "name: value", terminated by one of + // '>' or '}', or the end of the input. A name may also be + // "[extension]". + for { + tok := p.next() + if tok.err != nil { + return tok.err + } + if tok.value == terminator { + break + } + if tok.value == "[" { + // Looks like an extension. + // + // TODO: Check whether we need to handle + // namespace rooted names (e.g. ".something.Foo"). + tok = p.next() + if tok.err != nil { + return tok.err + } + var desc *ExtensionDesc + // This could be faster, but it's functional. + // TODO: Do something smarter than a linear scan. + for _, d := range RegisteredExtensions(reflect.New(st).Interface().(Message)) { + if d.Name == tok.value { + desc = d + break + } + } + if desc == nil { + return p.errorf("unrecognized extension %q", tok.value) + } + // Check the extension terminator. + tok = p.next() + if tok.err != nil { + return tok.err + } + if tok.value != "]" { + return p.errorf("unrecognized extension terminator %q", tok.value) + } + + props := &Properties{} + props.Parse(desc.Tag) + + typ := reflect.TypeOf(desc.ExtensionType) + if err := p.checkForColon(props, typ); err != nil { + return err + } + + rep := desc.repeated() + + // Read the extension structure, and set it in + // the value we're constructing. + var ext reflect.Value + if !rep { + ext = reflect.New(typ).Elem() + } else { + ext = reflect.New(typ.Elem()).Elem() + } + if err := p.readAny(ext, props); err != nil { + if _, ok := err.(*RequiredNotSetError); !ok { + return err + } + reqFieldErr = err + } + ep := sv.Addr().Interface().(extendableProto) + if !rep { + SetExtension(ep, desc, ext.Interface()) + } else { + old, err := GetExtension(ep, desc) + var sl reflect.Value + if err == nil { + sl = reflect.ValueOf(old) // existing slice + } else { + sl = reflect.MakeSlice(typ, 0, 1) + } + sl = reflect.Append(sl, ext) + SetExtension(ep, desc, sl.Interface()) + } + } else { + // This is a normal, non-extension field. + name := tok.value + fi, props, ok := structFieldByName(st, name) + if !ok { + return p.errorf("unknown field name %q in %v", name, st) + } + + dst := sv.Field(fi) + + if dst.Kind() == reflect.Map { + // Consume any colon. + if err := p.checkForColon(props, dst.Type()); err != nil { + return err + } + + // Construct the map if it doesn't already exist. + if dst.IsNil() { + dst.Set(reflect.MakeMap(dst.Type())) + } + key := reflect.New(dst.Type().Key()).Elem() + val := reflect.New(dst.Type().Elem()).Elem() + + // The map entry should be this sequence of tokens: + // < key : KEY value : VALUE > + // Technically the "key" and "value" could come in any order, + // but in practice they won't. + + tok := p.next() + var terminator string + switch tok.value { + case "<": + terminator = ">" + case "{": + terminator = "}" + default: + return p.errorf("expected '{' or '<', found %q", tok.value) + } + if err := p.consumeToken("key"); err != nil { + return err + } + if err := p.consumeToken(":"); err != nil { + return err + } + if err := p.readAny(key, props.mkeyprop); err != nil { + return err + } + if err := p.consumeOptionalSeparator(); err != nil { + return err + } + if err := p.consumeToken("value"); err != nil { + return err + } + if err := p.checkForColon(props.mvalprop, dst.Type().Elem()); err != nil { + return err + } + if err := p.readAny(val, props.mvalprop); err != nil { + return err + } + if err := p.consumeOptionalSeparator(); err != nil { + return err + } + if err := p.consumeToken(terminator); err != nil { + return err + } + + dst.SetMapIndex(key, val) + continue + } + + // Check that it's not already set if it's not a repeated field. + if !props.Repeated && fieldSet[name] { + return p.errorf("non-repeated field %q was repeated", name) + } + + if err := p.checkForColon(props, st.Field(fi).Type); err != nil { + return err + } + + // Parse into the field. + fieldSet[name] = true + if err := p.readAny(dst, props); err != nil { + if _, ok := err.(*RequiredNotSetError); !ok { + return err + } + reqFieldErr = err + } else if props.Required { + reqCount-- + } + } + + if err := p.consumeOptionalSeparator(); err != nil { + return err + } + + } + + if reqCount > 0 { + return p.missingRequiredFieldError(sv) + } + return reqFieldErr +} + +// consumeOptionalSeparator consumes an optional semicolon or comma. +// It is used in readStruct to provide backward compatibility. +func (p *textParser) consumeOptionalSeparator() error { + tok := p.next() + if tok.err != nil { + return tok.err + } + if tok.value != ";" && tok.value != "," { + p.back() + } + return nil +} + +func (p *textParser) readAny(v reflect.Value, props *Properties) error { + tok := p.next() + if tok.err != nil { + return tok.err + } + if tok.value == "" { + return p.errorf("unexpected EOF") + } + if len(props.CustomType) > 0 { + if props.Repeated { + t := reflect.TypeOf(v.Interface()) + if t.Kind() == reflect.Slice { + tc := reflect.TypeOf(new(Marshaler)) + ok := t.Elem().Implements(tc.Elem()) + if ok { + fv := v + flen := fv.Len() + if flen == fv.Cap() { + nav := reflect.MakeSlice(v.Type(), flen, 2*flen+1) + reflect.Copy(nav, fv) + fv.Set(nav) + } + fv.SetLen(flen + 1) + + // Read one. + p.back() + return p.readAny(fv.Index(flen), props) + } + } + } + if reflect.TypeOf(v.Interface()).Kind() == reflect.Ptr { + custom := reflect.New(props.ctype.Elem()).Interface().(Unmarshaler) + err := custom.Unmarshal([]byte(tok.unquoted)) + if err != nil { + return p.errorf("%v %v: %v", err, v.Type(), tok.value) + } + v.Set(reflect.ValueOf(custom)) + } else { + custom := reflect.New(reflect.TypeOf(v.Interface())).Interface().(Unmarshaler) + err := custom.Unmarshal([]byte(tok.unquoted)) + if err != nil { + return p.errorf("%v %v: %v", err, v.Type(), tok.value) + } + v.Set(reflect.Indirect(reflect.ValueOf(custom))) + } + return nil + } + switch fv := v; fv.Kind() { + case reflect.Slice: + at := v.Type() + if at.Elem().Kind() == reflect.Uint8 { + // Special case for []byte + if tok.value[0] != '"' && tok.value[0] != '\'' { + // Deliberately written out here, as the error after + // this switch statement would write "invalid []byte: ...", + // which is not as user-friendly. + return p.errorf("invalid string: %v", tok.value) + } + bytes := []byte(tok.unquoted) + fv.Set(reflect.ValueOf(bytes)) + return nil + } + // Repeated field. May already exist. + flen := fv.Len() + if flen == fv.Cap() { + nav := reflect.MakeSlice(at, flen, 2*flen+1) + reflect.Copy(nav, fv) + fv.Set(nav) + } + fv.SetLen(flen + 1) + + // Read one. + p.back() + return p.readAny(fv.Index(flen), props) + case reflect.Bool: + // Either "true", "false", 1 or 0. + switch tok.value { + case "true", "1": + fv.SetBool(true) + return nil + case "false", "0": + fv.SetBool(false) + return nil + } + case reflect.Float32, reflect.Float64: + v := tok.value + // Ignore 'f' for compatibility with output generated by C++, but don't + // remove 'f' when the value is "-inf" or "inf". + if strings.HasSuffix(v, "f") && tok.value != "-inf" && tok.value != "inf" { + v = v[:len(v)-1] + } + if f, err := strconv.ParseFloat(v, fv.Type().Bits()); err == nil { + fv.SetFloat(f) + return nil + } + case reflect.Int32: + if x, err := strconv.ParseInt(tok.value, 0, 32); err == nil { + fv.SetInt(x) + return nil + } + + if len(props.Enum) == 0 { + break + } + m, ok := enumValueMaps[props.Enum] + if !ok { + break + } + x, ok := m[tok.value] + if !ok { + break + } + fv.SetInt(int64(x)) + return nil + case reflect.Int64: + if x, err := strconv.ParseInt(tok.value, 0, 64); err == nil { + fv.SetInt(x) + return nil + } + + case reflect.Ptr: + // A basic field (indirected through pointer), or a repeated message/group + p.back() + fv.Set(reflect.New(fv.Type().Elem())) + return p.readAny(fv.Elem(), props) + case reflect.String: + if tok.value[0] == '"' || tok.value[0] == '\'' { + fv.SetString(tok.unquoted) + return nil + } + case reflect.Struct: + var terminator string + switch tok.value { + case "{": + terminator = "}" + case "<": + terminator = ">" + default: + return p.errorf("expected '{' or '<', found %q", tok.value) + } + // TODO: Handle nested messages which implement encoding.TextUnmarshaler. + return p.readStruct(fv, terminator) + case reflect.Uint32: + if x, err := strconv.ParseUint(tok.value, 0, 32); err == nil { + fv.SetUint(uint64(x)) + return nil + } + case reflect.Uint64: + if x, err := strconv.ParseUint(tok.value, 0, 64); err == nil { + fv.SetUint(x) + return nil + } + } + return p.errorf("invalid %v: %v", v.Type(), tok.value) +} + +// UnmarshalText reads a protocol buffer in Text format. UnmarshalText resets pb +// before starting to unmarshal, so any existing data in pb is always removed. +// If a required field is not set and no other error occurs, +// UnmarshalText returns *RequiredNotSetError. +func UnmarshalText(s string, pb Message) error { + if um, ok := pb.(encoding.TextUnmarshaler); ok { + err := um.UnmarshalText([]byte(s)) + return err + } + pb.Reset() + v := reflect.ValueOf(pb) + if pe := newTextParser(s).readStruct(v.Elem(), ""); pe != nil { + return pe + } + return nil +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text_parser_test.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text_parser_test.go new file mode 100644 index 0000000000000..f1c62319251b5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text_parser_test.go @@ -0,0 +1,511 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "math" + "reflect" + "testing" + + . "github.com/gogo/protobuf/proto" + proto3pb "github.com/gogo/protobuf/proto/proto3_proto" + . "github.com/gogo/protobuf/proto/testdata" +) + +type UnmarshalTextTest struct { + in string + err string // if "", no error expected + out *MyMessage +} + +func buildExtStructTest(text string) UnmarshalTextTest { + msg := &MyMessage{ + Count: Int32(42), + } + SetExtension(msg, E_Ext_More, &Ext{ + Data: String("Hello, world!"), + }) + return UnmarshalTextTest{in: text, out: msg} +} + +func buildExtDataTest(text string) UnmarshalTextTest { + msg := &MyMessage{ + Count: Int32(42), + } + SetExtension(msg, E_Ext_Text, String("Hello, world!")) + SetExtension(msg, E_Ext_Number, Int32(1729)) + return UnmarshalTextTest{in: text, out: msg} +} + +func buildExtRepStringTest(text string) UnmarshalTextTest { + msg := &MyMessage{ + Count: Int32(42), + } + if err := SetExtension(msg, E_Greeting, []string{"bula", "hola"}); err != nil { + panic(err) + } + return UnmarshalTextTest{in: text, out: msg} +} + +var unMarshalTextTests = []UnmarshalTextTest{ + // Basic + { + in: " count:42\n name:\"Dave\" ", + out: &MyMessage{ + Count: Int32(42), + Name: String("Dave"), + }, + }, + + // Empty quoted string + { + in: `count:42 name:""`, + out: &MyMessage{ + Count: Int32(42), + Name: String(""), + }, + }, + + // Quoted string concatenation + { + in: `count:42 name: "My name is "` + "\n" + `"elsewhere"`, + out: &MyMessage{ + Count: Int32(42), + Name: String("My name is elsewhere"), + }, + }, + + // Quoted string with escaped apostrophe + { + in: `count:42 name: "HOLIDAY - New Year\'s Day"`, + out: &MyMessage{ + Count: Int32(42), + Name: String("HOLIDAY - New Year's Day"), + }, + }, + + // Quoted string with single quote + { + in: `count:42 name: 'Roger "The Ramster" Ramjet'`, + out: &MyMessage{ + Count: Int32(42), + Name: String(`Roger "The Ramster" Ramjet`), + }, + }, + + // Quoted string with all the accepted special characters from the C++ test + { + in: `count:42 name: ` + "\"\\\"A string with \\' characters \\n and \\r newlines and \\t tabs and \\001 slashes \\\\ and multiple spaces\"", + out: &MyMessage{ + Count: Int32(42), + Name: String("\"A string with ' characters \n and \r newlines and \t tabs and \001 slashes \\ and multiple spaces"), + }, + }, + + // Quoted string with quoted backslash + { + in: `count:42 name: "\\'xyz"`, + out: &MyMessage{ + Count: Int32(42), + Name: String(`\'xyz`), + }, + }, + + // Quoted string with UTF-8 bytes. + { + in: "count:42 name: '\303\277\302\201\xAB'", + out: &MyMessage{ + Count: Int32(42), + Name: String("\303\277\302\201\xAB"), + }, + }, + + // Bad quoted string + { + in: `inner: < host: "\0" >` + "\n", + err: `line 1.15: invalid quoted string "\0": \0 requires 2 following digits`, + }, + + // Number too large for int64 + { + in: "count: 1 others { key: 123456789012345678901 }", + err: "line 1.23: invalid int64: 123456789012345678901", + }, + + // Number too large for int32 + { + in: "count: 1234567890123", + err: "line 1.7: invalid int32: 1234567890123", + }, + + // Number in hexadecimal + { + in: "count: 0x2beef", + out: &MyMessage{ + Count: Int32(0x2beef), + }, + }, + + // Number in octal + { + in: "count: 024601", + out: &MyMessage{ + Count: Int32(024601), + }, + }, + + // Floating point number with "f" suffix + { + in: "count: 4 others:< weight: 17.0f >", + out: &MyMessage{ + Count: Int32(4), + Others: []*OtherMessage{ + { + Weight: Float32(17), + }, + }, + }, + }, + + // Floating point positive infinity + { + in: "count: 4 bigfloat: inf", + out: &MyMessage{ + Count: Int32(4), + Bigfloat: Float64(math.Inf(1)), + }, + }, + + // Floating point negative infinity + { + in: "count: 4 bigfloat: -inf", + out: &MyMessage{ + Count: Int32(4), + Bigfloat: Float64(math.Inf(-1)), + }, + }, + + // Number too large for float32 + { + in: "others:< weight: 12345678901234567890123456789012345678901234567890 >", + err: "line 1.17: invalid float32: 12345678901234567890123456789012345678901234567890", + }, + + // Number posing as a quoted string + { + in: `inner: < host: 12 >` + "\n", + err: `line 1.15: invalid string: 12`, + }, + + // Quoted string posing as int32 + { + in: `count: "12"`, + err: `line 1.7: invalid int32: "12"`, + }, + + // Quoted string posing a float32 + { + in: `others:< weight: "17.4" >`, + err: `line 1.17: invalid float32: "17.4"`, + }, + + // Enum + { + in: `count:42 bikeshed: BLUE`, + out: &MyMessage{ + Count: Int32(42), + Bikeshed: MyMessage_BLUE.Enum(), + }, + }, + + // Repeated field + { + in: `count:42 pet: "horsey" pet:"bunny"`, + out: &MyMessage{ + Count: Int32(42), + Pet: []string{"horsey", "bunny"}, + }, + }, + + // Repeated message with/without colon and <>/{} + { + in: `count:42 others:{} others{} others:<> others:{}`, + out: &MyMessage{ + Count: Int32(42), + Others: []*OtherMessage{ + {}, + {}, + {}, + {}, + }, + }, + }, + + // Missing colon for inner message + { + in: `count:42 inner < host: "cauchy.syd" >`, + out: &MyMessage{ + Count: Int32(42), + Inner: &InnerMessage{ + Host: String("cauchy.syd"), + }, + }, + }, + + // Missing colon for string field + { + in: `name "Dave"`, + err: `line 1.5: expected ':', found "\"Dave\""`, + }, + + // Missing colon for int32 field + { + in: `count 42`, + err: `line 1.6: expected ':', found "42"`, + }, + + // Missing required field + { + in: `name: "Pawel"`, + err: `proto: required field "testdata.MyMessage.count" not set`, + out: &MyMessage{ + Name: String("Pawel"), + }, + }, + + // Repeated non-repeated field + { + in: `name: "Rob" name: "Russ"`, + err: `line 1.12: non-repeated field "name" was repeated`, + }, + + // Group + { + in: `count: 17 SomeGroup { group_field: 12 }`, + out: &MyMessage{ + Count: Int32(17), + Somegroup: &MyMessage_SomeGroup{ + GroupField: Int32(12), + }, + }, + }, + + // Semicolon between fields + { + in: `count:3;name:"Calvin"`, + out: &MyMessage{ + Count: Int32(3), + Name: String("Calvin"), + }, + }, + // Comma between fields + { + in: `count:4,name:"Ezekiel"`, + out: &MyMessage{ + Count: Int32(4), + Name: String("Ezekiel"), + }, + }, + + // Extension + buildExtStructTest(`count: 42 [testdata.Ext.more]:`), + buildExtStructTest(`count: 42 [testdata.Ext.more] {data:"Hello, world!"}`), + buildExtDataTest(`count: 42 [testdata.Ext.text]:"Hello, world!" [testdata.Ext.number]:1729`), + buildExtRepStringTest(`count: 42 [testdata.greeting]:"bula" [testdata.greeting]:"hola"`), + + // Big all-in-one + { + in: "count:42 # Meaning\n" + + `name:"Dave" ` + + `quote:"\"I didn't want to go.\"" ` + + `pet:"bunny" ` + + `pet:"kitty" ` + + `pet:"horsey" ` + + `inner:<` + + ` host:"footrest.syd" ` + + ` port:7001 ` + + ` connected:true ` + + `> ` + + `others:<` + + ` key:3735928559 ` + + ` value:"\x01A\a\f" ` + + `> ` + + `others:<` + + " weight:58.9 # Atomic weight of Co\n" + + ` inner:<` + + ` host:"lesha.mtv" ` + + ` port:8002 ` + + ` >` + + `>`, + out: &MyMessage{ + Count: Int32(42), + Name: String("Dave"), + Quote: String(`"I didn't want to go."`), + Pet: []string{"bunny", "kitty", "horsey"}, + Inner: &InnerMessage{ + Host: String("footrest.syd"), + Port: Int32(7001), + Connected: Bool(true), + }, + Others: []*OtherMessage{ + { + Key: Int64(3735928559), + Value: []byte{0x1, 'A', '\a', '\f'}, + }, + { + Weight: Float32(58.9), + Inner: &InnerMessage{ + Host: String("lesha.mtv"), + Port: Int32(8002), + }, + }, + }, + }, + }, +} + +func TestUnmarshalText(t *testing.T) { + for i, test := range unMarshalTextTests { + pb := new(MyMessage) + err := UnmarshalText(test.in, pb) + if test.err == "" { + // We don't expect failure. + if err != nil { + t.Errorf("Test %d: Unexpected error: %v", i, err) + } else if !reflect.DeepEqual(pb, test.out) { + t.Errorf("Test %d: Incorrect populated \nHave: %v\nWant: %v", + i, pb, test.out) + } + } else { + // We do expect failure. + if err == nil { + t.Errorf("Test %d: Didn't get expected error: %v", i, test.err) + } else if err.Error() != test.err { + t.Errorf("Test %d: Incorrect error.\nHave: %v\nWant: %v", + i, err.Error(), test.err) + } else if _, ok := err.(*RequiredNotSetError); ok && test.out != nil && !reflect.DeepEqual(pb, test.out) { + t.Errorf("Test %d: Incorrect populated \nHave: %v\nWant: %v", + i, pb, test.out) + } + } + } +} + +func TestUnmarshalTextCustomMessage(t *testing.T) { + msg := &textMessage{} + if err := UnmarshalText("custom", msg); err != nil { + t.Errorf("Unexpected error from custom unmarshal: %v", err) + } + if UnmarshalText("not custom", msg) == nil { + t.Errorf("Didn't get expected error from custom unmarshal") + } +} + +// Regression test; this caused a panic. +func TestRepeatedEnum(t *testing.T) { + pb := new(RepeatedEnum) + if err := UnmarshalText("color: RED", pb); err != nil { + t.Fatal(err) + } + exp := &RepeatedEnum{ + Color: []RepeatedEnum_Color{RepeatedEnum_RED}, + } + if !Equal(pb, exp) { + t.Errorf("Incorrect populated \nHave: %v\nWant: %v", pb, exp) + } +} + +func TestProto3TextParsing(t *testing.T) { + m := new(proto3pb.Message) + const in = `name: "Wallace" true_scotsman: true` + want := &proto3pb.Message{ + Name: "Wallace", + TrueScotsman: true, + } + if err := UnmarshalText(in, m); err != nil { + t.Fatal(err) + } + if !Equal(m, want) { + t.Errorf("\n got %v\nwant %v", m, want) + } +} + +func TestMapParsing(t *testing.T) { + m := new(MessageWithMap) + const in = `name_mapping: name_mapping:` + + `msg_mapping:,>` + // separating commas are okay + `msg_mapping>` + // no colon after "value" + `byte_mapping:` + want := &MessageWithMap{ + NameMapping: map[int32]string{ + 1: "Beatles", + 1234: "Feist", + }, + MsgMapping: map[int64]*FloatingPoint{ + -4: {F: Float64(2.0)}, + -2: {F: Float64(4.0)}, + }, + ByteMapping: map[bool][]byte{ + true: []byte("so be it"), + }, + } + if err := UnmarshalText(in, m); err != nil { + t.Fatal(err) + } + if !Equal(m, want) { + t.Errorf("\n got %v\nwant %v", m, want) + } +} + +var benchInput string + +func init() { + benchInput = "count: 4\n" + for i := 0; i < 1000; i++ { + benchInput += "pet: \"fido\"\n" + } + + // Check it is valid input. + pb := new(MyMessage) + err := UnmarshalText(benchInput, pb) + if err != nil { + panic("Bad benchmark input: " + err.Error()) + } +} + +func BenchmarkUnmarshalText(b *testing.B) { + pb := new(MyMessage) + for i := 0; i < b.N; i++ { + UnmarshalText(benchInput, pb) + } + b.SetBytes(int64(len(benchInput))) +} diff --git a/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text_test.go b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text_test.go new file mode 100644 index 0000000000000..a98504ef35d05 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gogo/protobuf/proto/text_test.go @@ -0,0 +1,450 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "bytes" + "errors" + "io/ioutil" + "math" + "strings" + "testing" + + "github.com/gogo/protobuf/proto" + + proto3pb "github.com/gogo/protobuf/proto/proto3_proto" + pb "github.com/gogo/protobuf/proto/testdata" +) + +// textMessage implements the methods that allow it to marshal and unmarshal +// itself as text. +type textMessage struct { +} + +func (*textMessage) MarshalText() ([]byte, error) { + return []byte("custom"), nil +} + +func (*textMessage) UnmarshalText(bytes []byte) error { + if string(bytes) != "custom" { + return errors.New("expected 'custom'") + } + return nil +} + +func (*textMessage) Reset() {} +func (*textMessage) String() string { return "" } +func (*textMessage) ProtoMessage() {} + +func newTestMessage() *pb.MyMessage { + msg := &pb.MyMessage{ + Count: proto.Int32(42), + Name: proto.String("Dave"), + Quote: proto.String(`"I didn't want to go."`), + Pet: []string{"bunny", "kitty", "horsey"}, + Inner: &pb.InnerMessage{ + Host: proto.String("footrest.syd"), + Port: proto.Int32(7001), + Connected: proto.Bool(true), + }, + Others: []*pb.OtherMessage{ + { + Key: proto.Int64(0xdeadbeef), + Value: []byte{1, 65, 7, 12}, + }, + { + Weight: proto.Float32(6.022), + Inner: &pb.InnerMessage{ + Host: proto.String("lesha.mtv"), + Port: proto.Int32(8002), + }, + }, + }, + Bikeshed: pb.MyMessage_BLUE.Enum(), + Somegroup: &pb.MyMessage_SomeGroup{ + GroupField: proto.Int32(8), + }, + // One normally wouldn't do this. + // This is an undeclared tag 13, as a varint (wire type 0) with value 4. + XXX_unrecognized: []byte{13<<3 | 0, 4}, + } + ext := &pb.Ext{ + Data: proto.String("Big gobs for big rats"), + } + if err := proto.SetExtension(msg, pb.E_Ext_More, ext); err != nil { + panic(err) + } + greetings := []string{"adg", "easy", "cow"} + if err := proto.SetExtension(msg, pb.E_Greeting, greetings); err != nil { + panic(err) + } + + // Add an unknown extension. We marshal a pb.Ext, and fake the ID. + b, err := proto.Marshal(&pb.Ext{Data: proto.String("3G skiing")}) + if err != nil { + panic(err) + } + b = append(proto.EncodeVarint(201<<3|proto.WireBytes), b...) + proto.SetRawExtension(msg, 201, b) + + // Extensions can be plain fields, too, so let's test that. + b = append(proto.EncodeVarint(202<<3|proto.WireVarint), 19) + proto.SetRawExtension(msg, 202, b) + + return msg +} + +const text = `count: 42 +name: "Dave" +quote: "\"I didn't want to go.\"" +pet: "bunny" +pet: "kitty" +pet: "horsey" +inner: < + host: "footrest.syd" + port: 7001 + connected: true +> +others: < + key: 3735928559 + value: "\001A\007\014" +> +others: < + weight: 6.022 + inner: < + host: "lesha.mtv" + port: 8002 + > +> +bikeshed: BLUE +SomeGroup { + group_field: 8 +} +/* 2 unknown bytes */ +13: 4 +[testdata.Ext.more]: < + data: "Big gobs for big rats" +> +[testdata.greeting]: "adg" +[testdata.greeting]: "easy" +[testdata.greeting]: "cow" +/* 13 unknown bytes */ +201: "\t3G skiing" +/* 3 unknown bytes */ +202: 19 +` + +func TestMarshalText(t *testing.T) { + buf := new(bytes.Buffer) + if err := proto.MarshalText(buf, newTestMessage()); err != nil { + t.Fatalf("proto.MarshalText: %v", err) + } + s := buf.String() + if s != text { + t.Errorf("Got:\n===\n%v===\nExpected:\n===\n%v===\n", s, text) + } +} + +func TestMarshalTextCustomMessage(t *testing.T) { + buf := new(bytes.Buffer) + if err := proto.MarshalText(buf, &textMessage{}); err != nil { + t.Fatalf("proto.MarshalText: %v", err) + } + s := buf.String() + if s != "custom" { + t.Errorf("Got %q, expected %q", s, "custom") + } +} +func TestMarshalTextNil(t *testing.T) { + want := "" + tests := []proto.Message{nil, (*pb.MyMessage)(nil)} + for i, test := range tests { + buf := new(bytes.Buffer) + if err := proto.MarshalText(buf, test); err != nil { + t.Fatal(err) + } + if got := buf.String(); got != want { + t.Errorf("%d: got %q want %q", i, got, want) + } + } +} + +func TestMarshalTextUnknownEnum(t *testing.T) { + // The Color enum only specifies values 0-2. + m := &pb.MyMessage{Bikeshed: pb.MyMessage_Color(3).Enum()} + got := m.String() + const want = `bikeshed:3 ` + if got != want { + t.Errorf("\n got %q\nwant %q", got, want) + } +} + +func BenchmarkMarshalTextBuffered(b *testing.B) { + buf := new(bytes.Buffer) + m := newTestMessage() + for i := 0; i < b.N; i++ { + buf.Reset() + proto.MarshalText(buf, m) + } +} + +func BenchmarkMarshalTextUnbuffered(b *testing.B) { + w := ioutil.Discard + m := newTestMessage() + for i := 0; i < b.N; i++ { + proto.MarshalText(w, m) + } +} + +func compact(src string) string { + // s/[ \n]+/ /g; s/ $//; + dst := make([]byte, len(src)) + space, comment := false, false + j := 0 + for i := 0; i < len(src); i++ { + if strings.HasPrefix(src[i:], "/*") { + comment = true + i++ + continue + } + if comment && strings.HasPrefix(src[i:], "*/") { + comment = false + i++ + continue + } + if comment { + continue + } + c := src[i] + if c == ' ' || c == '\n' { + space = true + continue + } + if j > 0 && (dst[j-1] == ':' || dst[j-1] == '<' || dst[j-1] == '{') { + space = false + } + if c == '{' { + space = false + } + if space { + dst[j] = ' ' + j++ + space = false + } + dst[j] = c + j++ + } + if space { + dst[j] = ' ' + j++ + } + return string(dst[0:j]) +} + +var compactText = compact(text) + +func TestCompactText(t *testing.T) { + s := proto.CompactTextString(newTestMessage()) + if s != compactText { + t.Errorf("Got:\n===\n%v===\nExpected:\n===\n%v\n===\n", s, compactText) + } +} + +func TestStringEscaping(t *testing.T) { + testCases := []struct { + in *pb.Strings + out string + }{ + { + // Test data from C++ test (TextFormatTest.StringEscape). + // Single divergence: we don't escape apostrophes. + &pb.Strings{StringField: proto.String("\"A string with ' characters \n and \r newlines and \t tabs and \001 slashes \\ and multiple spaces")}, + "string_field: \"\\\"A string with ' characters \\n and \\r newlines and \\t tabs and \\001 slashes \\\\ and multiple spaces\"\n", + }, + { + // Test data from the same C++ test. + &pb.Strings{StringField: proto.String("\350\260\267\346\255\214")}, + "string_field: \"\\350\\260\\267\\346\\255\\214\"\n", + }, + { + // Some UTF-8. + &pb.Strings{StringField: proto.String("\x00\x01\xff\x81")}, + `string_field: "\000\001\377\201"` + "\n", + }, + } + + for i, tc := range testCases { + var buf bytes.Buffer + if err := proto.MarshalText(&buf, tc.in); err != nil { + t.Errorf("proto.MarsalText: %v", err) + continue + } + s := buf.String() + if s != tc.out { + t.Errorf("#%d: Got:\n%s\nExpected:\n%s\n", i, s, tc.out) + continue + } + + // Check round-trip. + pb := new(pb.Strings) + if err := proto.UnmarshalText(s, pb); err != nil { + t.Errorf("#%d: UnmarshalText: %v", i, err) + continue + } + if !proto.Equal(pb, tc.in) { + t.Errorf("#%d: Round-trip failed:\nstart: %v\n end: %v", i, tc.in, pb) + } + } +} + +// A limitedWriter accepts some output before it fails. +// This is a proxy for something like a nearly-full or imminently-failing disk, +// or a network connection that is about to die. +type limitedWriter struct { + b bytes.Buffer + limit int +} + +var outOfSpace = errors.New("proto: insufficient space") + +func (w *limitedWriter) Write(p []byte) (n int, err error) { + var avail = w.limit - w.b.Len() + if avail <= 0 { + return 0, outOfSpace + } + if len(p) <= avail { + return w.b.Write(p) + } + n, _ = w.b.Write(p[:avail]) + return n, outOfSpace +} + +func TestMarshalTextFailing(t *testing.T) { + // Try lots of different sizes to exercise more error code-paths. + for lim := 0; lim < len(text); lim++ { + buf := new(limitedWriter) + buf.limit = lim + err := proto.MarshalText(buf, newTestMessage()) + // We expect a certain error, but also some partial results in the buffer. + if err != outOfSpace { + t.Errorf("Got:\n===\n%v===\nExpected:\n===\n%v===\n", err, outOfSpace) + } + s := buf.b.String() + x := text[:buf.limit] + if s != x { + t.Errorf("Got:\n===\n%v===\nExpected:\n===\n%v===\n", s, x) + } + } +} + +func TestFloats(t *testing.T) { + tests := []struct { + f float64 + want string + }{ + {0, "0"}, + {4.7, "4.7"}, + {math.Inf(1), "inf"}, + {math.Inf(-1), "-inf"}, + {math.NaN(), "nan"}, + } + for _, test := range tests { + msg := &pb.FloatingPoint{F: &test.f} + got := strings.TrimSpace(msg.String()) + want := `f:` + test.want + if got != want { + t.Errorf("f=%f: got %q, want %q", test.f, got, want) + } + } +} + +func TestRepeatedNilText(t *testing.T) { + m := &pb.MessageList{ + Message: []*pb.MessageList_Message{ + nil, + { + Name: proto.String("Horse"), + }, + nil, + }, + } + want := `Message +Message { + name: "Horse" +} +Message +` + if s := proto.MarshalTextString(m); s != want { + t.Errorf(" got: %s\nwant: %s", s, want) + } +} + +func TestProto3Text(t *testing.T) { + tests := []struct { + m proto.Message + want string + }{ + // zero message + {&proto3pb.Message{}, ``}, + // zero message except for an empty byte slice + {&proto3pb.Message{Data: []byte{}}, ``}, + // trivial case + {&proto3pb.Message{Name: "Rob", HeightInCm: 175}, `name:"Rob" height_in_cm:175`}, + // empty map + {&pb.MessageWithMap{}, ``}, + // non-empty map; map format is the same as a repeated struct, + // and they are sorted by key (numerically for numeric keys). + { + &pb.MessageWithMap{NameMapping: map[int32]string{ + -1: "Negatory", + 7: "Lucky", + 1234: "Feist", + 6345789: "Otis", + }}, + `name_mapping: ` + + `name_mapping: ` + + `name_mapping: ` + + `name_mapping:`, + }, + // map with nil value; not well-defined, but we shouldn't crash + { + &pb.MessageWithMap{MsgMapping: map[int64]*pb.FloatingPoint{7: nil}}, + `msg_mapping:`, + }, + } + for _, test := range tests { + got := strings.TrimSpace(test.m.String()) + if got != test.want { + t.Errorf("\n got %s\nwant %s", got, test.want) + } + } +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/Makefile b/Godeps/_workspace/src/github.com/golang/protobuf/proto/Makefile new file mode 100644 index 0000000000000..f1f06564a157d --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/Makefile @@ -0,0 +1,43 @@ +# Go support for Protocol Buffers - Google's data interchange format +# +# Copyright 2010 The Go Authors. All rights reserved. +# https://github.com/golang/protobuf +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are +# met: +# +# * Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +# * Redistributions in binary form must reproduce the above +# copyright notice, this list of conditions and the following disclaimer +# in the documentation and/or other materials provided with the +# distribution. +# * Neither the name of Google Inc. nor the names of its +# contributors may be used to endorse or promote products derived from +# this software without specific prior written permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +install: + go install + +test: install generate-test-pbs + go test + + +generate-test-pbs: + make install + make -C testdata + protoc --go_out=Mtestdata/test.proto=github.com/golang/protobuf/proto/testdata:. proto3_proto/proto3.proto + make diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/all_test.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/all_test.go new file mode 100644 index 0000000000000..b787d58aad049 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/all_test.go @@ -0,0 +1,2104 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "bytes" + "encoding/json" + "errors" + "fmt" + "math" + "math/rand" + "reflect" + "runtime/debug" + "strings" + "testing" + "time" + + . "github.com/golang/protobuf/proto" + . "github.com/golang/protobuf/proto/testdata" +) + +var globalO *Buffer + +func old() *Buffer { + if globalO == nil { + globalO = NewBuffer(nil) + } + globalO.Reset() + return globalO +} + +func equalbytes(b1, b2 []byte, t *testing.T) { + if len(b1) != len(b2) { + t.Errorf("wrong lengths: 2*%d != %d", len(b1), len(b2)) + return + } + for i := 0; i < len(b1); i++ { + if b1[i] != b2[i] { + t.Errorf("bad byte[%d]:%x %x: %s %s", i, b1[i], b2[i], b1, b2) + } + } +} + +func initGoTestField() *GoTestField { + f := new(GoTestField) + f.Label = String("label") + f.Type = String("type") + return f +} + +// These are all structurally equivalent but the tag numbers differ. +// (It's remarkable that required, optional, and repeated all have +// 8 letters.) +func initGoTest_RequiredGroup() *GoTest_RequiredGroup { + return &GoTest_RequiredGroup{ + RequiredField: String("required"), + } +} + +func initGoTest_OptionalGroup() *GoTest_OptionalGroup { + return &GoTest_OptionalGroup{ + RequiredField: String("optional"), + } +} + +func initGoTest_RepeatedGroup() *GoTest_RepeatedGroup { + return &GoTest_RepeatedGroup{ + RequiredField: String("repeated"), + } +} + +func initGoTest(setdefaults bool) *GoTest { + pb := new(GoTest) + if setdefaults { + pb.F_BoolDefaulted = Bool(Default_GoTest_F_BoolDefaulted) + pb.F_Int32Defaulted = Int32(Default_GoTest_F_Int32Defaulted) + pb.F_Int64Defaulted = Int64(Default_GoTest_F_Int64Defaulted) + pb.F_Fixed32Defaulted = Uint32(Default_GoTest_F_Fixed32Defaulted) + pb.F_Fixed64Defaulted = Uint64(Default_GoTest_F_Fixed64Defaulted) + pb.F_Uint32Defaulted = Uint32(Default_GoTest_F_Uint32Defaulted) + pb.F_Uint64Defaulted = Uint64(Default_GoTest_F_Uint64Defaulted) + pb.F_FloatDefaulted = Float32(Default_GoTest_F_FloatDefaulted) + pb.F_DoubleDefaulted = Float64(Default_GoTest_F_DoubleDefaulted) + pb.F_StringDefaulted = String(Default_GoTest_F_StringDefaulted) + pb.F_BytesDefaulted = Default_GoTest_F_BytesDefaulted + pb.F_Sint32Defaulted = Int32(Default_GoTest_F_Sint32Defaulted) + pb.F_Sint64Defaulted = Int64(Default_GoTest_F_Sint64Defaulted) + } + + pb.Kind = GoTest_TIME.Enum() + pb.RequiredField = initGoTestField() + pb.F_BoolRequired = Bool(true) + pb.F_Int32Required = Int32(3) + pb.F_Int64Required = Int64(6) + pb.F_Fixed32Required = Uint32(32) + pb.F_Fixed64Required = Uint64(64) + pb.F_Uint32Required = Uint32(3232) + pb.F_Uint64Required = Uint64(6464) + pb.F_FloatRequired = Float32(3232) + pb.F_DoubleRequired = Float64(6464) + pb.F_StringRequired = String("string") + pb.F_BytesRequired = []byte("bytes") + pb.F_Sint32Required = Int32(-32) + pb.F_Sint64Required = Int64(-64) + pb.Requiredgroup = initGoTest_RequiredGroup() + + return pb +} + +func fail(msg string, b *bytes.Buffer, s string, t *testing.T) { + data := b.Bytes() + ld := len(data) + ls := len(s) / 2 + + fmt.Printf("fail %s ld=%d ls=%d\n", msg, ld, ls) + + // find the interesting spot - n + n := ls + if ld < ls { + n = ld + } + j := 0 + for i := 0; i < n; i++ { + bs := hex(s[j])*16 + hex(s[j+1]) + j += 2 + if data[i] == bs { + continue + } + n = i + break + } + l := n - 10 + if l < 0 { + l = 0 + } + h := n + 10 + + // find the interesting spot - n + fmt.Printf("is[%d]:", l) + for i := l; i < h; i++ { + if i >= ld { + fmt.Printf(" --") + continue + } + fmt.Printf(" %.2x", data[i]) + } + fmt.Printf("\n") + + fmt.Printf("sb[%d]:", l) + for i := l; i < h; i++ { + if i >= ls { + fmt.Printf(" --") + continue + } + bs := hex(s[j])*16 + hex(s[j+1]) + j += 2 + fmt.Printf(" %.2x", bs) + } + fmt.Printf("\n") + + t.Fail() + + // t.Errorf("%s: \ngood: %s\nbad: %x", msg, s, b.Bytes()) + // Print the output in a partially-decoded format; can + // be helpful when updating the test. It produces the output + // that is pasted, with minor edits, into the argument to verify(). + // data := b.Bytes() + // nesting := 0 + // for b.Len() > 0 { + // start := len(data) - b.Len() + // var u uint64 + // u, err := DecodeVarint(b) + // if err != nil { + // fmt.Printf("decode error on varint:", err) + // return + // } + // wire := u & 0x7 + // tag := u >> 3 + // switch wire { + // case WireVarint: + // v, err := DecodeVarint(b) + // if err != nil { + // fmt.Printf("decode error on varint:", err) + // return + // } + // fmt.Printf("\t\t\"%x\" // field %d, encoding %d, value %d\n", + // data[start:len(data)-b.Len()], tag, wire, v) + // case WireFixed32: + // v, err := DecodeFixed32(b) + // if err != nil { + // fmt.Printf("decode error on fixed32:", err) + // return + // } + // fmt.Printf("\t\t\"%x\" // field %d, encoding %d, value %d\n", + // data[start:len(data)-b.Len()], tag, wire, v) + // case WireFixed64: + // v, err := DecodeFixed64(b) + // if err != nil { + // fmt.Printf("decode error on fixed64:", err) + // return + // } + // fmt.Printf("\t\t\"%x\" // field %d, encoding %d, value %d\n", + // data[start:len(data)-b.Len()], tag, wire, v) + // case WireBytes: + // nb, err := DecodeVarint(b) + // if err != nil { + // fmt.Printf("decode error on bytes:", err) + // return + // } + // after_tag := len(data) - b.Len() + // str := make([]byte, nb) + // _, err = b.Read(str) + // if err != nil { + // fmt.Printf("decode error on bytes:", err) + // return + // } + // fmt.Printf("\t\t\"%x\" \"%x\" // field %d, encoding %d (FIELD)\n", + // data[start:after_tag], str, tag, wire) + // case WireStartGroup: + // nesting++ + // fmt.Printf("\t\t\"%x\"\t\t// start group field %d level %d\n", + // data[start:len(data)-b.Len()], tag, nesting) + // case WireEndGroup: + // fmt.Printf("\t\t\"%x\"\t\t// end group field %d level %d\n", + // data[start:len(data)-b.Len()], tag, nesting) + // nesting-- + // default: + // fmt.Printf("unrecognized wire type %d\n", wire) + // return + // } + // } +} + +func hex(c uint8) uint8 { + if '0' <= c && c <= '9' { + return c - '0' + } + if 'a' <= c && c <= 'f' { + return 10 + c - 'a' + } + if 'A' <= c && c <= 'F' { + return 10 + c - 'A' + } + return 0 +} + +func equal(b []byte, s string, t *testing.T) bool { + if 2*len(b) != len(s) { + // fail(fmt.Sprintf("wrong lengths: 2*%d != %d", len(b), len(s)), b, s, t) + fmt.Printf("wrong lengths: 2*%d != %d\n", len(b), len(s)) + return false + } + for i, j := 0, 0; i < len(b); i, j = i+1, j+2 { + x := hex(s[j])*16 + hex(s[j+1]) + if b[i] != x { + // fail(fmt.Sprintf("bad byte[%d]:%x %x", i, b[i], x), b, s, t) + fmt.Printf("bad byte[%d]:%x %x", i, b[i], x) + return false + } + } + return true +} + +func overify(t *testing.T, pb *GoTest, expected string) { + o := old() + err := o.Marshal(pb) + if err != nil { + fmt.Printf("overify marshal-1 err = %v", err) + o.DebugPrint("", o.Bytes()) + t.Fatalf("expected = %s", expected) + } + if !equal(o.Bytes(), expected, t) { + o.DebugPrint("overify neq 1", o.Bytes()) + t.Fatalf("expected = %s", expected) + } + + // Now test Unmarshal by recreating the original buffer. + pbd := new(GoTest) + err = o.Unmarshal(pbd) + if err != nil { + t.Fatalf("overify unmarshal err = %v", err) + o.DebugPrint("", o.Bytes()) + t.Fatalf("string = %s", expected) + } + o.Reset() + err = o.Marshal(pbd) + if err != nil { + t.Errorf("overify marshal-2 err = %v", err) + o.DebugPrint("", o.Bytes()) + t.Fatalf("string = %s", expected) + } + if !equal(o.Bytes(), expected, t) { + o.DebugPrint("overify neq 2", o.Bytes()) + t.Fatalf("string = %s", expected) + } +} + +// Simple tests for numeric encode/decode primitives (varint, etc.) +func TestNumericPrimitives(t *testing.T) { + for i := uint64(0); i < 1e6; i += 111 { + o := old() + if o.EncodeVarint(i) != nil { + t.Error("EncodeVarint") + break + } + x, e := o.DecodeVarint() + if e != nil { + t.Fatal("DecodeVarint") + } + if x != i { + t.Fatal("varint decode fail:", i, x) + } + + o = old() + if o.EncodeFixed32(i) != nil { + t.Fatal("encFixed32") + } + x, e = o.DecodeFixed32() + if e != nil { + t.Fatal("decFixed32") + } + if x != i { + t.Fatal("fixed32 decode fail:", i, x) + } + + o = old() + if o.EncodeFixed64(i*1234567) != nil { + t.Error("encFixed64") + break + } + x, e = o.DecodeFixed64() + if e != nil { + t.Error("decFixed64") + break + } + if x != i*1234567 { + t.Error("fixed64 decode fail:", i*1234567, x) + break + } + + o = old() + i32 := int32(i - 12345) + if o.EncodeZigzag32(uint64(i32)) != nil { + t.Fatal("EncodeZigzag32") + } + x, e = o.DecodeZigzag32() + if e != nil { + t.Fatal("DecodeZigzag32") + } + if x != uint64(uint32(i32)) { + t.Fatal("zigzag32 decode fail:", i32, x) + } + + o = old() + i64 := int64(i - 12345) + if o.EncodeZigzag64(uint64(i64)) != nil { + t.Fatal("EncodeZigzag64") + } + x, e = o.DecodeZigzag64() + if e != nil { + t.Fatal("DecodeZigzag64") + } + if x != uint64(i64) { + t.Fatal("zigzag64 decode fail:", i64, x) + } + } +} + +// fakeMarshaler is a simple struct implementing Marshaler and Message interfaces. +type fakeMarshaler struct { + b []byte + err error +} + +func (f *fakeMarshaler) Marshal() ([]byte, error) { return f.b, f.err } +func (f *fakeMarshaler) String() string { return fmt.Sprintf("Bytes: %v Error: %v", f.b, f.err) } +func (f *fakeMarshaler) ProtoMessage() {} +func (f *fakeMarshaler) Reset() {} + +type msgWithFakeMarshaler struct { + M *fakeMarshaler `protobuf:"bytes,1,opt,name=fake"` +} + +func (m *msgWithFakeMarshaler) String() string { return CompactTextString(m) } +func (m *msgWithFakeMarshaler) ProtoMessage() {} +func (m *msgWithFakeMarshaler) Reset() {} + +// Simple tests for proto messages that implement the Marshaler interface. +func TestMarshalerEncoding(t *testing.T) { + tests := []struct { + name string + m Message + want []byte + wantErr error + }{ + { + name: "Marshaler that fails", + m: &fakeMarshaler{ + err: errors.New("some marshal err"), + b: []byte{5, 6, 7}, + }, + // Since there's an error, nothing should be written to buffer. + want: nil, + wantErr: errors.New("some marshal err"), + }, + { + name: "Marshaler that fails with RequiredNotSetError", + m: &msgWithFakeMarshaler{ + M: &fakeMarshaler{ + err: &RequiredNotSetError{}, + b: []byte{5, 6, 7}, + }, + }, + // Since there's an error that can be continued after, + // the buffer should be written. + want: []byte{ + 10, 3, // for &msgWithFakeMarshaler + 5, 6, 7, // for &fakeMarshaler + }, + wantErr: &RequiredNotSetError{}, + }, + { + name: "Marshaler that succeeds", + m: &fakeMarshaler{ + b: []byte{0, 1, 2, 3, 4, 127, 255}, + }, + want: []byte{0, 1, 2, 3, 4, 127, 255}, + wantErr: nil, + }, + } + for _, test := range tests { + b := NewBuffer(nil) + err := b.Marshal(test.m) + if _, ok := err.(*RequiredNotSetError); ok { + // We're not in package proto, so we can only assert the type in this case. + err = &RequiredNotSetError{} + } + if !reflect.DeepEqual(test.wantErr, err) { + t.Errorf("%s: got err %v wanted %v", test.name, err, test.wantErr) + } + if !reflect.DeepEqual(test.want, b.Bytes()) { + t.Errorf("%s: got bytes %v wanted %v", test.name, b.Bytes(), test.want) + } + } +} + +// Simple tests for bytes +func TestBytesPrimitives(t *testing.T) { + o := old() + bytes := []byte{'n', 'o', 'w', ' ', 'i', 's', ' ', 't', 'h', 'e', ' ', 't', 'i', 'm', 'e'} + if o.EncodeRawBytes(bytes) != nil { + t.Error("EncodeRawBytes") + } + decb, e := o.DecodeRawBytes(false) + if e != nil { + t.Error("DecodeRawBytes") + } + equalbytes(bytes, decb, t) +} + +// Simple tests for strings +func TestStringPrimitives(t *testing.T) { + o := old() + s := "now is the time" + if o.EncodeStringBytes(s) != nil { + t.Error("enc_string") + } + decs, e := o.DecodeStringBytes() + if e != nil { + t.Error("dec_string") + } + if s != decs { + t.Error("string encode/decode fail:", s, decs) + } +} + +// Do we catch the "required bit not set" case? +func TestRequiredBit(t *testing.T) { + o := old() + pb := new(GoTest) + err := o.Marshal(pb) + if err == nil { + t.Error("did not catch missing required fields") + } else if strings.Index(err.Error(), "Kind") < 0 { + t.Error("wrong error type:", err) + } +} + +// Check that all fields are nil. +// Clearly silly, and a residue from a more interesting test with an earlier, +// different initialization property, but it once caught a compiler bug so +// it lives. +func checkInitialized(pb *GoTest, t *testing.T) { + if pb.F_BoolDefaulted != nil { + t.Error("New or Reset did not set boolean:", *pb.F_BoolDefaulted) + } + if pb.F_Int32Defaulted != nil { + t.Error("New or Reset did not set int32:", *pb.F_Int32Defaulted) + } + if pb.F_Int64Defaulted != nil { + t.Error("New or Reset did not set int64:", *pb.F_Int64Defaulted) + } + if pb.F_Fixed32Defaulted != nil { + t.Error("New or Reset did not set fixed32:", *pb.F_Fixed32Defaulted) + } + if pb.F_Fixed64Defaulted != nil { + t.Error("New or Reset did not set fixed64:", *pb.F_Fixed64Defaulted) + } + if pb.F_Uint32Defaulted != nil { + t.Error("New or Reset did not set uint32:", *pb.F_Uint32Defaulted) + } + if pb.F_Uint64Defaulted != nil { + t.Error("New or Reset did not set uint64:", *pb.F_Uint64Defaulted) + } + if pb.F_FloatDefaulted != nil { + t.Error("New or Reset did not set float:", *pb.F_FloatDefaulted) + } + if pb.F_DoubleDefaulted != nil { + t.Error("New or Reset did not set double:", *pb.F_DoubleDefaulted) + } + if pb.F_StringDefaulted != nil { + t.Error("New or Reset did not set string:", *pb.F_StringDefaulted) + } + if pb.F_BytesDefaulted != nil { + t.Error("New or Reset did not set bytes:", string(pb.F_BytesDefaulted)) + } + if pb.F_Sint32Defaulted != nil { + t.Error("New or Reset did not set int32:", *pb.F_Sint32Defaulted) + } + if pb.F_Sint64Defaulted != nil { + t.Error("New or Reset did not set int64:", *pb.F_Sint64Defaulted) + } +} + +// Does Reset() reset? +func TestReset(t *testing.T) { + pb := initGoTest(true) + // muck with some values + pb.F_BoolDefaulted = Bool(false) + pb.F_Int32Defaulted = Int32(237) + pb.F_Int64Defaulted = Int64(12346) + pb.F_Fixed32Defaulted = Uint32(32000) + pb.F_Fixed64Defaulted = Uint64(666) + pb.F_Uint32Defaulted = Uint32(323232) + pb.F_Uint64Defaulted = nil + pb.F_FloatDefaulted = nil + pb.F_DoubleDefaulted = Float64(0) + pb.F_StringDefaulted = String("gotcha") + pb.F_BytesDefaulted = []byte("asdfasdf") + pb.F_Sint32Defaulted = Int32(123) + pb.F_Sint64Defaulted = Int64(789) + pb.Reset() + checkInitialized(pb, t) +} + +// All required fields set, no defaults provided. +func TestEncodeDecode1(t *testing.T) { + pb := initGoTest(false) + overify(t, pb, + "0807"+ // field 1, encoding 0, value 7 + "220d"+"0a056c6162656c120474797065"+ // field 4, encoding 2 (GoTestField) + "5001"+ // field 10, encoding 0, value 1 + "5803"+ // field 11, encoding 0, value 3 + "6006"+ // field 12, encoding 0, value 6 + "6d20000000"+ // field 13, encoding 5, value 0x20 + "714000000000000000"+ // field 14, encoding 1, value 0x40 + "78a019"+ // field 15, encoding 0, value 0xca0 = 3232 + "8001c032"+ // field 16, encoding 0, value 0x1940 = 6464 + "8d0100004a45"+ // field 17, encoding 5, value 3232.0 + "9101000000000040b940"+ // field 18, encoding 1, value 6464.0 + "9a0106"+"737472696e67"+ // field 19, encoding 2, string "string" + "b304"+ // field 70, encoding 3, start group + "ba0408"+"7265717569726564"+ // field 71, encoding 2, string "required" + "b404"+ // field 70, encoding 4, end group + "aa0605"+"6279746573"+ // field 101, encoding 2, string "bytes" + "b0063f"+ // field 102, encoding 0, 0x3f zigzag32 + "b8067f") // field 103, encoding 0, 0x7f zigzag64 +} + +// All required fields set, defaults provided. +func TestEncodeDecode2(t *testing.T) { + pb := initGoTest(true) + overify(t, pb, + "0807"+ // field 1, encoding 0, value 7 + "220d"+"0a056c6162656c120474797065"+ // field 4, encoding 2 (GoTestField) + "5001"+ // field 10, encoding 0, value 1 + "5803"+ // field 11, encoding 0, value 3 + "6006"+ // field 12, encoding 0, value 6 + "6d20000000"+ // field 13, encoding 5, value 32 + "714000000000000000"+ // field 14, encoding 1, value 64 + "78a019"+ // field 15, encoding 0, value 3232 + "8001c032"+ // field 16, encoding 0, value 6464 + "8d0100004a45"+ // field 17, encoding 5, value 3232.0 + "9101000000000040b940"+ // field 18, encoding 1, value 6464.0 + "9a0106"+"737472696e67"+ // field 19, encoding 2 string "string" + "c00201"+ // field 40, encoding 0, value 1 + "c80220"+ // field 41, encoding 0, value 32 + "d00240"+ // field 42, encoding 0, value 64 + "dd0240010000"+ // field 43, encoding 5, value 320 + "e1028002000000000000"+ // field 44, encoding 1, value 640 + "e8028019"+ // field 45, encoding 0, value 3200 + "f0028032"+ // field 46, encoding 0, value 6400 + "fd02e0659948"+ // field 47, encoding 5, value 314159.0 + "81030000000050971041"+ // field 48, encoding 1, value 271828.0 + "8a0310"+"68656c6c6f2c2022776f726c6421220a"+ // field 49, encoding 2 string "hello, \"world!\"\n" + "b304"+ // start group field 70 level 1 + "ba0408"+"7265717569726564"+ // field 71, encoding 2, string "required" + "b404"+ // end group field 70 level 1 + "aa0605"+"6279746573"+ // field 101, encoding 2 string "bytes" + "b0063f"+ // field 102, encoding 0, 0x3f zigzag32 + "b8067f"+ // field 103, encoding 0, 0x7f zigzag64 + "8a1907"+"4269676e6f7365"+ // field 401, encoding 2, string "Bignose" + "90193f"+ // field 402, encoding 0, value 63 + "98197f") // field 403, encoding 0, value 127 + +} + +// All default fields set to their default value by hand +func TestEncodeDecode3(t *testing.T) { + pb := initGoTest(false) + pb.F_BoolDefaulted = Bool(true) + pb.F_Int32Defaulted = Int32(32) + pb.F_Int64Defaulted = Int64(64) + pb.F_Fixed32Defaulted = Uint32(320) + pb.F_Fixed64Defaulted = Uint64(640) + pb.F_Uint32Defaulted = Uint32(3200) + pb.F_Uint64Defaulted = Uint64(6400) + pb.F_FloatDefaulted = Float32(314159) + pb.F_DoubleDefaulted = Float64(271828) + pb.F_StringDefaulted = String("hello, \"world!\"\n") + pb.F_BytesDefaulted = []byte("Bignose") + pb.F_Sint32Defaulted = Int32(-32) + pb.F_Sint64Defaulted = Int64(-64) + + overify(t, pb, + "0807"+ // field 1, encoding 0, value 7 + "220d"+"0a056c6162656c120474797065"+ // field 4, encoding 2 (GoTestField) + "5001"+ // field 10, encoding 0, value 1 + "5803"+ // field 11, encoding 0, value 3 + "6006"+ // field 12, encoding 0, value 6 + "6d20000000"+ // field 13, encoding 5, value 32 + "714000000000000000"+ // field 14, encoding 1, value 64 + "78a019"+ // field 15, encoding 0, value 3232 + "8001c032"+ // field 16, encoding 0, value 6464 + "8d0100004a45"+ // field 17, encoding 5, value 3232.0 + "9101000000000040b940"+ // field 18, encoding 1, value 6464.0 + "9a0106"+"737472696e67"+ // field 19, encoding 2 string "string" + "c00201"+ // field 40, encoding 0, value 1 + "c80220"+ // field 41, encoding 0, value 32 + "d00240"+ // field 42, encoding 0, value 64 + "dd0240010000"+ // field 43, encoding 5, value 320 + "e1028002000000000000"+ // field 44, encoding 1, value 640 + "e8028019"+ // field 45, encoding 0, value 3200 + "f0028032"+ // field 46, encoding 0, value 6400 + "fd02e0659948"+ // field 47, encoding 5, value 314159.0 + "81030000000050971041"+ // field 48, encoding 1, value 271828.0 + "8a0310"+"68656c6c6f2c2022776f726c6421220a"+ // field 49, encoding 2 string "hello, \"world!\"\n" + "b304"+ // start group field 70 level 1 + "ba0408"+"7265717569726564"+ // field 71, encoding 2, string "required" + "b404"+ // end group field 70 level 1 + "aa0605"+"6279746573"+ // field 101, encoding 2 string "bytes" + "b0063f"+ // field 102, encoding 0, 0x3f zigzag32 + "b8067f"+ // field 103, encoding 0, 0x7f zigzag64 + "8a1907"+"4269676e6f7365"+ // field 401, encoding 2, string "Bignose" + "90193f"+ // field 402, encoding 0, value 63 + "98197f") // field 403, encoding 0, value 127 + +} + +// All required fields set, defaults provided, all non-defaulted optional fields have values. +func TestEncodeDecode4(t *testing.T) { + pb := initGoTest(true) + pb.Table = String("hello") + pb.Param = Int32(7) + pb.OptionalField = initGoTestField() + pb.F_BoolOptional = Bool(true) + pb.F_Int32Optional = Int32(32) + pb.F_Int64Optional = Int64(64) + pb.F_Fixed32Optional = Uint32(3232) + pb.F_Fixed64Optional = Uint64(6464) + pb.F_Uint32Optional = Uint32(323232) + pb.F_Uint64Optional = Uint64(646464) + pb.F_FloatOptional = Float32(32.) + pb.F_DoubleOptional = Float64(64.) + pb.F_StringOptional = String("hello") + pb.F_BytesOptional = []byte("Bignose") + pb.F_Sint32Optional = Int32(-32) + pb.F_Sint64Optional = Int64(-64) + pb.Optionalgroup = initGoTest_OptionalGroup() + + overify(t, pb, + "0807"+ // field 1, encoding 0, value 7 + "1205"+"68656c6c6f"+ // field 2, encoding 2, string "hello" + "1807"+ // field 3, encoding 0, value 7 + "220d"+"0a056c6162656c120474797065"+ // field 4, encoding 2 (GoTestField) + "320d"+"0a056c6162656c120474797065"+ // field 6, encoding 2 (GoTestField) + "5001"+ // field 10, encoding 0, value 1 + "5803"+ // field 11, encoding 0, value 3 + "6006"+ // field 12, encoding 0, value 6 + "6d20000000"+ // field 13, encoding 5, value 32 + "714000000000000000"+ // field 14, encoding 1, value 64 + "78a019"+ // field 15, encoding 0, value 3232 + "8001c032"+ // field 16, encoding 0, value 6464 + "8d0100004a45"+ // field 17, encoding 5, value 3232.0 + "9101000000000040b940"+ // field 18, encoding 1, value 6464.0 + "9a0106"+"737472696e67"+ // field 19, encoding 2 string "string" + "f00101"+ // field 30, encoding 0, value 1 + "f80120"+ // field 31, encoding 0, value 32 + "800240"+ // field 32, encoding 0, value 64 + "8d02a00c0000"+ // field 33, encoding 5, value 3232 + "91024019000000000000"+ // field 34, encoding 1, value 6464 + "9802a0dd13"+ // field 35, encoding 0, value 323232 + "a002c0ba27"+ // field 36, encoding 0, value 646464 + "ad0200000042"+ // field 37, encoding 5, value 32.0 + "b1020000000000005040"+ // field 38, encoding 1, value 64.0 + "ba0205"+"68656c6c6f"+ // field 39, encoding 2, string "hello" + "c00201"+ // field 40, encoding 0, value 1 + "c80220"+ // field 41, encoding 0, value 32 + "d00240"+ // field 42, encoding 0, value 64 + "dd0240010000"+ // field 43, encoding 5, value 320 + "e1028002000000000000"+ // field 44, encoding 1, value 640 + "e8028019"+ // field 45, encoding 0, value 3200 + "f0028032"+ // field 46, encoding 0, value 6400 + "fd02e0659948"+ // field 47, encoding 5, value 314159.0 + "81030000000050971041"+ // field 48, encoding 1, value 271828.0 + "8a0310"+"68656c6c6f2c2022776f726c6421220a"+ // field 49, encoding 2 string "hello, \"world!\"\n" + "b304"+ // start group field 70 level 1 + "ba0408"+"7265717569726564"+ // field 71, encoding 2, string "required" + "b404"+ // end group field 70 level 1 + "d305"+ // start group field 90 level 1 + "da0508"+"6f7074696f6e616c"+ // field 91, encoding 2, string "optional" + "d405"+ // end group field 90 level 1 + "aa0605"+"6279746573"+ // field 101, encoding 2 string "bytes" + "b0063f"+ // field 102, encoding 0, 0x3f zigzag32 + "b8067f"+ // field 103, encoding 0, 0x7f zigzag64 + "ea1207"+"4269676e6f7365"+ // field 301, encoding 2, string "Bignose" + "f0123f"+ // field 302, encoding 0, value 63 + "f8127f"+ // field 303, encoding 0, value 127 + "8a1907"+"4269676e6f7365"+ // field 401, encoding 2, string "Bignose" + "90193f"+ // field 402, encoding 0, value 63 + "98197f") // field 403, encoding 0, value 127 + +} + +// All required fields set, defaults provided, all repeated fields given two values. +func TestEncodeDecode5(t *testing.T) { + pb := initGoTest(true) + pb.RepeatedField = []*GoTestField{initGoTestField(), initGoTestField()} + pb.F_BoolRepeated = []bool{false, true} + pb.F_Int32Repeated = []int32{32, 33} + pb.F_Int64Repeated = []int64{64, 65} + pb.F_Fixed32Repeated = []uint32{3232, 3333} + pb.F_Fixed64Repeated = []uint64{6464, 6565} + pb.F_Uint32Repeated = []uint32{323232, 333333} + pb.F_Uint64Repeated = []uint64{646464, 656565} + pb.F_FloatRepeated = []float32{32., 33.} + pb.F_DoubleRepeated = []float64{64., 65.} + pb.F_StringRepeated = []string{"hello", "sailor"} + pb.F_BytesRepeated = [][]byte{[]byte("big"), []byte("nose")} + pb.F_Sint32Repeated = []int32{32, -32} + pb.F_Sint64Repeated = []int64{64, -64} + pb.Repeatedgroup = []*GoTest_RepeatedGroup{initGoTest_RepeatedGroup(), initGoTest_RepeatedGroup()} + + overify(t, pb, + "0807"+ // field 1, encoding 0, value 7 + "220d"+"0a056c6162656c120474797065"+ // field 4, encoding 2 (GoTestField) + "2a0d"+"0a056c6162656c120474797065"+ // field 5, encoding 2 (GoTestField) + "2a0d"+"0a056c6162656c120474797065"+ // field 5, encoding 2 (GoTestField) + "5001"+ // field 10, encoding 0, value 1 + "5803"+ // field 11, encoding 0, value 3 + "6006"+ // field 12, encoding 0, value 6 + "6d20000000"+ // field 13, encoding 5, value 32 + "714000000000000000"+ // field 14, encoding 1, value 64 + "78a019"+ // field 15, encoding 0, value 3232 + "8001c032"+ // field 16, encoding 0, value 6464 + "8d0100004a45"+ // field 17, encoding 5, value 3232.0 + "9101000000000040b940"+ // field 18, encoding 1, value 6464.0 + "9a0106"+"737472696e67"+ // field 19, encoding 2 string "string" + "a00100"+ // field 20, encoding 0, value 0 + "a00101"+ // field 20, encoding 0, value 1 + "a80120"+ // field 21, encoding 0, value 32 + "a80121"+ // field 21, encoding 0, value 33 + "b00140"+ // field 22, encoding 0, value 64 + "b00141"+ // field 22, encoding 0, value 65 + "bd01a00c0000"+ // field 23, encoding 5, value 3232 + "bd01050d0000"+ // field 23, encoding 5, value 3333 + "c1014019000000000000"+ // field 24, encoding 1, value 6464 + "c101a519000000000000"+ // field 24, encoding 1, value 6565 + "c801a0dd13"+ // field 25, encoding 0, value 323232 + "c80195ac14"+ // field 25, encoding 0, value 333333 + "d001c0ba27"+ // field 26, encoding 0, value 646464 + "d001b58928"+ // field 26, encoding 0, value 656565 + "dd0100000042"+ // field 27, encoding 5, value 32.0 + "dd0100000442"+ // field 27, encoding 5, value 33.0 + "e1010000000000005040"+ // field 28, encoding 1, value 64.0 + "e1010000000000405040"+ // field 28, encoding 1, value 65.0 + "ea0105"+"68656c6c6f"+ // field 29, encoding 2, string "hello" + "ea0106"+"7361696c6f72"+ // field 29, encoding 2, string "sailor" + "c00201"+ // field 40, encoding 0, value 1 + "c80220"+ // field 41, encoding 0, value 32 + "d00240"+ // field 42, encoding 0, value 64 + "dd0240010000"+ // field 43, encoding 5, value 320 + "e1028002000000000000"+ // field 44, encoding 1, value 640 + "e8028019"+ // field 45, encoding 0, value 3200 + "f0028032"+ // field 46, encoding 0, value 6400 + "fd02e0659948"+ // field 47, encoding 5, value 314159.0 + "81030000000050971041"+ // field 48, encoding 1, value 271828.0 + "8a0310"+"68656c6c6f2c2022776f726c6421220a"+ // field 49, encoding 2 string "hello, \"world!\"\n" + "b304"+ // start group field 70 level 1 + "ba0408"+"7265717569726564"+ // field 71, encoding 2, string "required" + "b404"+ // end group field 70 level 1 + "8305"+ // start group field 80 level 1 + "8a0508"+"7265706561746564"+ // field 81, encoding 2, string "repeated" + "8405"+ // end group field 80 level 1 + "8305"+ // start group field 80 level 1 + "8a0508"+"7265706561746564"+ // field 81, encoding 2, string "repeated" + "8405"+ // end group field 80 level 1 + "aa0605"+"6279746573"+ // field 101, encoding 2 string "bytes" + "b0063f"+ // field 102, encoding 0, 0x3f zigzag32 + "b8067f"+ // field 103, encoding 0, 0x7f zigzag64 + "ca0c03"+"626967"+ // field 201, encoding 2, string "big" + "ca0c04"+"6e6f7365"+ // field 201, encoding 2, string "nose" + "d00c40"+ // field 202, encoding 0, value 32 + "d00c3f"+ // field 202, encoding 0, value -32 + "d80c8001"+ // field 203, encoding 0, value 64 + "d80c7f"+ // field 203, encoding 0, value -64 + "8a1907"+"4269676e6f7365"+ // field 401, encoding 2, string "Bignose" + "90193f"+ // field 402, encoding 0, value 63 + "98197f") // field 403, encoding 0, value 127 + +} + +// All required fields set, all packed repeated fields given two values. +func TestEncodeDecode6(t *testing.T) { + pb := initGoTest(false) + pb.F_BoolRepeatedPacked = []bool{false, true} + pb.F_Int32RepeatedPacked = []int32{32, 33} + pb.F_Int64RepeatedPacked = []int64{64, 65} + pb.F_Fixed32RepeatedPacked = []uint32{3232, 3333} + pb.F_Fixed64RepeatedPacked = []uint64{6464, 6565} + pb.F_Uint32RepeatedPacked = []uint32{323232, 333333} + pb.F_Uint64RepeatedPacked = []uint64{646464, 656565} + pb.F_FloatRepeatedPacked = []float32{32., 33.} + pb.F_DoubleRepeatedPacked = []float64{64., 65.} + pb.F_Sint32RepeatedPacked = []int32{32, -32} + pb.F_Sint64RepeatedPacked = []int64{64, -64} + + overify(t, pb, + "0807"+ // field 1, encoding 0, value 7 + "220d"+"0a056c6162656c120474797065"+ // field 4, encoding 2 (GoTestField) + "5001"+ // field 10, encoding 0, value 1 + "5803"+ // field 11, encoding 0, value 3 + "6006"+ // field 12, encoding 0, value 6 + "6d20000000"+ // field 13, encoding 5, value 32 + "714000000000000000"+ // field 14, encoding 1, value 64 + "78a019"+ // field 15, encoding 0, value 3232 + "8001c032"+ // field 16, encoding 0, value 6464 + "8d0100004a45"+ // field 17, encoding 5, value 3232.0 + "9101000000000040b940"+ // field 18, encoding 1, value 6464.0 + "9a0106"+"737472696e67"+ // field 19, encoding 2 string "string" + "9203020001"+ // field 50, encoding 2, 2 bytes, value 0, value 1 + "9a03022021"+ // field 51, encoding 2, 2 bytes, value 32, value 33 + "a203024041"+ // field 52, encoding 2, 2 bytes, value 64, value 65 + "aa0308"+ // field 53, encoding 2, 8 bytes + "a00c0000050d0000"+ // value 3232, value 3333 + "b20310"+ // field 54, encoding 2, 16 bytes + "4019000000000000a519000000000000"+ // value 6464, value 6565 + "ba0306"+ // field 55, encoding 2, 6 bytes + "a0dd1395ac14"+ // value 323232, value 333333 + "c20306"+ // field 56, encoding 2, 6 bytes + "c0ba27b58928"+ // value 646464, value 656565 + "ca0308"+ // field 57, encoding 2, 8 bytes + "0000004200000442"+ // value 32.0, value 33.0 + "d20310"+ // field 58, encoding 2, 16 bytes + "00000000000050400000000000405040"+ // value 64.0, value 65.0 + "b304"+ // start group field 70 level 1 + "ba0408"+"7265717569726564"+ // field 71, encoding 2, string "required" + "b404"+ // end group field 70 level 1 + "aa0605"+"6279746573"+ // field 101, encoding 2 string "bytes" + "b0063f"+ // field 102, encoding 0, 0x3f zigzag32 + "b8067f"+ // field 103, encoding 0, 0x7f zigzag64 + "b21f02"+ // field 502, encoding 2, 2 bytes + "403f"+ // value 32, value -32 + "ba1f03"+ // field 503, encoding 2, 3 bytes + "80017f") // value 64, value -64 +} + +// Test that we can encode empty bytes fields. +func TestEncodeDecodeBytes1(t *testing.T) { + pb := initGoTest(false) + + // Create our bytes + pb.F_BytesRequired = []byte{} + pb.F_BytesRepeated = [][]byte{{}} + pb.F_BytesOptional = []byte{} + + d, err := Marshal(pb) + if err != nil { + t.Error(err) + } + + pbd := new(GoTest) + if err := Unmarshal(d, pbd); err != nil { + t.Error(err) + } + + if pbd.F_BytesRequired == nil || len(pbd.F_BytesRequired) != 0 { + t.Error("required empty bytes field is incorrect") + } + if pbd.F_BytesRepeated == nil || len(pbd.F_BytesRepeated) == 1 && pbd.F_BytesRepeated[0] == nil { + t.Error("repeated empty bytes field is incorrect") + } + if pbd.F_BytesOptional == nil || len(pbd.F_BytesOptional) != 0 { + t.Error("optional empty bytes field is incorrect") + } +} + +// Test that we encode nil-valued fields of a repeated bytes field correctly. +// Since entries in a repeated field cannot be nil, nil must mean empty value. +func TestEncodeDecodeBytes2(t *testing.T) { + pb := initGoTest(false) + + // Create our bytes + pb.F_BytesRepeated = [][]byte{nil} + + d, err := Marshal(pb) + if err != nil { + t.Error(err) + } + + pbd := new(GoTest) + if err := Unmarshal(d, pbd); err != nil { + t.Error(err) + } + + if len(pbd.F_BytesRepeated) != 1 || pbd.F_BytesRepeated[0] == nil { + t.Error("Unexpected value for repeated bytes field") + } +} + +// All required fields set, defaults provided, all repeated fields given two values. +func TestSkippingUnrecognizedFields(t *testing.T) { + o := old() + pb := initGoTestField() + + // Marshal it normally. + o.Marshal(pb) + + // Now new a GoSkipTest record. + skip := &GoSkipTest{ + SkipInt32: Int32(32), + SkipFixed32: Uint32(3232), + SkipFixed64: Uint64(6464), + SkipString: String("skipper"), + Skipgroup: &GoSkipTest_SkipGroup{ + GroupInt32: Int32(75), + GroupString: String("wxyz"), + }, + } + + // Marshal it into same buffer. + o.Marshal(skip) + + pbd := new(GoTestField) + o.Unmarshal(pbd) + + // The __unrecognized field should be a marshaling of GoSkipTest + skipd := new(GoSkipTest) + + o.SetBuf(pbd.XXX_unrecognized) + o.Unmarshal(skipd) + + if *skipd.SkipInt32 != *skip.SkipInt32 { + t.Error("skip int32", skipd.SkipInt32) + } + if *skipd.SkipFixed32 != *skip.SkipFixed32 { + t.Error("skip fixed32", skipd.SkipFixed32) + } + if *skipd.SkipFixed64 != *skip.SkipFixed64 { + t.Error("skip fixed64", skipd.SkipFixed64) + } + if *skipd.SkipString != *skip.SkipString { + t.Error("skip string", *skipd.SkipString) + } + if *skipd.Skipgroup.GroupInt32 != *skip.Skipgroup.GroupInt32 { + t.Error("skip group int32", skipd.Skipgroup.GroupInt32) + } + if *skipd.Skipgroup.GroupString != *skip.Skipgroup.GroupString { + t.Error("skip group string", *skipd.Skipgroup.GroupString) + } +} + +// Check that unrecognized fields of a submessage are preserved. +func TestSubmessageUnrecognizedFields(t *testing.T) { + nm := &NewMessage{ + Nested: &NewMessage_Nested{ + Name: String("Nigel"), + FoodGroup: String("carbs"), + }, + } + b, err := Marshal(nm) + if err != nil { + t.Fatalf("Marshal of NewMessage: %v", err) + } + + // Unmarshal into an OldMessage. + om := new(OldMessage) + if err := Unmarshal(b, om); err != nil { + t.Fatalf("Unmarshal to OldMessage: %v", err) + } + exp := &OldMessage{ + Nested: &OldMessage_Nested{ + Name: String("Nigel"), + // normal protocol buffer users should not do this + XXX_unrecognized: []byte("\x12\x05carbs"), + }, + } + if !Equal(om, exp) { + t.Errorf("om = %v, want %v", om, exp) + } + + // Clone the OldMessage. + om = Clone(om).(*OldMessage) + if !Equal(om, exp) { + t.Errorf("Clone(om) = %v, want %v", om, exp) + } + + // Marshal the OldMessage, then unmarshal it into an empty NewMessage. + if b, err = Marshal(om); err != nil { + t.Fatalf("Marshal of OldMessage: %v", err) + } + t.Logf("Marshal(%v) -> %q", om, b) + nm2 := new(NewMessage) + if err := Unmarshal(b, nm2); err != nil { + t.Fatalf("Unmarshal to NewMessage: %v", err) + } + if !Equal(nm, nm2) { + t.Errorf("NewMessage round-trip: %v => %v", nm, nm2) + } +} + +// Check that an int32 field can be upgraded to an int64 field. +func TestNegativeInt32(t *testing.T) { + om := &OldMessage{ + Num: Int32(-1), + } + b, err := Marshal(om) + if err != nil { + t.Fatalf("Marshal of OldMessage: %v", err) + } + + // Check the size. It should be 11 bytes; + // 1 for the field/wire type, and 10 for the negative number. + if len(b) != 11 { + t.Errorf("%v marshaled as %q, wanted 11 bytes", om, b) + } + + // Unmarshal into a NewMessage. + nm := new(NewMessage) + if err := Unmarshal(b, nm); err != nil { + t.Fatalf("Unmarshal to NewMessage: %v", err) + } + want := &NewMessage{ + Num: Int64(-1), + } + if !Equal(nm, want) { + t.Errorf("nm = %v, want %v", nm, want) + } +} + +// Check that we can grow an array (repeated field) to have many elements. +// This test doesn't depend only on our encoding; for variety, it makes sure +// we create, encode, and decode the correct contents explicitly. It's therefore +// a bit messier. +// This test also uses (and hence tests) the Marshal/Unmarshal functions +// instead of the methods. +func TestBigRepeated(t *testing.T) { + pb := initGoTest(true) + + // Create the arrays + const N = 50 // Internally the library starts much smaller. + pb.Repeatedgroup = make([]*GoTest_RepeatedGroup, N) + pb.F_Sint64Repeated = make([]int64, N) + pb.F_Sint32Repeated = make([]int32, N) + pb.F_BytesRepeated = make([][]byte, N) + pb.F_StringRepeated = make([]string, N) + pb.F_DoubleRepeated = make([]float64, N) + pb.F_FloatRepeated = make([]float32, N) + pb.F_Uint64Repeated = make([]uint64, N) + pb.F_Uint32Repeated = make([]uint32, N) + pb.F_Fixed64Repeated = make([]uint64, N) + pb.F_Fixed32Repeated = make([]uint32, N) + pb.F_Int64Repeated = make([]int64, N) + pb.F_Int32Repeated = make([]int32, N) + pb.F_BoolRepeated = make([]bool, N) + pb.RepeatedField = make([]*GoTestField, N) + + // Fill in the arrays with checkable values. + igtf := initGoTestField() + igtrg := initGoTest_RepeatedGroup() + for i := 0; i < N; i++ { + pb.Repeatedgroup[i] = igtrg + pb.F_Sint64Repeated[i] = int64(i) + pb.F_Sint32Repeated[i] = int32(i) + s := fmt.Sprint(i) + pb.F_BytesRepeated[i] = []byte(s) + pb.F_StringRepeated[i] = s + pb.F_DoubleRepeated[i] = float64(i) + pb.F_FloatRepeated[i] = float32(i) + pb.F_Uint64Repeated[i] = uint64(i) + pb.F_Uint32Repeated[i] = uint32(i) + pb.F_Fixed64Repeated[i] = uint64(i) + pb.F_Fixed32Repeated[i] = uint32(i) + pb.F_Int64Repeated[i] = int64(i) + pb.F_Int32Repeated[i] = int32(i) + pb.F_BoolRepeated[i] = i%2 == 0 + pb.RepeatedField[i] = igtf + } + + // Marshal. + buf, _ := Marshal(pb) + + // Now test Unmarshal by recreating the original buffer. + pbd := new(GoTest) + Unmarshal(buf, pbd) + + // Check the checkable values + for i := uint64(0); i < N; i++ { + if pbd.Repeatedgroup[i] == nil { // TODO: more checking? + t.Error("pbd.Repeatedgroup bad") + } + var x uint64 + x = uint64(pbd.F_Sint64Repeated[i]) + if x != i { + t.Error("pbd.F_Sint64Repeated bad", x, i) + } + x = uint64(pbd.F_Sint32Repeated[i]) + if x != i { + t.Error("pbd.F_Sint32Repeated bad", x, i) + } + s := fmt.Sprint(i) + equalbytes(pbd.F_BytesRepeated[i], []byte(s), t) + if pbd.F_StringRepeated[i] != s { + t.Error("pbd.F_Sint32Repeated bad", pbd.F_StringRepeated[i], i) + } + x = uint64(pbd.F_DoubleRepeated[i]) + if x != i { + t.Error("pbd.F_DoubleRepeated bad", x, i) + } + x = uint64(pbd.F_FloatRepeated[i]) + if x != i { + t.Error("pbd.F_FloatRepeated bad", x, i) + } + x = pbd.F_Uint64Repeated[i] + if x != i { + t.Error("pbd.F_Uint64Repeated bad", x, i) + } + x = uint64(pbd.F_Uint32Repeated[i]) + if x != i { + t.Error("pbd.F_Uint32Repeated bad", x, i) + } + x = pbd.F_Fixed64Repeated[i] + if x != i { + t.Error("pbd.F_Fixed64Repeated bad", x, i) + } + x = uint64(pbd.F_Fixed32Repeated[i]) + if x != i { + t.Error("pbd.F_Fixed32Repeated bad", x, i) + } + x = uint64(pbd.F_Int64Repeated[i]) + if x != i { + t.Error("pbd.F_Int64Repeated bad", x, i) + } + x = uint64(pbd.F_Int32Repeated[i]) + if x != i { + t.Error("pbd.F_Int32Repeated bad", x, i) + } + if pbd.F_BoolRepeated[i] != (i%2 == 0) { + t.Error("pbd.F_BoolRepeated bad", x, i) + } + if pbd.RepeatedField[i] == nil { // TODO: more checking? + t.Error("pbd.RepeatedField bad") + } + } +} + +// Verify we give a useful message when decoding to the wrong structure type. +func TestTypeMismatch(t *testing.T) { + pb1 := initGoTest(true) + + // Marshal + o := old() + o.Marshal(pb1) + + // Now Unmarshal it to the wrong type. + pb2 := initGoTestField() + err := o.Unmarshal(pb2) + if err == nil { + t.Error("expected error, got no error") + } else if !strings.Contains(err.Error(), "bad wiretype") { + t.Error("expected bad wiretype error, got", err) + } +} + +func encodeDecode(t *testing.T, in, out Message, msg string) { + buf, err := Marshal(in) + if err != nil { + t.Fatalf("failed marshaling %v: %v", msg, err) + } + if err := Unmarshal(buf, out); err != nil { + t.Fatalf("failed unmarshaling %v: %v", msg, err) + } +} + +func TestPackedNonPackedDecoderSwitching(t *testing.T) { + np, p := new(NonPackedTest), new(PackedTest) + + // non-packed -> packed + np.A = []int32{0, 1, 1, 2, 3, 5} + encodeDecode(t, np, p, "non-packed -> packed") + if !reflect.DeepEqual(np.A, p.B) { + t.Errorf("failed non-packed -> packed; np.A=%+v, p.B=%+v", np.A, p.B) + } + + // packed -> non-packed + np.Reset() + p.B = []int32{3, 1, 4, 1, 5, 9} + encodeDecode(t, p, np, "packed -> non-packed") + if !reflect.DeepEqual(p.B, np.A) { + t.Errorf("failed packed -> non-packed; p.B=%+v, np.A=%+v", p.B, np.A) + } +} + +func TestProto1RepeatedGroup(t *testing.T) { + pb := &MessageList{ + Message: []*MessageList_Message{ + { + Name: String("blah"), + Count: Int32(7), + }, + // NOTE: pb.Message[1] is a nil + nil, + }, + } + + o := old() + err := o.Marshal(pb) + if err == nil || !strings.Contains(err.Error(), "repeated field Message has nil") { + t.Fatalf("unexpected or no error when marshaling: %v", err) + } +} + +// Test that enums work. Checks for a bug introduced by making enums +// named types instead of int32: newInt32FromUint64 would crash with +// a type mismatch in reflect.PointTo. +func TestEnum(t *testing.T) { + pb := new(GoEnum) + pb.Foo = FOO_FOO1.Enum() + o := old() + if err := o.Marshal(pb); err != nil { + t.Fatal("error encoding enum:", err) + } + pb1 := new(GoEnum) + if err := o.Unmarshal(pb1); err != nil { + t.Fatal("error decoding enum:", err) + } + if *pb1.Foo != FOO_FOO1 { + t.Error("expected 7 but got ", *pb1.Foo) + } +} + +// Enum types have String methods. Check that enum fields can be printed. +// We don't care what the value actually is, just as long as it doesn't crash. +func TestPrintingNilEnumFields(t *testing.T) { + pb := new(GoEnum) + fmt.Sprintf("%+v", pb) +} + +// Verify that absent required fields cause Marshal/Unmarshal to return errors. +func TestRequiredFieldEnforcement(t *testing.T) { + pb := new(GoTestField) + _, err := Marshal(pb) + if err == nil { + t.Error("marshal: expected error, got nil") + } else if strings.Index(err.Error(), "Label") < 0 { + t.Errorf("marshal: bad error type: %v", err) + } + + // A slightly sneaky, yet valid, proto. It encodes the same required field twice, + // so simply counting the required fields is insufficient. + // field 1, encoding 2, value "hi" + buf := []byte("\x0A\x02hi\x0A\x02hi") + err = Unmarshal(buf, pb) + if err == nil { + t.Error("unmarshal: expected error, got nil") + } else if strings.Index(err.Error(), "{Unknown}") < 0 { + t.Errorf("unmarshal: bad error type: %v", err) + } +} + +func TestTypedNilMarshal(t *testing.T) { + // A typed nil should return ErrNil and not crash. + _, err := Marshal((*GoEnum)(nil)) + if err != ErrNil { + t.Errorf("Marshal: got err %v, want ErrNil", err) + } +} + +// A type that implements the Marshaler interface, but is not nillable. +type nonNillableInt uint64 + +func (nni nonNillableInt) Marshal() ([]byte, error) { + return EncodeVarint(uint64(nni)), nil +} + +type NNIMessage struct { + nni nonNillableInt +} + +func (*NNIMessage) Reset() {} +func (*NNIMessage) String() string { return "" } +func (*NNIMessage) ProtoMessage() {} + +// A type that implements the Marshaler interface and is nillable. +type nillableMessage struct { + x uint64 +} + +func (nm *nillableMessage) Marshal() ([]byte, error) { + return EncodeVarint(nm.x), nil +} + +type NMMessage struct { + nm *nillableMessage +} + +func (*NMMessage) Reset() {} +func (*NMMessage) String() string { return "" } +func (*NMMessage) ProtoMessage() {} + +// Verify a type that uses the Marshaler interface, but has a nil pointer. +func TestNilMarshaler(t *testing.T) { + // Try a struct with a Marshaler field that is nil. + // It should be directly marshable. + nmm := new(NMMessage) + if _, err := Marshal(nmm); err != nil { + t.Error("unexpected error marshaling nmm: ", err) + } + + // Try a struct with a Marshaler field that is not nillable. + nnim := new(NNIMessage) + nnim.nni = 7 + var _ Marshaler = nnim.nni // verify it is truly a Marshaler + if _, err := Marshal(nnim); err != nil { + t.Error("unexpected error marshaling nnim: ", err) + } +} + +func TestAllSetDefaults(t *testing.T) { + // Exercise SetDefaults with all scalar field types. + m := &Defaults{ + // NaN != NaN, so override that here. + F_Nan: Float32(1.7), + } + expected := &Defaults{ + F_Bool: Bool(true), + F_Int32: Int32(32), + F_Int64: Int64(64), + F_Fixed32: Uint32(320), + F_Fixed64: Uint64(640), + F_Uint32: Uint32(3200), + F_Uint64: Uint64(6400), + F_Float: Float32(314159), + F_Double: Float64(271828), + F_String: String(`hello, "world!"` + "\n"), + F_Bytes: []byte("Bignose"), + F_Sint32: Int32(-32), + F_Sint64: Int64(-64), + F_Enum: Defaults_GREEN.Enum(), + F_Pinf: Float32(float32(math.Inf(1))), + F_Ninf: Float32(float32(math.Inf(-1))), + F_Nan: Float32(1.7), + StrZero: String(""), + } + SetDefaults(m) + if !Equal(m, expected) { + t.Errorf("SetDefaults failed\n got %v\nwant %v", m, expected) + } +} + +func TestSetDefaultsWithSetField(t *testing.T) { + // Check that a set value is not overridden. + m := &Defaults{ + F_Int32: Int32(12), + } + SetDefaults(m) + if v := m.GetF_Int32(); v != 12 { + t.Errorf("m.FInt32 = %v, want 12", v) + } +} + +func TestSetDefaultsWithSubMessage(t *testing.T) { + m := &OtherMessage{ + Key: Int64(123), + Inner: &InnerMessage{ + Host: String("gopher"), + }, + } + expected := &OtherMessage{ + Key: Int64(123), + Inner: &InnerMessage{ + Host: String("gopher"), + Port: Int32(4000), + }, + } + SetDefaults(m) + if !Equal(m, expected) { + t.Errorf("\n got %v\nwant %v", m, expected) + } +} + +func TestSetDefaultsWithRepeatedSubMessage(t *testing.T) { + m := &MyMessage{ + RepInner: []*InnerMessage{{}}, + } + expected := &MyMessage{ + RepInner: []*InnerMessage{{ + Port: Int32(4000), + }}, + } + SetDefaults(m) + if !Equal(m, expected) { + t.Errorf("\n got %v\nwant %v", m, expected) + } +} + +func TestSetDefaultWithRepeatedNonMessage(t *testing.T) { + m := &MyMessage{ + Pet: []string{"turtle", "wombat"}, + } + expected := Clone(m) + SetDefaults(m) + if !Equal(m, expected) { + t.Errorf("\n got %v\nwant %v", m, expected) + } +} + +func TestMaximumTagNumber(t *testing.T) { + m := &MaxTag{ + LastField: String("natural goat essence"), + } + buf, err := Marshal(m) + if err != nil { + t.Fatalf("proto.Marshal failed: %v", err) + } + m2 := new(MaxTag) + if err := Unmarshal(buf, m2); err != nil { + t.Fatalf("proto.Unmarshal failed: %v", err) + } + if got, want := m2.GetLastField(), *m.LastField; got != want { + t.Errorf("got %q, want %q", got, want) + } +} + +func TestJSON(t *testing.T) { + m := &MyMessage{ + Count: Int32(4), + Pet: []string{"bunny", "kitty"}, + Inner: &InnerMessage{ + Host: String("cauchy"), + }, + Bikeshed: MyMessage_GREEN.Enum(), + } + const expected = `{"count":4,"pet":["bunny","kitty"],"inner":{"host":"cauchy"},"bikeshed":1}` + + b, err := json.Marshal(m) + if err != nil { + t.Fatalf("json.Marshal failed: %v", err) + } + s := string(b) + if s != expected { + t.Errorf("got %s\nwant %s", s, expected) + } + + received := new(MyMessage) + if err := json.Unmarshal(b, received); err != nil { + t.Fatalf("json.Unmarshal failed: %v", err) + } + if !Equal(received, m) { + t.Fatalf("got %s, want %s", received, m) + } + + // Test unmarshalling of JSON with symbolic enum name. + const old = `{"count":4,"pet":["bunny","kitty"],"inner":{"host":"cauchy"},"bikeshed":"GREEN"}` + received.Reset() + if err := json.Unmarshal([]byte(old), received); err != nil { + t.Fatalf("json.Unmarshal failed: %v", err) + } + if !Equal(received, m) { + t.Fatalf("got %s, want %s", received, m) + } +} + +func TestBadWireType(t *testing.T) { + b := []byte{7<<3 | 6} // field 7, wire type 6 + pb := new(OtherMessage) + if err := Unmarshal(b, pb); err == nil { + t.Errorf("Unmarshal did not fail") + } else if !strings.Contains(err.Error(), "unknown wire type") { + t.Errorf("wrong error: %v", err) + } +} + +func TestBytesWithInvalidLength(t *testing.T) { + // If a byte sequence has an invalid (negative) length, Unmarshal should not panic. + b := []byte{2<<3 | WireBytes, 0xff, 0xff, 0xff, 0xff, 0xff, 0} + Unmarshal(b, new(MyMessage)) +} + +func TestLengthOverflow(t *testing.T) { + // Overflowing a length should not panic. + b := []byte{2<<3 | WireBytes, 1, 1, 3<<3 | WireBytes, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x7f, 0x01} + Unmarshal(b, new(MyMessage)) +} + +func TestVarintOverflow(t *testing.T) { + // Overflowing a 64-bit length should not be allowed. + b := []byte{1<<3 | WireVarint, 0x01, 3<<3 | WireBytes, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80, 0x01} + if err := Unmarshal(b, new(MyMessage)); err == nil { + t.Fatalf("Overflowed uint64 length without error") + } +} + +func TestUnmarshalFuzz(t *testing.T) { + const N = 1000 + seed := time.Now().UnixNano() + t.Logf("RNG seed is %d", seed) + rng := rand.New(rand.NewSource(seed)) + buf := make([]byte, 20) + for i := 0; i < N; i++ { + for j := range buf { + buf[j] = byte(rng.Intn(256)) + } + fuzzUnmarshal(t, buf) + } +} + +func TestMergeMessages(t *testing.T) { + pb := &MessageList{Message: []*MessageList_Message{{Name: String("x"), Count: Int32(1)}}} + data, err := Marshal(pb) + if err != nil { + t.Fatalf("Marshal: %v", err) + } + + pb1 := new(MessageList) + if err := Unmarshal(data, pb1); err != nil { + t.Fatalf("first Unmarshal: %v", err) + } + if err := Unmarshal(data, pb1); err != nil { + t.Fatalf("second Unmarshal: %v", err) + } + if len(pb1.Message) != 1 { + t.Errorf("two Unmarshals produced %d Messages, want 1", len(pb1.Message)) + } + + pb2 := new(MessageList) + if err := UnmarshalMerge(data, pb2); err != nil { + t.Fatalf("first UnmarshalMerge: %v", err) + } + if err := UnmarshalMerge(data, pb2); err != nil { + t.Fatalf("second UnmarshalMerge: %v", err) + } + if len(pb2.Message) != 2 { + t.Errorf("two UnmarshalMerges produced %d Messages, want 2", len(pb2.Message)) + } +} + +func TestExtensionMarshalOrder(t *testing.T) { + m := &MyMessage{Count: Int(123)} + if err := SetExtension(m, E_Ext_More, &Ext{Data: String("alpha")}); err != nil { + t.Fatalf("SetExtension: %v", err) + } + if err := SetExtension(m, E_Ext_Text, String("aleph")); err != nil { + t.Fatalf("SetExtension: %v", err) + } + if err := SetExtension(m, E_Ext_Number, Int32(1)); err != nil { + t.Fatalf("SetExtension: %v", err) + } + + // Serialize m several times, and check we get the same bytes each time. + var orig []byte + for i := 0; i < 100; i++ { + b, err := Marshal(m) + if err != nil { + t.Fatalf("Marshal: %v", err) + } + if i == 0 { + orig = b + continue + } + if !bytes.Equal(b, orig) { + t.Errorf("Bytes differ on attempt #%d", i) + } + } +} + +// Many extensions, because small maps might not iterate differently on each iteration. +var exts = []*ExtensionDesc{ + E_X201, + E_X202, + E_X203, + E_X204, + E_X205, + E_X206, + E_X207, + E_X208, + E_X209, + E_X210, + E_X211, + E_X212, + E_X213, + E_X214, + E_X215, + E_X216, + E_X217, + E_X218, + E_X219, + E_X220, + E_X221, + E_X222, + E_X223, + E_X224, + E_X225, + E_X226, + E_X227, + E_X228, + E_X229, + E_X230, + E_X231, + E_X232, + E_X233, + E_X234, + E_X235, + E_X236, + E_X237, + E_X238, + E_X239, + E_X240, + E_X241, + E_X242, + E_X243, + E_X244, + E_X245, + E_X246, + E_X247, + E_X248, + E_X249, + E_X250, +} + +func TestMessageSetMarshalOrder(t *testing.T) { + m := &MyMessageSet{} + for _, x := range exts { + if err := SetExtension(m, x, &Empty{}); err != nil { + t.Fatalf("SetExtension: %v", err) + } + } + + buf, err := Marshal(m) + if err != nil { + t.Fatalf("Marshal: %v", err) + } + + // Serialize m several times, and check we get the same bytes each time. + for i := 0; i < 10; i++ { + b1, err := Marshal(m) + if err != nil { + t.Fatalf("Marshal: %v", err) + } + if !bytes.Equal(b1, buf) { + t.Errorf("Bytes differ on re-Marshal #%d", i) + } + + m2 := &MyMessageSet{} + if err := Unmarshal(buf, m2); err != nil { + t.Errorf("Unmarshal: %v", err) + } + b2, err := Marshal(m2) + if err != nil { + t.Errorf("re-Marshal: %v", err) + } + if !bytes.Equal(b2, buf) { + t.Errorf("Bytes differ on round-trip #%d", i) + } + } +} + +func TestUnmarshalMergesMessages(t *testing.T) { + // If a nested message occurs twice in the input, + // the fields should be merged when decoding. + a := &OtherMessage{ + Key: Int64(123), + Inner: &InnerMessage{ + Host: String("polhode"), + Port: Int32(1234), + }, + } + aData, err := Marshal(a) + if err != nil { + t.Fatalf("Marshal(a): %v", err) + } + b := &OtherMessage{ + Weight: Float32(1.2), + Inner: &InnerMessage{ + Host: String("herpolhode"), + Connected: Bool(true), + }, + } + bData, err := Marshal(b) + if err != nil { + t.Fatalf("Marshal(b): %v", err) + } + want := &OtherMessage{ + Key: Int64(123), + Weight: Float32(1.2), + Inner: &InnerMessage{ + Host: String("herpolhode"), + Port: Int32(1234), + Connected: Bool(true), + }, + } + got := new(OtherMessage) + if err := Unmarshal(append(aData, bData...), got); err != nil { + t.Fatalf("Unmarshal: %v", err) + } + if !Equal(got, want) { + t.Errorf("\n got %v\nwant %v", got, want) + } +} + +func TestEncodingSizes(t *testing.T) { + tests := []struct { + m Message + n int + }{ + {&Defaults{F_Int32: Int32(math.MaxInt32)}, 6}, + {&Defaults{F_Int32: Int32(math.MinInt32)}, 11}, + {&Defaults{F_Uint32: Uint32(uint32(math.MaxInt32) + 1)}, 6}, + {&Defaults{F_Uint32: Uint32(math.MaxUint32)}, 6}, + } + for _, test := range tests { + b, err := Marshal(test.m) + if err != nil { + t.Errorf("Marshal(%v): %v", test.m, err) + continue + } + if len(b) != test.n { + t.Errorf("Marshal(%v) yielded %d bytes, want %d bytes", test.m, len(b), test.n) + } + } +} + +func TestRequiredNotSetError(t *testing.T) { + pb := initGoTest(false) + pb.RequiredField.Label = nil + pb.F_Int32Required = nil + pb.F_Int64Required = nil + + expected := "0807" + // field 1, encoding 0, value 7 + "2206" + "120474797065" + // field 4, encoding 2 (GoTestField) + "5001" + // field 10, encoding 0, value 1 + "6d20000000" + // field 13, encoding 5, value 0x20 + "714000000000000000" + // field 14, encoding 1, value 0x40 + "78a019" + // field 15, encoding 0, value 0xca0 = 3232 + "8001c032" + // field 16, encoding 0, value 0x1940 = 6464 + "8d0100004a45" + // field 17, encoding 5, value 3232.0 + "9101000000000040b940" + // field 18, encoding 1, value 6464.0 + "9a0106" + "737472696e67" + // field 19, encoding 2, string "string" + "b304" + // field 70, encoding 3, start group + "ba0408" + "7265717569726564" + // field 71, encoding 2, string "required" + "b404" + // field 70, encoding 4, end group + "aa0605" + "6279746573" + // field 101, encoding 2, string "bytes" + "b0063f" + // field 102, encoding 0, 0x3f zigzag32 + "b8067f" // field 103, encoding 0, 0x7f zigzag64 + + o := old() + bytes, err := Marshal(pb) + if _, ok := err.(*RequiredNotSetError); !ok { + fmt.Printf("marshal-1 err = %v, want *RequiredNotSetError", err) + o.DebugPrint("", bytes) + t.Fatalf("expected = %s", expected) + } + if strings.Index(err.Error(), "RequiredField.Label") < 0 { + t.Errorf("marshal-1 wrong err msg: %v", err) + } + if !equal(bytes, expected, t) { + o.DebugPrint("neq 1", bytes) + t.Fatalf("expected = %s", expected) + } + + // Now test Unmarshal by recreating the original buffer. + pbd := new(GoTest) + err = Unmarshal(bytes, pbd) + if _, ok := err.(*RequiredNotSetError); !ok { + t.Fatalf("unmarshal err = %v, want *RequiredNotSetError", err) + o.DebugPrint("", bytes) + t.Fatalf("string = %s", expected) + } + if strings.Index(err.Error(), "RequiredField.{Unknown}") < 0 { + t.Errorf("unmarshal wrong err msg: %v", err) + } + bytes, err = Marshal(pbd) + if _, ok := err.(*RequiredNotSetError); !ok { + t.Errorf("marshal-2 err = %v, want *RequiredNotSetError", err) + o.DebugPrint("", bytes) + t.Fatalf("string = %s", expected) + } + if strings.Index(err.Error(), "RequiredField.Label") < 0 { + t.Errorf("marshal-2 wrong err msg: %v", err) + } + if !equal(bytes, expected, t) { + o.DebugPrint("neq 2", bytes) + t.Fatalf("string = %s", expected) + } +} + +func fuzzUnmarshal(t *testing.T, data []byte) { + defer func() { + if e := recover(); e != nil { + t.Errorf("These bytes caused a panic: %+v", data) + t.Logf("Stack:\n%s", debug.Stack()) + t.FailNow() + } + }() + + pb := new(MyMessage) + Unmarshal(data, pb) +} + +func TestMapFieldMarshal(t *testing.T) { + m := &MessageWithMap{ + NameMapping: map[int32]string{ + 1: "Rob", + 4: "Ian", + 8: "Dave", + }, + } + b, err := Marshal(m) + if err != nil { + t.Fatalf("Marshal: %v", err) + } + + // b should be the concatenation of these three byte sequences in some order. + parts := []string{ + "\n\a\b\x01\x12\x03Rob", + "\n\a\b\x04\x12\x03Ian", + "\n\b\b\x08\x12\x04Dave", + } + ok := false + for i := range parts { + for j := range parts { + if j == i { + continue + } + for k := range parts { + if k == i || k == j { + continue + } + try := parts[i] + parts[j] + parts[k] + if bytes.Equal(b, []byte(try)) { + ok = true + break + } + } + } + } + if !ok { + t.Fatalf("Incorrect Marshal output.\n got %q\nwant %q (or a permutation of that)", b, parts[0]+parts[1]+parts[2]) + } + t.Logf("FYI b: %q", b) + + (new(Buffer)).DebugPrint("Dump of b", b) +} + +func TestMapFieldRoundTrips(t *testing.T) { + m := &MessageWithMap{ + NameMapping: map[int32]string{ + 1: "Rob", + 4: "Ian", + 8: "Dave", + }, + MsgMapping: map[int64]*FloatingPoint{ + 0x7001: &FloatingPoint{F: Float64(2.0)}, + }, + ByteMapping: map[bool][]byte{ + false: []byte("that's not right!"), + true: []byte("aye, 'tis true!"), + }, + } + b, err := Marshal(m) + if err != nil { + t.Fatalf("Marshal: %v", err) + } + t.Logf("FYI b: %q", b) + m2 := new(MessageWithMap) + if err := Unmarshal(b, m2); err != nil { + t.Fatalf("Unmarshal: %v", err) + } + for _, pair := range [][2]interface{}{ + {m.NameMapping, m2.NameMapping}, + {m.MsgMapping, m2.MsgMapping}, + {m.ByteMapping, m2.ByteMapping}, + } { + if !reflect.DeepEqual(pair[0], pair[1]) { + t.Errorf("Map did not survive a round trip.\ninitial: %v\n final: %v", pair[0], pair[1]) + } + } +} + +func TestMapFieldWithNil(t *testing.T) { + m := &MessageWithMap{ + MsgMapping: map[int64]*FloatingPoint{ + 1: nil, + }, + } + b, err := Marshal(m) + if err == nil { + t.Fatalf("Marshal of bad map should have failed, got these bytes: %v", b) + } +} + +// Benchmarks + +func testMsg() *GoTest { + pb := initGoTest(true) + const N = 1000 // Internally the library starts much smaller. + pb.F_Int32Repeated = make([]int32, N) + pb.F_DoubleRepeated = make([]float64, N) + for i := 0; i < N; i++ { + pb.F_Int32Repeated[i] = int32(i) + pb.F_DoubleRepeated[i] = float64(i) + } + return pb +} + +func bytesMsg() *GoTest { + pb := initGoTest(true) + buf := make([]byte, 4000) + for i := range buf { + buf[i] = byte(i) + } + pb.F_BytesDefaulted = buf + return pb +} + +func benchmarkMarshal(b *testing.B, pb Message, marshal func(Message) ([]byte, error)) { + d, _ := marshal(pb) + b.SetBytes(int64(len(d))) + b.ResetTimer() + for i := 0; i < b.N; i++ { + marshal(pb) + } +} + +func benchmarkBufferMarshal(b *testing.B, pb Message) { + p := NewBuffer(nil) + benchmarkMarshal(b, pb, func(pb0 Message) ([]byte, error) { + p.Reset() + err := p.Marshal(pb0) + return p.Bytes(), err + }) +} + +func benchmarkSize(b *testing.B, pb Message) { + benchmarkMarshal(b, pb, func(pb0 Message) ([]byte, error) { + Size(pb) + return nil, nil + }) +} + +func newOf(pb Message) Message { + in := reflect.ValueOf(pb) + if in.IsNil() { + return pb + } + return reflect.New(in.Type().Elem()).Interface().(Message) +} + +func benchmarkUnmarshal(b *testing.B, pb Message, unmarshal func([]byte, Message) error) { + d, _ := Marshal(pb) + b.SetBytes(int64(len(d))) + pbd := newOf(pb) + + b.ResetTimer() + for i := 0; i < b.N; i++ { + unmarshal(d, pbd) + } +} + +func benchmarkBufferUnmarshal(b *testing.B, pb Message) { + p := NewBuffer(nil) + benchmarkUnmarshal(b, pb, func(d []byte, pb0 Message) error { + p.SetBuf(d) + return p.Unmarshal(pb0) + }) +} + +// Benchmark{Marshal,BufferMarshal,Size,Unmarshal,BufferUnmarshal}{,Bytes} + +func BenchmarkMarshal(b *testing.B) { + benchmarkMarshal(b, testMsg(), Marshal) +} + +func BenchmarkBufferMarshal(b *testing.B) { + benchmarkBufferMarshal(b, testMsg()) +} + +func BenchmarkSize(b *testing.B) { + benchmarkSize(b, testMsg()) +} + +func BenchmarkUnmarshal(b *testing.B) { + benchmarkUnmarshal(b, testMsg(), Unmarshal) +} + +func BenchmarkBufferUnmarshal(b *testing.B) { + benchmarkBufferUnmarshal(b, testMsg()) +} + +func BenchmarkMarshalBytes(b *testing.B) { + benchmarkMarshal(b, bytesMsg(), Marshal) +} + +func BenchmarkBufferMarshalBytes(b *testing.B) { + benchmarkBufferMarshal(b, bytesMsg()) +} + +func BenchmarkSizeBytes(b *testing.B) { + benchmarkSize(b, bytesMsg()) +} + +func BenchmarkUnmarshalBytes(b *testing.B) { + benchmarkUnmarshal(b, bytesMsg(), Unmarshal) +} + +func BenchmarkBufferUnmarshalBytes(b *testing.B) { + benchmarkBufferUnmarshal(b, bytesMsg()) +} + +func BenchmarkUnmarshalUnrecognizedFields(b *testing.B) { + b.StopTimer() + pb := initGoTestField() + skip := &GoSkipTest{ + SkipInt32: Int32(32), + SkipFixed32: Uint32(3232), + SkipFixed64: Uint64(6464), + SkipString: String("skipper"), + Skipgroup: &GoSkipTest_SkipGroup{ + GroupInt32: Int32(75), + GroupString: String("wxyz"), + }, + } + + pbd := new(GoTestField) + p := NewBuffer(nil) + p.Marshal(pb) + p.Marshal(skip) + p2 := NewBuffer(nil) + + b.StartTimer() + for i := 0; i < b.N; i++ { + p2.SetBuf(p.Bytes()) + p2.Unmarshal(pbd) + } +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/clone.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/clone.go new file mode 100644 index 0000000000000..915a68b8ec3ad --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/clone.go @@ -0,0 +1,212 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2011 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// Protocol buffer deep copy and merge. +// TODO: MessageSet and RawMessage. + +package proto + +import ( + "log" + "reflect" + "strings" +) + +// Clone returns a deep copy of a protocol buffer. +func Clone(pb Message) Message { + in := reflect.ValueOf(pb) + if in.IsNil() { + return pb + } + + out := reflect.New(in.Type().Elem()) + // out is empty so a merge is a deep copy. + mergeStruct(out.Elem(), in.Elem()) + return out.Interface().(Message) +} + +// Merge merges src into dst. +// Required and optional fields that are set in src will be set to that value in dst. +// Elements of repeated fields will be appended. +// Merge panics if src and dst are not the same type, or if dst is nil. +func Merge(dst, src Message) { + in := reflect.ValueOf(src) + out := reflect.ValueOf(dst) + if out.IsNil() { + panic("proto: nil destination") + } + if in.Type() != out.Type() { + // Explicit test prior to mergeStruct so that mistyped nils will fail + panic("proto: type mismatch") + } + if in.IsNil() { + // Merging nil into non-nil is a quiet no-op + return + } + mergeStruct(out.Elem(), in.Elem()) +} + +func mergeStruct(out, in reflect.Value) { + sprop := GetProperties(in.Type()) + for i := 0; i < in.NumField(); i++ { + f := in.Type().Field(i) + if strings.HasPrefix(f.Name, "XXX_") { + continue + } + mergeAny(out.Field(i), in.Field(i), false, sprop.Prop[i]) + } + + if emIn, ok := in.Addr().Interface().(extendableProto); ok { + emOut := out.Addr().Interface().(extendableProto) + mergeExtension(emOut.ExtensionMap(), emIn.ExtensionMap()) + } + + uf := in.FieldByName("XXX_unrecognized") + if !uf.IsValid() { + return + } + uin := uf.Bytes() + if len(uin) > 0 { + out.FieldByName("XXX_unrecognized").SetBytes(append([]byte(nil), uin...)) + } +} + +// mergeAny performs a merge between two values of the same type. +// viaPtr indicates whether the values were indirected through a pointer (implying proto2). +// prop is set if this is a struct field (it may be nil). +func mergeAny(out, in reflect.Value, viaPtr bool, prop *Properties) { + if in.Type() == protoMessageType { + if !in.IsNil() { + if out.IsNil() { + out.Set(reflect.ValueOf(Clone(in.Interface().(Message)))) + } else { + Merge(out.Interface().(Message), in.Interface().(Message)) + } + } + return + } + switch in.Kind() { + case reflect.Bool, reflect.Float32, reflect.Float64, reflect.Int32, reflect.Int64, + reflect.String, reflect.Uint32, reflect.Uint64: + if !viaPtr && isProto3Zero(in) { + return + } + out.Set(in) + case reflect.Map: + if in.Len() == 0 { + return + } + if out.IsNil() { + out.Set(reflect.MakeMap(in.Type())) + } + // For maps with value types of *T or []byte we need to deep copy each value. + elemKind := in.Type().Elem().Kind() + for _, key := range in.MapKeys() { + var val reflect.Value + switch elemKind { + case reflect.Ptr: + val = reflect.New(in.Type().Elem().Elem()) + mergeAny(val, in.MapIndex(key), false, nil) + case reflect.Slice: + val = in.MapIndex(key) + val = reflect.ValueOf(append([]byte{}, val.Bytes()...)) + default: + val = in.MapIndex(key) + } + out.SetMapIndex(key, val) + } + case reflect.Ptr: + if in.IsNil() { + return + } + if out.IsNil() { + out.Set(reflect.New(in.Elem().Type())) + } + mergeAny(out.Elem(), in.Elem(), true, nil) + case reflect.Slice: + if in.IsNil() { + return + } + if in.Type().Elem().Kind() == reflect.Uint8 { + // []byte is a scalar bytes field, not a repeated field. + + // Edge case: if this is in a proto3 message, a zero length + // bytes field is considered the zero value, and should not + // be merged. + if prop != nil && prop.proto3 && in.Len() == 0 { + return + } + + // Make a deep copy. + // Append to []byte{} instead of []byte(nil) so that we never end up + // with a nil result. + out.SetBytes(append([]byte{}, in.Bytes()...)) + return + } + n := in.Len() + if out.IsNil() { + out.Set(reflect.MakeSlice(in.Type(), 0, n)) + } + switch in.Type().Elem().Kind() { + case reflect.Bool, reflect.Float32, reflect.Float64, reflect.Int32, reflect.Int64, + reflect.String, reflect.Uint32, reflect.Uint64: + out.Set(reflect.AppendSlice(out, in)) + default: + for i := 0; i < n; i++ { + x := reflect.Indirect(reflect.New(in.Type().Elem())) + mergeAny(x, in.Index(i), false, nil) + out.Set(reflect.Append(out, x)) + } + } + case reflect.Struct: + mergeStruct(out, in) + default: + // unknown type, so not a protocol buffer + log.Printf("proto: don't know how to copy %v", in) + } +} + +func mergeExtension(out, in map[int32]Extension) { + for extNum, eIn := range in { + eOut := Extension{desc: eIn.desc} + if eIn.value != nil { + v := reflect.New(reflect.TypeOf(eIn.value)).Elem() + mergeAny(v, reflect.ValueOf(eIn.value), false, nil) + eOut.value = v.Interface() + } + if eIn.enc != nil { + eOut.enc = make([]byte, len(eIn.enc)) + copy(eOut.enc, eIn.enc) + } + + out[extNum] = eOut + } +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/clone_test.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/clone_test.go new file mode 100644 index 0000000000000..a1c697bc8422c --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/clone_test.go @@ -0,0 +1,245 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2011 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "testing" + + "github.com/golang/protobuf/proto" + + proto3pb "github.com/golang/protobuf/proto/proto3_proto" + pb "github.com/golang/protobuf/proto/testdata" +) + +var cloneTestMessage = &pb.MyMessage{ + Count: proto.Int32(42), + Name: proto.String("Dave"), + Pet: []string{"bunny", "kitty", "horsey"}, + Inner: &pb.InnerMessage{ + Host: proto.String("niles"), + Port: proto.Int32(9099), + Connected: proto.Bool(true), + }, + Others: []*pb.OtherMessage{ + { + Value: []byte("some bytes"), + }, + }, + Somegroup: &pb.MyMessage_SomeGroup{ + GroupField: proto.Int32(6), + }, + RepBytes: [][]byte{[]byte("sham"), []byte("wow")}, +} + +func init() { + ext := &pb.Ext{ + Data: proto.String("extension"), + } + if err := proto.SetExtension(cloneTestMessage, pb.E_Ext_More, ext); err != nil { + panic("SetExtension: " + err.Error()) + } +} + +func TestClone(t *testing.T) { + m := proto.Clone(cloneTestMessage).(*pb.MyMessage) + if !proto.Equal(m, cloneTestMessage) { + t.Errorf("Clone(%v) = %v", cloneTestMessage, m) + } + + // Verify it was a deep copy. + *m.Inner.Port++ + if proto.Equal(m, cloneTestMessage) { + t.Error("Mutating clone changed the original") + } + // Byte fields and repeated fields should be copied. + if &m.Pet[0] == &cloneTestMessage.Pet[0] { + t.Error("Pet: repeated field not copied") + } + if &m.Others[0] == &cloneTestMessage.Others[0] { + t.Error("Others: repeated field not copied") + } + if &m.Others[0].Value[0] == &cloneTestMessage.Others[0].Value[0] { + t.Error("Others[0].Value: bytes field not copied") + } + if &m.RepBytes[0] == &cloneTestMessage.RepBytes[0] { + t.Error("RepBytes: repeated field not copied") + } + if &m.RepBytes[0][0] == &cloneTestMessage.RepBytes[0][0] { + t.Error("RepBytes[0]: bytes field not copied") + } +} + +func TestCloneNil(t *testing.T) { + var m *pb.MyMessage + if c := proto.Clone(m); !proto.Equal(m, c) { + t.Errorf("Clone(%v) = %v", m, c) + } +} + +var mergeTests = []struct { + src, dst, want proto.Message +}{ + { + src: &pb.MyMessage{ + Count: proto.Int32(42), + }, + dst: &pb.MyMessage{ + Name: proto.String("Dave"), + }, + want: &pb.MyMessage{ + Count: proto.Int32(42), + Name: proto.String("Dave"), + }, + }, + { + src: &pb.MyMessage{ + Inner: &pb.InnerMessage{ + Host: proto.String("hey"), + Connected: proto.Bool(true), + }, + Pet: []string{"horsey"}, + Others: []*pb.OtherMessage{ + { + Value: []byte("some bytes"), + }, + }, + }, + dst: &pb.MyMessage{ + Inner: &pb.InnerMessage{ + Host: proto.String("niles"), + Port: proto.Int32(9099), + }, + Pet: []string{"bunny", "kitty"}, + Others: []*pb.OtherMessage{ + { + Key: proto.Int64(31415926535), + }, + { + // Explicitly test a src=nil field + Inner: nil, + }, + }, + }, + want: &pb.MyMessage{ + Inner: &pb.InnerMessage{ + Host: proto.String("hey"), + Connected: proto.Bool(true), + Port: proto.Int32(9099), + }, + Pet: []string{"bunny", "kitty", "horsey"}, + Others: []*pb.OtherMessage{ + { + Key: proto.Int64(31415926535), + }, + {}, + { + Value: []byte("some bytes"), + }, + }, + }, + }, + { + src: &pb.MyMessage{ + RepBytes: [][]byte{[]byte("wow")}, + }, + dst: &pb.MyMessage{ + Somegroup: &pb.MyMessage_SomeGroup{ + GroupField: proto.Int32(6), + }, + RepBytes: [][]byte{[]byte("sham")}, + }, + want: &pb.MyMessage{ + Somegroup: &pb.MyMessage_SomeGroup{ + GroupField: proto.Int32(6), + }, + RepBytes: [][]byte{[]byte("sham"), []byte("wow")}, + }, + }, + // Check that a scalar bytes field replaces rather than appends. + { + src: &pb.OtherMessage{Value: []byte("foo")}, + dst: &pb.OtherMessage{Value: []byte("bar")}, + want: &pb.OtherMessage{Value: []byte("foo")}, + }, + { + src: &pb.MessageWithMap{ + NameMapping: map[int32]string{6: "Nigel"}, + MsgMapping: map[int64]*pb.FloatingPoint{ + 0x4001: &pb.FloatingPoint{F: proto.Float64(2.0)}, + }, + ByteMapping: map[bool][]byte{true: []byte("wowsa")}, + }, + dst: &pb.MessageWithMap{ + NameMapping: map[int32]string{ + 6: "Bruce", // should be overwritten + 7: "Andrew", + }, + }, + want: &pb.MessageWithMap{ + NameMapping: map[int32]string{ + 6: "Nigel", + 7: "Andrew", + }, + MsgMapping: map[int64]*pb.FloatingPoint{ + 0x4001: &pb.FloatingPoint{F: proto.Float64(2.0)}, + }, + ByteMapping: map[bool][]byte{true: []byte("wowsa")}, + }, + }, + // proto3 shouldn't merge zero values, + // in the same way that proto2 shouldn't merge nils. + { + src: &proto3pb.Message{ + Name: "Aaron", + Data: []byte(""), // zero value, but not nil + }, + dst: &proto3pb.Message{ + HeightInCm: 176, + Data: []byte("texas!"), + }, + want: &proto3pb.Message{ + Name: "Aaron", + HeightInCm: 176, + Data: []byte("texas!"), + }, + }, +} + +func TestMerge(t *testing.T) { + for _, m := range mergeTests { + got := proto.Clone(m.dst) + proto.Merge(got, m.src) + if !proto.Equal(got, m.want) { + t.Errorf("Merge(%v, %v)\n got %v\nwant %v\n", m.dst, m.src, got, m.want) + } + } +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/decode.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/decode.go new file mode 100644 index 0000000000000..bf71dcad1072b --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/decode.go @@ -0,0 +1,827 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +/* + * Routines for decoding protocol buffer data to construct in-memory representations. + */ + +import ( + "errors" + "fmt" + "io" + "os" + "reflect" +) + +// errOverflow is returned when an integer is too large to be represented. +var errOverflow = errors.New("proto: integer overflow") + +// The fundamental decoders that interpret bytes on the wire. +// Those that take integer types all return uint64 and are +// therefore of type valueDecoder. + +// DecodeVarint reads a varint-encoded integer from the slice. +// It returns the integer and the number of bytes consumed, or +// zero if there is not enough. +// This is the format for the +// int32, int64, uint32, uint64, bool, and enum +// protocol buffer types. +func DecodeVarint(buf []byte) (x uint64, n int) { + // x, n already 0 + for shift := uint(0); shift < 64; shift += 7 { + if n >= len(buf) { + return 0, 0 + } + b := uint64(buf[n]) + n++ + x |= (b & 0x7F) << shift + if (b & 0x80) == 0 { + return x, n + } + } + + // The number is too large to represent in a 64-bit value. + return 0, 0 +} + +// DecodeVarint reads a varint-encoded integer from the Buffer. +// This is the format for the +// int32, int64, uint32, uint64, bool, and enum +// protocol buffer types. +func (p *Buffer) DecodeVarint() (x uint64, err error) { + // x, err already 0 + + i := p.index + l := len(p.buf) + + for shift := uint(0); shift < 64; shift += 7 { + if i >= l { + err = io.ErrUnexpectedEOF + return + } + b := p.buf[i] + i++ + x |= (uint64(b) & 0x7F) << shift + if b < 0x80 { + p.index = i + return + } + } + + // The number is too large to represent in a 64-bit value. + err = errOverflow + return +} + +// DecodeFixed64 reads a 64-bit integer from the Buffer. +// This is the format for the +// fixed64, sfixed64, and double protocol buffer types. +func (p *Buffer) DecodeFixed64() (x uint64, err error) { + // x, err already 0 + i := p.index + 8 + if i < 0 || i > len(p.buf) { + err = io.ErrUnexpectedEOF + return + } + p.index = i + + x = uint64(p.buf[i-8]) + x |= uint64(p.buf[i-7]) << 8 + x |= uint64(p.buf[i-6]) << 16 + x |= uint64(p.buf[i-5]) << 24 + x |= uint64(p.buf[i-4]) << 32 + x |= uint64(p.buf[i-3]) << 40 + x |= uint64(p.buf[i-2]) << 48 + x |= uint64(p.buf[i-1]) << 56 + return +} + +// DecodeFixed32 reads a 32-bit integer from the Buffer. +// This is the format for the +// fixed32, sfixed32, and float protocol buffer types. +func (p *Buffer) DecodeFixed32() (x uint64, err error) { + // x, err already 0 + i := p.index + 4 + if i < 0 || i > len(p.buf) { + err = io.ErrUnexpectedEOF + return + } + p.index = i + + x = uint64(p.buf[i-4]) + x |= uint64(p.buf[i-3]) << 8 + x |= uint64(p.buf[i-2]) << 16 + x |= uint64(p.buf[i-1]) << 24 + return +} + +// DecodeZigzag64 reads a zigzag-encoded 64-bit integer +// from the Buffer. +// This is the format used for the sint64 protocol buffer type. +func (p *Buffer) DecodeZigzag64() (x uint64, err error) { + x, err = p.DecodeVarint() + if err != nil { + return + } + x = (x >> 1) ^ uint64((int64(x&1)<<63)>>63) + return +} + +// DecodeZigzag32 reads a zigzag-encoded 32-bit integer +// from the Buffer. +// This is the format used for the sint32 protocol buffer type. +func (p *Buffer) DecodeZigzag32() (x uint64, err error) { + x, err = p.DecodeVarint() + if err != nil { + return + } + x = uint64((uint32(x) >> 1) ^ uint32((int32(x&1)<<31)>>31)) + return +} + +// These are not ValueDecoders: they produce an array of bytes or a string. +// bytes, embedded messages + +// DecodeRawBytes reads a count-delimited byte buffer from the Buffer. +// This is the format used for the bytes protocol buffer +// type and for embedded messages. +func (p *Buffer) DecodeRawBytes(alloc bool) (buf []byte, err error) { + n, err := p.DecodeVarint() + if err != nil { + return nil, err + } + + nb := int(n) + if nb < 0 { + return nil, fmt.Errorf("proto: bad byte length %d", nb) + } + end := p.index + nb + if end < p.index || end > len(p.buf) { + return nil, io.ErrUnexpectedEOF + } + + if !alloc { + // todo: check if can get more uses of alloc=false + buf = p.buf[p.index:end] + p.index += nb + return + } + + buf = make([]byte, nb) + copy(buf, p.buf[p.index:]) + p.index += nb + return +} + +// DecodeStringBytes reads an encoded string from the Buffer. +// This is the format used for the proto2 string type. +func (p *Buffer) DecodeStringBytes() (s string, err error) { + buf, err := p.DecodeRawBytes(false) + if err != nil { + return + } + return string(buf), nil +} + +// Skip the next item in the buffer. Its wire type is decoded and presented as an argument. +// If the protocol buffer has extensions, and the field matches, add it as an extension. +// Otherwise, if the XXX_unrecognized field exists, append the skipped data there. +func (o *Buffer) skipAndSave(t reflect.Type, tag, wire int, base structPointer, unrecField field) error { + oi := o.index + + err := o.skip(t, tag, wire) + if err != nil { + return err + } + + if !unrecField.IsValid() { + return nil + } + + ptr := structPointer_Bytes(base, unrecField) + + // Add the skipped field to struct field + obuf := o.buf + + o.buf = *ptr + o.EncodeVarint(uint64(tag<<3 | wire)) + *ptr = append(o.buf, obuf[oi:o.index]...) + + o.buf = obuf + + return nil +} + +// Skip the next item in the buffer. Its wire type is decoded and presented as an argument. +func (o *Buffer) skip(t reflect.Type, tag, wire int) error { + + var u uint64 + var err error + + switch wire { + case WireVarint: + _, err = o.DecodeVarint() + case WireFixed64: + _, err = o.DecodeFixed64() + case WireBytes: + _, err = o.DecodeRawBytes(false) + case WireFixed32: + _, err = o.DecodeFixed32() + case WireStartGroup: + for { + u, err = o.DecodeVarint() + if err != nil { + break + } + fwire := int(u & 0x7) + if fwire == WireEndGroup { + break + } + ftag := int(u >> 3) + err = o.skip(t, ftag, fwire) + if err != nil { + break + } + } + default: + err = fmt.Errorf("proto: can't skip unknown wire type %d for %s", wire, t) + } + return err +} + +// Unmarshaler is the interface representing objects that can +// unmarshal themselves. The method should reset the receiver before +// decoding starts. The argument points to data that may be +// overwritten, so implementations should not keep references to the +// buffer. +type Unmarshaler interface { + Unmarshal([]byte) error +} + +// Unmarshal parses the protocol buffer representation in buf and places the +// decoded result in pb. If the struct underlying pb does not match +// the data in buf, the results can be unpredictable. +// +// Unmarshal resets pb before starting to unmarshal, so any +// existing data in pb is always removed. Use UnmarshalMerge +// to preserve and append to existing data. +func Unmarshal(buf []byte, pb Message) error { + pb.Reset() + return UnmarshalMerge(buf, pb) +} + +// UnmarshalMerge parses the protocol buffer representation in buf and +// writes the decoded result to pb. If the struct underlying pb does not match +// the data in buf, the results can be unpredictable. +// +// UnmarshalMerge merges into existing data in pb. +// Most code should use Unmarshal instead. +func UnmarshalMerge(buf []byte, pb Message) error { + // If the object can unmarshal itself, let it. + if u, ok := pb.(Unmarshaler); ok { + return u.Unmarshal(buf) + } + return NewBuffer(buf).Unmarshal(pb) +} + +// Unmarshal parses the protocol buffer representation in the +// Buffer and places the decoded result in pb. If the struct +// underlying pb does not match the data in the buffer, the results can be +// unpredictable. +func (p *Buffer) Unmarshal(pb Message) error { + // If the object can unmarshal itself, let it. + if u, ok := pb.(Unmarshaler); ok { + err := u.Unmarshal(p.buf[p.index:]) + p.index = len(p.buf) + return err + } + + typ, base, err := getbase(pb) + if err != nil { + return err + } + + err = p.unmarshalType(typ.Elem(), GetProperties(typ.Elem()), false, base) + + if collectStats { + stats.Decode++ + } + + return err +} + +// unmarshalType does the work of unmarshaling a structure. +func (o *Buffer) unmarshalType(st reflect.Type, prop *StructProperties, is_group bool, base structPointer) error { + var state errorState + required, reqFields := prop.reqCount, uint64(0) + + var err error + for err == nil && o.index < len(o.buf) { + oi := o.index + var u uint64 + u, err = o.DecodeVarint() + if err != nil { + break + } + wire := int(u & 0x7) + if wire == WireEndGroup { + if is_group { + return nil // input is satisfied + } + return fmt.Errorf("proto: %s: wiretype end group for non-group", st) + } + tag := int(u >> 3) + if tag <= 0 { + return fmt.Errorf("proto: %s: illegal tag %d (wire type %d)", st, tag, wire) + } + fieldnum, ok := prop.decoderTags.get(tag) + if !ok { + // Maybe it's an extension? + if prop.extendable { + if e := structPointer_Interface(base, st).(extendableProto); isExtensionField(e, int32(tag)) { + if err = o.skip(st, tag, wire); err == nil { + ext := e.ExtensionMap()[int32(tag)] // may be missing + ext.enc = append(ext.enc, o.buf[oi:o.index]...) + e.ExtensionMap()[int32(tag)] = ext + } + continue + } + } + err = o.skipAndSave(st, tag, wire, base, prop.unrecField) + continue + } + p := prop.Prop[fieldnum] + + if p.dec == nil { + fmt.Fprintf(os.Stderr, "proto: no protobuf decoder for %s.%s\n", st, st.Field(fieldnum).Name) + continue + } + dec := p.dec + if wire != WireStartGroup && wire != p.WireType { + if wire == WireBytes && p.packedDec != nil { + // a packable field + dec = p.packedDec + } else { + err = fmt.Errorf("proto: bad wiretype for field %s.%s: got wiretype %d, want %d", st, st.Field(fieldnum).Name, wire, p.WireType) + continue + } + } + decErr := dec(o, p, base) + if decErr != nil && !state.shouldContinue(decErr, p) { + err = decErr + } + if err == nil && p.Required { + // Successfully decoded a required field. + if tag <= 64 { + // use bitmap for fields 1-64 to catch field reuse. + var mask uint64 = 1 << uint64(tag-1) + if reqFields&mask == 0 { + // new required field + reqFields |= mask + required-- + } + } else { + // This is imprecise. It can be fooled by a required field + // with a tag > 64 that is encoded twice; that's very rare. + // A fully correct implementation would require allocating + // a data structure, which we would like to avoid. + required-- + } + } + } + if err == nil { + if is_group { + return io.ErrUnexpectedEOF + } + if state.err != nil { + return state.err + } + if required > 0 { + // Not enough information to determine the exact field. If we use extra + // CPU, we could determine the field only if the missing required field + // has a tag <= 64 and we check reqFields. + return &RequiredNotSetError{"{Unknown}"} + } + } + return err +} + +// Individual type decoders +// For each, +// u is the decoded value, +// v is a pointer to the field (pointer) in the struct + +// Sizes of the pools to allocate inside the Buffer. +// The goal is modest amortization and allocation +// on at least 16-byte boundaries. +const ( + boolPoolSize = 16 + uint32PoolSize = 8 + uint64PoolSize = 4 +) + +// Decode a bool. +func (o *Buffer) dec_bool(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + if len(o.bools) == 0 { + o.bools = make([]bool, boolPoolSize) + } + o.bools[0] = u != 0 + *structPointer_Bool(base, p.field) = &o.bools[0] + o.bools = o.bools[1:] + return nil +} + +func (o *Buffer) dec_proto3_bool(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + *structPointer_BoolVal(base, p.field) = u != 0 + return nil +} + +// Decode an int32. +func (o *Buffer) dec_int32(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + word32_Set(structPointer_Word32(base, p.field), o, uint32(u)) + return nil +} + +func (o *Buffer) dec_proto3_int32(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + word32Val_Set(structPointer_Word32Val(base, p.field), uint32(u)) + return nil +} + +// Decode an int64. +func (o *Buffer) dec_int64(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + word64_Set(structPointer_Word64(base, p.field), o, u) + return nil +} + +func (o *Buffer) dec_proto3_int64(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + word64Val_Set(structPointer_Word64Val(base, p.field), o, u) + return nil +} + +// Decode a string. +func (o *Buffer) dec_string(p *Properties, base structPointer) error { + s, err := o.DecodeStringBytes() + if err != nil { + return err + } + *structPointer_String(base, p.field) = &s + return nil +} + +func (o *Buffer) dec_proto3_string(p *Properties, base structPointer) error { + s, err := o.DecodeStringBytes() + if err != nil { + return err + } + *structPointer_StringVal(base, p.field) = s + return nil +} + +// Decode a slice of bytes ([]byte). +func (o *Buffer) dec_slice_byte(p *Properties, base structPointer) error { + b, err := o.DecodeRawBytes(true) + if err != nil { + return err + } + *structPointer_Bytes(base, p.field) = b + return nil +} + +// Decode a slice of bools ([]bool). +func (o *Buffer) dec_slice_bool(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + v := structPointer_BoolSlice(base, p.field) + *v = append(*v, u != 0) + return nil +} + +// Decode a slice of bools ([]bool) in packed format. +func (o *Buffer) dec_slice_packed_bool(p *Properties, base structPointer) error { + v := structPointer_BoolSlice(base, p.field) + + nn, err := o.DecodeVarint() + if err != nil { + return err + } + nb := int(nn) // number of bytes of encoded bools + + y := *v + for i := 0; i < nb; i++ { + u, err := p.valDec(o) + if err != nil { + return err + } + y = append(y, u != 0) + } + + *v = y + return nil +} + +// Decode a slice of int32s ([]int32). +func (o *Buffer) dec_slice_int32(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + structPointer_Word32Slice(base, p.field).Append(uint32(u)) + return nil +} + +// Decode a slice of int32s ([]int32) in packed format. +func (o *Buffer) dec_slice_packed_int32(p *Properties, base structPointer) error { + v := structPointer_Word32Slice(base, p.field) + + nn, err := o.DecodeVarint() + if err != nil { + return err + } + nb := int(nn) // number of bytes of encoded int32s + + fin := o.index + nb + if fin < o.index { + return errOverflow + } + for o.index < fin { + u, err := p.valDec(o) + if err != nil { + return err + } + v.Append(uint32(u)) + } + return nil +} + +// Decode a slice of int64s ([]int64). +func (o *Buffer) dec_slice_int64(p *Properties, base structPointer) error { + u, err := p.valDec(o) + if err != nil { + return err + } + + structPointer_Word64Slice(base, p.field).Append(u) + return nil +} + +// Decode a slice of int64s ([]int64) in packed format. +func (o *Buffer) dec_slice_packed_int64(p *Properties, base structPointer) error { + v := structPointer_Word64Slice(base, p.field) + + nn, err := o.DecodeVarint() + if err != nil { + return err + } + nb := int(nn) // number of bytes of encoded int64s + + fin := o.index + nb + if fin < o.index { + return errOverflow + } + for o.index < fin { + u, err := p.valDec(o) + if err != nil { + return err + } + v.Append(u) + } + return nil +} + +// Decode a slice of strings ([]string). +func (o *Buffer) dec_slice_string(p *Properties, base structPointer) error { + s, err := o.DecodeStringBytes() + if err != nil { + return err + } + v := structPointer_StringSlice(base, p.field) + *v = append(*v, s) + return nil +} + +// Decode a slice of slice of bytes ([][]byte). +func (o *Buffer) dec_slice_slice_byte(p *Properties, base structPointer) error { + b, err := o.DecodeRawBytes(true) + if err != nil { + return err + } + v := structPointer_BytesSlice(base, p.field) + *v = append(*v, b) + return nil +} + +// Decode a map field. +func (o *Buffer) dec_new_map(p *Properties, base structPointer) error { + raw, err := o.DecodeRawBytes(false) + if err != nil { + return err + } + oi := o.index // index at the end of this map entry + o.index -= len(raw) // move buffer back to start of map entry + + mptr := structPointer_NewAt(base, p.field, p.mtype) // *map[K]V + if mptr.Elem().IsNil() { + mptr.Elem().Set(reflect.MakeMap(mptr.Type().Elem())) + } + v := mptr.Elem() // map[K]V + + // Prepare addressable doubly-indirect placeholders for the key and value types. + // See enc_new_map for why. + keyptr := reflect.New(reflect.PtrTo(p.mtype.Key())).Elem() // addressable *K + keybase := toStructPointer(keyptr.Addr()) // **K + + var valbase structPointer + var valptr reflect.Value + switch p.mtype.Elem().Kind() { + case reflect.Slice: + // []byte + var dummy []byte + valptr = reflect.ValueOf(&dummy) // *[]byte + valbase = toStructPointer(valptr) // *[]byte + case reflect.Ptr: + // message; valptr is **Msg; need to allocate the intermediate pointer + valptr = reflect.New(reflect.PtrTo(p.mtype.Elem())).Elem() // addressable *V + valptr.Set(reflect.New(valptr.Type().Elem())) + valbase = toStructPointer(valptr) + default: + // everything else + valptr = reflect.New(reflect.PtrTo(p.mtype.Elem())).Elem() // addressable *V + valbase = toStructPointer(valptr.Addr()) // **V + } + + // Decode. + // This parses a restricted wire format, namely the encoding of a message + // with two fields. See enc_new_map for the format. + for o.index < oi { + // tagcode for key and value properties are always a single byte + // because they have tags 1 and 2. + tagcode := o.buf[o.index] + o.index++ + switch tagcode { + case p.mkeyprop.tagcode[0]: + if err := p.mkeyprop.dec(o, p.mkeyprop, keybase); err != nil { + return err + } + case p.mvalprop.tagcode[0]: + if err := p.mvalprop.dec(o, p.mvalprop, valbase); err != nil { + return err + } + default: + // TODO: Should we silently skip this instead? + return fmt.Errorf("proto: bad map data tag %d", raw[0]) + } + } + keyelem, valelem := keyptr.Elem(), valptr.Elem() + if !keyelem.IsValid() || !valelem.IsValid() { + // We did not decode the key or the value in the map entry. + // Either way, it's an invalid map entry. + return fmt.Errorf("proto: bad map data: missing key/val") + } + + v.SetMapIndex(keyelem, valelem) + return nil +} + +// Decode a group. +func (o *Buffer) dec_struct_group(p *Properties, base structPointer) error { + bas := structPointer_GetStructPointer(base, p.field) + if structPointer_IsNil(bas) { + // allocate new nested message + bas = toStructPointer(reflect.New(p.stype)) + structPointer_SetStructPointer(base, p.field, bas) + } + return o.unmarshalType(p.stype, p.sprop, true, bas) +} + +// Decode an embedded message. +func (o *Buffer) dec_struct_message(p *Properties, base structPointer) (err error) { + raw, e := o.DecodeRawBytes(false) + if e != nil { + return e + } + + bas := structPointer_GetStructPointer(base, p.field) + if structPointer_IsNil(bas) { + // allocate new nested message + bas = toStructPointer(reflect.New(p.stype)) + structPointer_SetStructPointer(base, p.field, bas) + } + + // If the object can unmarshal itself, let it. + if p.isUnmarshaler { + iv := structPointer_Interface(bas, p.stype) + return iv.(Unmarshaler).Unmarshal(raw) + } + + obuf := o.buf + oi := o.index + o.buf = raw + o.index = 0 + + err = o.unmarshalType(p.stype, p.sprop, false, bas) + o.buf = obuf + o.index = oi + + return err +} + +// Decode a slice of embedded messages. +func (o *Buffer) dec_slice_struct_message(p *Properties, base structPointer) error { + return o.dec_slice_struct(p, false, base) +} + +// Decode a slice of embedded groups. +func (o *Buffer) dec_slice_struct_group(p *Properties, base structPointer) error { + return o.dec_slice_struct(p, true, base) +} + +// Decode a slice of structs ([]*struct). +func (o *Buffer) dec_slice_struct(p *Properties, is_group bool, base structPointer) error { + v := reflect.New(p.stype) + bas := toStructPointer(v) + structPointer_StructPointerSlice(base, p.field).Append(bas) + + if is_group { + err := o.unmarshalType(p.stype, p.sprop, is_group, bas) + return err + } + + raw, err := o.DecodeRawBytes(false) + if err != nil { + return err + } + + // If the object can unmarshal itself, let it. + if p.isUnmarshaler { + iv := v.Interface() + return iv.(Unmarshaler).Unmarshal(raw) + } + + obuf := o.buf + oi := o.index + o.buf = raw + o.index = 0 + + err = o.unmarshalType(p.stype, p.sprop, is_group, bas) + + o.buf = obuf + o.index = oi + + return err +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/encode.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/encode.go new file mode 100644 index 0000000000000..91f3f0784d2c4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/encode.go @@ -0,0 +1,1293 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +/* + * Routines for encoding data into the wire format for protocol buffers. + */ + +import ( + "errors" + "fmt" + "reflect" + "sort" +) + +// RequiredNotSetError is the error returned if Marshal is called with +// a protocol buffer struct whose required fields have not +// all been initialized. It is also the error returned if Unmarshal is +// called with an encoded protocol buffer that does not include all the +// required fields. +// +// When printed, RequiredNotSetError reports the first unset required field in a +// message. If the field cannot be precisely determined, it is reported as +// "{Unknown}". +type RequiredNotSetError struct { + field string +} + +func (e *RequiredNotSetError) Error() string { + return fmt.Sprintf("proto: required field %q not set", e.field) +} + +var ( + // errRepeatedHasNil is the error returned if Marshal is called with + // a struct with a repeated field containing a nil element. + errRepeatedHasNil = errors.New("proto: repeated field has nil element") + + // ErrNil is the error returned if Marshal is called with nil. + ErrNil = errors.New("proto: Marshal called with nil") +) + +// The fundamental encoders that put bytes on the wire. +// Those that take integer types all accept uint64 and are +// therefore of type valueEncoder. + +const maxVarintBytes = 10 // maximum length of a varint + +// EncodeVarint returns the varint encoding of x. +// This is the format for the +// int32, int64, uint32, uint64, bool, and enum +// protocol buffer types. +// Not used by the package itself, but helpful to clients +// wishing to use the same encoding. +func EncodeVarint(x uint64) []byte { + var buf [maxVarintBytes]byte + var n int + for n = 0; x > 127; n++ { + buf[n] = 0x80 | uint8(x&0x7F) + x >>= 7 + } + buf[n] = uint8(x) + n++ + return buf[0:n] +} + +// EncodeVarint writes a varint-encoded integer to the Buffer. +// This is the format for the +// int32, int64, uint32, uint64, bool, and enum +// protocol buffer types. +func (p *Buffer) EncodeVarint(x uint64) error { + for x >= 1<<7 { + p.buf = append(p.buf, uint8(x&0x7f|0x80)) + x >>= 7 + } + p.buf = append(p.buf, uint8(x)) + return nil +} + +func sizeVarint(x uint64) (n int) { + for { + n++ + x >>= 7 + if x == 0 { + break + } + } + return n +} + +// EncodeFixed64 writes a 64-bit integer to the Buffer. +// This is the format for the +// fixed64, sfixed64, and double protocol buffer types. +func (p *Buffer) EncodeFixed64(x uint64) error { + p.buf = append(p.buf, + uint8(x), + uint8(x>>8), + uint8(x>>16), + uint8(x>>24), + uint8(x>>32), + uint8(x>>40), + uint8(x>>48), + uint8(x>>56)) + return nil +} + +func sizeFixed64(x uint64) int { + return 8 +} + +// EncodeFixed32 writes a 32-bit integer to the Buffer. +// This is the format for the +// fixed32, sfixed32, and float protocol buffer types. +func (p *Buffer) EncodeFixed32(x uint64) error { + p.buf = append(p.buf, + uint8(x), + uint8(x>>8), + uint8(x>>16), + uint8(x>>24)) + return nil +} + +func sizeFixed32(x uint64) int { + return 4 +} + +// EncodeZigzag64 writes a zigzag-encoded 64-bit integer +// to the Buffer. +// This is the format used for the sint64 protocol buffer type. +func (p *Buffer) EncodeZigzag64(x uint64) error { + // use signed number to get arithmetic right shift. + return p.EncodeVarint(uint64((x << 1) ^ uint64((int64(x) >> 63)))) +} + +func sizeZigzag64(x uint64) int { + return sizeVarint(uint64((x << 1) ^ uint64((int64(x) >> 63)))) +} + +// EncodeZigzag32 writes a zigzag-encoded 32-bit integer +// to the Buffer. +// This is the format used for the sint32 protocol buffer type. +func (p *Buffer) EncodeZigzag32(x uint64) error { + // use signed number to get arithmetic right shift. + return p.EncodeVarint(uint64((uint32(x) << 1) ^ uint32((int32(x) >> 31)))) +} + +func sizeZigzag32(x uint64) int { + return sizeVarint(uint64((uint32(x) << 1) ^ uint32((int32(x) >> 31)))) +} + +// EncodeRawBytes writes a count-delimited byte buffer to the Buffer. +// This is the format used for the bytes protocol buffer +// type and for embedded messages. +func (p *Buffer) EncodeRawBytes(b []byte) error { + p.EncodeVarint(uint64(len(b))) + p.buf = append(p.buf, b...) + return nil +} + +func sizeRawBytes(b []byte) int { + return sizeVarint(uint64(len(b))) + + len(b) +} + +// EncodeStringBytes writes an encoded string to the Buffer. +// This is the format used for the proto2 string type. +func (p *Buffer) EncodeStringBytes(s string) error { + p.EncodeVarint(uint64(len(s))) + p.buf = append(p.buf, s...) + return nil +} + +func sizeStringBytes(s string) int { + return sizeVarint(uint64(len(s))) + + len(s) +} + +// Marshaler is the interface representing objects that can marshal themselves. +type Marshaler interface { + Marshal() ([]byte, error) +} + +// Marshal takes the protocol buffer +// and encodes it into the wire format, returning the data. +func Marshal(pb Message) ([]byte, error) { + // Can the object marshal itself? + if m, ok := pb.(Marshaler); ok { + return m.Marshal() + } + p := NewBuffer(nil) + err := p.Marshal(pb) + var state errorState + if err != nil && !state.shouldContinue(err, nil) { + return nil, err + } + if p.buf == nil && err == nil { + // Return a non-nil slice on success. + return []byte{}, nil + } + return p.buf, err +} + +// Marshal takes the protocol buffer +// and encodes it into the wire format, writing the result to the +// Buffer. +func (p *Buffer) Marshal(pb Message) error { + // Can the object marshal itself? + if m, ok := pb.(Marshaler); ok { + data, err := m.Marshal() + if err != nil { + return err + } + p.buf = append(p.buf, data...) + return nil + } + + t, base, err := getbase(pb) + if structPointer_IsNil(base) { + return ErrNil + } + if err == nil { + err = p.enc_struct(GetProperties(t.Elem()), base) + } + + if collectStats { + stats.Encode++ + } + + return err +} + +// Size returns the encoded size of a protocol buffer. +func Size(pb Message) (n int) { + // Can the object marshal itself? If so, Size is slow. + // TODO: add Size to Marshaler, or add a Sizer interface. + if m, ok := pb.(Marshaler); ok { + b, _ := m.Marshal() + return len(b) + } + + t, base, err := getbase(pb) + if structPointer_IsNil(base) { + return 0 + } + if err == nil { + n = size_struct(GetProperties(t.Elem()), base) + } + + if collectStats { + stats.Size++ + } + + return +} + +// Individual type encoders. + +// Encode a bool. +func (o *Buffer) enc_bool(p *Properties, base structPointer) error { + v := *structPointer_Bool(base, p.field) + if v == nil { + return ErrNil + } + x := 0 + if *v { + x = 1 + } + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, uint64(x)) + return nil +} + +func (o *Buffer) enc_proto3_bool(p *Properties, base structPointer) error { + v := *structPointer_BoolVal(base, p.field) + if !v { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, 1) + return nil +} + +func size_bool(p *Properties, base structPointer) int { + v := *structPointer_Bool(base, p.field) + if v == nil { + return 0 + } + return len(p.tagcode) + 1 // each bool takes exactly one byte +} + +func size_proto3_bool(p *Properties, base structPointer) int { + v := *structPointer_BoolVal(base, p.field) + if !v { + return 0 + } + return len(p.tagcode) + 1 // each bool takes exactly one byte +} + +// Encode an int32. +func (o *Buffer) enc_int32(p *Properties, base structPointer) error { + v := structPointer_Word32(base, p.field) + if word32_IsNil(v) { + return ErrNil + } + x := int32(word32_Get(v)) // permit sign extension to use full 64-bit range + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, uint64(x)) + return nil +} + +func (o *Buffer) enc_proto3_int32(p *Properties, base structPointer) error { + v := structPointer_Word32Val(base, p.field) + x := int32(word32Val_Get(v)) // permit sign extension to use full 64-bit range + if x == 0 { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, uint64(x)) + return nil +} + +func size_int32(p *Properties, base structPointer) (n int) { + v := structPointer_Word32(base, p.field) + if word32_IsNil(v) { + return 0 + } + x := int32(word32_Get(v)) // permit sign extension to use full 64-bit range + n += len(p.tagcode) + n += p.valSize(uint64(x)) + return +} + +func size_proto3_int32(p *Properties, base structPointer) (n int) { + v := structPointer_Word32Val(base, p.field) + x := int32(word32Val_Get(v)) // permit sign extension to use full 64-bit range + if x == 0 { + return 0 + } + n += len(p.tagcode) + n += p.valSize(uint64(x)) + return +} + +// Encode a uint32. +// Exactly the same as int32, except for no sign extension. +func (o *Buffer) enc_uint32(p *Properties, base structPointer) error { + v := structPointer_Word32(base, p.field) + if word32_IsNil(v) { + return ErrNil + } + x := word32_Get(v) + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, uint64(x)) + return nil +} + +func (o *Buffer) enc_proto3_uint32(p *Properties, base structPointer) error { + v := structPointer_Word32Val(base, p.field) + x := word32Val_Get(v) + if x == 0 { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, uint64(x)) + return nil +} + +func size_uint32(p *Properties, base structPointer) (n int) { + v := structPointer_Word32(base, p.field) + if word32_IsNil(v) { + return 0 + } + x := word32_Get(v) + n += len(p.tagcode) + n += p.valSize(uint64(x)) + return +} + +func size_proto3_uint32(p *Properties, base structPointer) (n int) { + v := structPointer_Word32Val(base, p.field) + x := word32Val_Get(v) + if x == 0 { + return 0 + } + n += len(p.tagcode) + n += p.valSize(uint64(x)) + return +} + +// Encode an int64. +func (o *Buffer) enc_int64(p *Properties, base structPointer) error { + v := structPointer_Word64(base, p.field) + if word64_IsNil(v) { + return ErrNil + } + x := word64_Get(v) + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, x) + return nil +} + +func (o *Buffer) enc_proto3_int64(p *Properties, base structPointer) error { + v := structPointer_Word64Val(base, p.field) + x := word64Val_Get(v) + if x == 0 { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, x) + return nil +} + +func size_int64(p *Properties, base structPointer) (n int) { + v := structPointer_Word64(base, p.field) + if word64_IsNil(v) { + return 0 + } + x := word64_Get(v) + n += len(p.tagcode) + n += p.valSize(x) + return +} + +func size_proto3_int64(p *Properties, base structPointer) (n int) { + v := structPointer_Word64Val(base, p.field) + x := word64Val_Get(v) + if x == 0 { + return 0 + } + n += len(p.tagcode) + n += p.valSize(x) + return +} + +// Encode a string. +func (o *Buffer) enc_string(p *Properties, base structPointer) error { + v := *structPointer_String(base, p.field) + if v == nil { + return ErrNil + } + x := *v + o.buf = append(o.buf, p.tagcode...) + o.EncodeStringBytes(x) + return nil +} + +func (o *Buffer) enc_proto3_string(p *Properties, base structPointer) error { + v := *structPointer_StringVal(base, p.field) + if v == "" { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeStringBytes(v) + return nil +} + +func size_string(p *Properties, base structPointer) (n int) { + v := *structPointer_String(base, p.field) + if v == nil { + return 0 + } + x := *v + n += len(p.tagcode) + n += sizeStringBytes(x) + return +} + +func size_proto3_string(p *Properties, base structPointer) (n int) { + v := *structPointer_StringVal(base, p.field) + if v == "" { + return 0 + } + n += len(p.tagcode) + n += sizeStringBytes(v) + return +} + +// All protocol buffer fields are nillable, but be careful. +func isNil(v reflect.Value) bool { + switch v.Kind() { + case reflect.Interface, reflect.Map, reflect.Ptr, reflect.Slice: + return v.IsNil() + } + return false +} + +// Encode a message struct. +func (o *Buffer) enc_struct_message(p *Properties, base structPointer) error { + var state errorState + structp := structPointer_GetStructPointer(base, p.field) + if structPointer_IsNil(structp) { + return ErrNil + } + + // Can the object marshal itself? + if p.isMarshaler { + m := structPointer_Interface(structp, p.stype).(Marshaler) + data, err := m.Marshal() + if err != nil && !state.shouldContinue(err, nil) { + return err + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(data) + return state.err + } + + o.buf = append(o.buf, p.tagcode...) + return o.enc_len_struct(p.sprop, structp, &state) +} + +func size_struct_message(p *Properties, base structPointer) int { + structp := structPointer_GetStructPointer(base, p.field) + if structPointer_IsNil(structp) { + return 0 + } + + // Can the object marshal itself? + if p.isMarshaler { + m := structPointer_Interface(structp, p.stype).(Marshaler) + data, _ := m.Marshal() + n0 := len(p.tagcode) + n1 := sizeRawBytes(data) + return n0 + n1 + } + + n0 := len(p.tagcode) + n1 := size_struct(p.sprop, structp) + n2 := sizeVarint(uint64(n1)) // size of encoded length + return n0 + n1 + n2 +} + +// Encode a group struct. +func (o *Buffer) enc_struct_group(p *Properties, base structPointer) error { + var state errorState + b := structPointer_GetStructPointer(base, p.field) + if structPointer_IsNil(b) { + return ErrNil + } + + o.EncodeVarint(uint64((p.Tag << 3) | WireStartGroup)) + err := o.enc_struct(p.sprop, b) + if err != nil && !state.shouldContinue(err, nil) { + return err + } + o.EncodeVarint(uint64((p.Tag << 3) | WireEndGroup)) + return state.err +} + +func size_struct_group(p *Properties, base structPointer) (n int) { + b := structPointer_GetStructPointer(base, p.field) + if structPointer_IsNil(b) { + return 0 + } + + n += sizeVarint(uint64((p.Tag << 3) | WireStartGroup)) + n += size_struct(p.sprop, b) + n += sizeVarint(uint64((p.Tag << 3) | WireEndGroup)) + return +} + +// Encode a slice of bools ([]bool). +func (o *Buffer) enc_slice_bool(p *Properties, base structPointer) error { + s := *structPointer_BoolSlice(base, p.field) + l := len(s) + if l == 0 { + return ErrNil + } + for _, x := range s { + o.buf = append(o.buf, p.tagcode...) + v := uint64(0) + if x { + v = 1 + } + p.valEnc(o, v) + } + return nil +} + +func size_slice_bool(p *Properties, base structPointer) int { + s := *structPointer_BoolSlice(base, p.field) + l := len(s) + if l == 0 { + return 0 + } + return l * (len(p.tagcode) + 1) // each bool takes exactly one byte +} + +// Encode a slice of bools ([]bool) in packed format. +func (o *Buffer) enc_slice_packed_bool(p *Properties, base structPointer) error { + s := *structPointer_BoolSlice(base, p.field) + l := len(s) + if l == 0 { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeVarint(uint64(l)) // each bool takes exactly one byte + for _, x := range s { + v := uint64(0) + if x { + v = 1 + } + p.valEnc(o, v) + } + return nil +} + +func size_slice_packed_bool(p *Properties, base structPointer) (n int) { + s := *structPointer_BoolSlice(base, p.field) + l := len(s) + if l == 0 { + return 0 + } + n += len(p.tagcode) + n += sizeVarint(uint64(l)) + n += l // each bool takes exactly one byte + return +} + +// Encode a slice of bytes ([]byte). +func (o *Buffer) enc_slice_byte(p *Properties, base structPointer) error { + s := *structPointer_Bytes(base, p.field) + if s == nil { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(s) + return nil +} + +func (o *Buffer) enc_proto3_slice_byte(p *Properties, base structPointer) error { + s := *structPointer_Bytes(base, p.field) + if len(s) == 0 { + return ErrNil + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(s) + return nil +} + +func size_slice_byte(p *Properties, base structPointer) (n int) { + s := *structPointer_Bytes(base, p.field) + if s == nil { + return 0 + } + n += len(p.tagcode) + n += sizeRawBytes(s) + return +} + +func size_proto3_slice_byte(p *Properties, base structPointer) (n int) { + s := *structPointer_Bytes(base, p.field) + if len(s) == 0 { + return 0 + } + n += len(p.tagcode) + n += sizeRawBytes(s) + return +} + +// Encode a slice of int32s ([]int32). +func (o *Buffer) enc_slice_int32(p *Properties, base structPointer) error { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return ErrNil + } + for i := 0; i < l; i++ { + o.buf = append(o.buf, p.tagcode...) + x := int32(s.Index(i)) // permit sign extension to use full 64-bit range + p.valEnc(o, uint64(x)) + } + return nil +} + +func size_slice_int32(p *Properties, base structPointer) (n int) { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return 0 + } + for i := 0; i < l; i++ { + n += len(p.tagcode) + x := int32(s.Index(i)) // permit sign extension to use full 64-bit range + n += p.valSize(uint64(x)) + } + return +} + +// Encode a slice of int32s ([]int32) in packed format. +func (o *Buffer) enc_slice_packed_int32(p *Properties, base structPointer) error { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return ErrNil + } + // TODO: Reuse a Buffer. + buf := NewBuffer(nil) + for i := 0; i < l; i++ { + x := int32(s.Index(i)) // permit sign extension to use full 64-bit range + p.valEnc(buf, uint64(x)) + } + + o.buf = append(o.buf, p.tagcode...) + o.EncodeVarint(uint64(len(buf.buf))) + o.buf = append(o.buf, buf.buf...) + return nil +} + +func size_slice_packed_int32(p *Properties, base structPointer) (n int) { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return 0 + } + var bufSize int + for i := 0; i < l; i++ { + x := int32(s.Index(i)) // permit sign extension to use full 64-bit range + bufSize += p.valSize(uint64(x)) + } + + n += len(p.tagcode) + n += sizeVarint(uint64(bufSize)) + n += bufSize + return +} + +// Encode a slice of uint32s ([]uint32). +// Exactly the same as int32, except for no sign extension. +func (o *Buffer) enc_slice_uint32(p *Properties, base structPointer) error { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return ErrNil + } + for i := 0; i < l; i++ { + o.buf = append(o.buf, p.tagcode...) + x := s.Index(i) + p.valEnc(o, uint64(x)) + } + return nil +} + +func size_slice_uint32(p *Properties, base structPointer) (n int) { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return 0 + } + for i := 0; i < l; i++ { + n += len(p.tagcode) + x := s.Index(i) + n += p.valSize(uint64(x)) + } + return +} + +// Encode a slice of uint32s ([]uint32) in packed format. +// Exactly the same as int32, except for no sign extension. +func (o *Buffer) enc_slice_packed_uint32(p *Properties, base structPointer) error { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return ErrNil + } + // TODO: Reuse a Buffer. + buf := NewBuffer(nil) + for i := 0; i < l; i++ { + p.valEnc(buf, uint64(s.Index(i))) + } + + o.buf = append(o.buf, p.tagcode...) + o.EncodeVarint(uint64(len(buf.buf))) + o.buf = append(o.buf, buf.buf...) + return nil +} + +func size_slice_packed_uint32(p *Properties, base structPointer) (n int) { + s := structPointer_Word32Slice(base, p.field) + l := s.Len() + if l == 0 { + return 0 + } + var bufSize int + for i := 0; i < l; i++ { + bufSize += p.valSize(uint64(s.Index(i))) + } + + n += len(p.tagcode) + n += sizeVarint(uint64(bufSize)) + n += bufSize + return +} + +// Encode a slice of int64s ([]int64). +func (o *Buffer) enc_slice_int64(p *Properties, base structPointer) error { + s := structPointer_Word64Slice(base, p.field) + l := s.Len() + if l == 0 { + return ErrNil + } + for i := 0; i < l; i++ { + o.buf = append(o.buf, p.tagcode...) + p.valEnc(o, s.Index(i)) + } + return nil +} + +func size_slice_int64(p *Properties, base structPointer) (n int) { + s := structPointer_Word64Slice(base, p.field) + l := s.Len() + if l == 0 { + return 0 + } + for i := 0; i < l; i++ { + n += len(p.tagcode) + n += p.valSize(s.Index(i)) + } + return +} + +// Encode a slice of int64s ([]int64) in packed format. +func (o *Buffer) enc_slice_packed_int64(p *Properties, base structPointer) error { + s := structPointer_Word64Slice(base, p.field) + l := s.Len() + if l == 0 { + return ErrNil + } + // TODO: Reuse a Buffer. + buf := NewBuffer(nil) + for i := 0; i < l; i++ { + p.valEnc(buf, s.Index(i)) + } + + o.buf = append(o.buf, p.tagcode...) + o.EncodeVarint(uint64(len(buf.buf))) + o.buf = append(o.buf, buf.buf...) + return nil +} + +func size_slice_packed_int64(p *Properties, base structPointer) (n int) { + s := structPointer_Word64Slice(base, p.field) + l := s.Len() + if l == 0 { + return 0 + } + var bufSize int + for i := 0; i < l; i++ { + bufSize += p.valSize(s.Index(i)) + } + + n += len(p.tagcode) + n += sizeVarint(uint64(bufSize)) + n += bufSize + return +} + +// Encode a slice of slice of bytes ([][]byte). +func (o *Buffer) enc_slice_slice_byte(p *Properties, base structPointer) error { + ss := *structPointer_BytesSlice(base, p.field) + l := len(ss) + if l == 0 { + return ErrNil + } + for i := 0; i < l; i++ { + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(ss[i]) + } + return nil +} + +func size_slice_slice_byte(p *Properties, base structPointer) (n int) { + ss := *structPointer_BytesSlice(base, p.field) + l := len(ss) + if l == 0 { + return 0 + } + n += l * len(p.tagcode) + for i := 0; i < l; i++ { + n += sizeRawBytes(ss[i]) + } + return +} + +// Encode a slice of strings ([]string). +func (o *Buffer) enc_slice_string(p *Properties, base structPointer) error { + ss := *structPointer_StringSlice(base, p.field) + l := len(ss) + for i := 0; i < l; i++ { + o.buf = append(o.buf, p.tagcode...) + o.EncodeStringBytes(ss[i]) + } + return nil +} + +func size_slice_string(p *Properties, base structPointer) (n int) { + ss := *structPointer_StringSlice(base, p.field) + l := len(ss) + n += l * len(p.tagcode) + for i := 0; i < l; i++ { + n += sizeStringBytes(ss[i]) + } + return +} + +// Encode a slice of message structs ([]*struct). +func (o *Buffer) enc_slice_struct_message(p *Properties, base structPointer) error { + var state errorState + s := structPointer_StructPointerSlice(base, p.field) + l := s.Len() + + for i := 0; i < l; i++ { + structp := s.Index(i) + if structPointer_IsNil(structp) { + return errRepeatedHasNil + } + + // Can the object marshal itself? + if p.isMarshaler { + m := structPointer_Interface(structp, p.stype).(Marshaler) + data, err := m.Marshal() + if err != nil && !state.shouldContinue(err, nil) { + return err + } + o.buf = append(o.buf, p.tagcode...) + o.EncodeRawBytes(data) + continue + } + + o.buf = append(o.buf, p.tagcode...) + err := o.enc_len_struct(p.sprop, structp, &state) + if err != nil && !state.shouldContinue(err, nil) { + if err == ErrNil { + return errRepeatedHasNil + } + return err + } + } + return state.err +} + +func size_slice_struct_message(p *Properties, base structPointer) (n int) { + s := structPointer_StructPointerSlice(base, p.field) + l := s.Len() + n += l * len(p.tagcode) + for i := 0; i < l; i++ { + structp := s.Index(i) + if structPointer_IsNil(structp) { + return // return the size up to this point + } + + // Can the object marshal itself? + if p.isMarshaler { + m := structPointer_Interface(structp, p.stype).(Marshaler) + data, _ := m.Marshal() + n += len(p.tagcode) + n += sizeRawBytes(data) + continue + } + + n0 := size_struct(p.sprop, structp) + n1 := sizeVarint(uint64(n0)) // size of encoded length + n += n0 + n1 + } + return +} + +// Encode a slice of group structs ([]*struct). +func (o *Buffer) enc_slice_struct_group(p *Properties, base structPointer) error { + var state errorState + s := structPointer_StructPointerSlice(base, p.field) + l := s.Len() + + for i := 0; i < l; i++ { + b := s.Index(i) + if structPointer_IsNil(b) { + return errRepeatedHasNil + } + + o.EncodeVarint(uint64((p.Tag << 3) | WireStartGroup)) + + err := o.enc_struct(p.sprop, b) + + if err != nil && !state.shouldContinue(err, nil) { + if err == ErrNil { + return errRepeatedHasNil + } + return err + } + + o.EncodeVarint(uint64((p.Tag << 3) | WireEndGroup)) + } + return state.err +} + +func size_slice_struct_group(p *Properties, base structPointer) (n int) { + s := structPointer_StructPointerSlice(base, p.field) + l := s.Len() + + n += l * sizeVarint(uint64((p.Tag<<3)|WireStartGroup)) + n += l * sizeVarint(uint64((p.Tag<<3)|WireEndGroup)) + for i := 0; i < l; i++ { + b := s.Index(i) + if structPointer_IsNil(b) { + return // return size up to this point + } + + n += size_struct(p.sprop, b) + } + return +} + +// Encode an extension map. +func (o *Buffer) enc_map(p *Properties, base structPointer) error { + v := *structPointer_ExtMap(base, p.field) + if err := encodeExtensionMap(v); err != nil { + return err + } + // Fast-path for common cases: zero or one extensions. + if len(v) <= 1 { + for _, e := range v { + o.buf = append(o.buf, e.enc...) + } + return nil + } + + // Sort keys to provide a deterministic encoding. + keys := make([]int, 0, len(v)) + for k := range v { + keys = append(keys, int(k)) + } + sort.Ints(keys) + + for _, k := range keys { + o.buf = append(o.buf, v[int32(k)].enc...) + } + return nil +} + +func size_map(p *Properties, base structPointer) int { + v := *structPointer_ExtMap(base, p.field) + return sizeExtensionMap(v) +} + +// Encode a map field. +func (o *Buffer) enc_new_map(p *Properties, base structPointer) error { + var state errorState // XXX: or do we need to plumb this through? + + /* + A map defined as + map map_field = N; + is encoded in the same way as + message MapFieldEntry { + key_type key = 1; + value_type value = 2; + } + repeated MapFieldEntry map_field = N; + */ + + v := structPointer_NewAt(base, p.field, p.mtype).Elem() // map[K]V + if v.Len() == 0 { + return nil + } + + keycopy, valcopy, keybase, valbase := mapEncodeScratch(p.mtype) + + enc := func() error { + if err := p.mkeyprop.enc(o, p.mkeyprop, keybase); err != nil { + return err + } + if err := p.mvalprop.enc(o, p.mvalprop, valbase); err != nil { + return err + } + return nil + } + + keys := v.MapKeys() + sort.Sort(mapKeys(keys)) + for _, key := range keys { + val := v.MapIndex(key) + + // The only illegal map entry values are nil message pointers. + if val.Kind() == reflect.Ptr && val.IsNil() { + return errors.New("proto: map has nil element") + } + + keycopy.Set(key) + valcopy.Set(val) + + o.buf = append(o.buf, p.tagcode...) + if err := o.enc_len_thing(enc, &state); err != nil { + return err + } + } + return nil +} + +func size_new_map(p *Properties, base structPointer) int { + v := structPointer_NewAt(base, p.field, p.mtype).Elem() // map[K]V + + keycopy, valcopy, keybase, valbase := mapEncodeScratch(p.mtype) + + n := 0 + for _, key := range v.MapKeys() { + val := v.MapIndex(key) + keycopy.Set(key) + valcopy.Set(val) + + // Tag codes for key and val are the responsibility of the sub-sizer. + keysize := p.mkeyprop.size(p.mkeyprop, keybase) + valsize := p.mvalprop.size(p.mvalprop, valbase) + entry := keysize + valsize + // Add on tag code and length of map entry itself. + n += len(p.tagcode) + sizeVarint(uint64(entry)) + entry + } + return n +} + +// mapEncodeScratch returns a new reflect.Value matching the map's value type, +// and a structPointer suitable for passing to an encoder or sizer. +func mapEncodeScratch(mapType reflect.Type) (keycopy, valcopy reflect.Value, keybase, valbase structPointer) { + // Prepare addressable doubly-indirect placeholders for the key and value types. + // This is needed because the element-type encoders expect **T, but the map iteration produces T. + + keycopy = reflect.New(mapType.Key()).Elem() // addressable K + keyptr := reflect.New(reflect.PtrTo(keycopy.Type())).Elem() // addressable *K + keyptr.Set(keycopy.Addr()) // + keybase = toStructPointer(keyptr.Addr()) // **K + + // Value types are more varied and require special handling. + switch mapType.Elem().Kind() { + case reflect.Slice: + // []byte + var dummy []byte + valcopy = reflect.ValueOf(&dummy).Elem() // addressable []byte + valbase = toStructPointer(valcopy.Addr()) + case reflect.Ptr: + // message; the generated field type is map[K]*Msg (so V is *Msg), + // so we only need one level of indirection. + valcopy = reflect.New(mapType.Elem()).Elem() // addressable V + valbase = toStructPointer(valcopy.Addr()) + default: + // everything else + valcopy = reflect.New(mapType.Elem()).Elem() // addressable V + valptr := reflect.New(reflect.PtrTo(valcopy.Type())).Elem() // addressable *V + valptr.Set(valcopy.Addr()) // + valbase = toStructPointer(valptr.Addr()) // **V + } + return +} + +// Encode a struct. +func (o *Buffer) enc_struct(prop *StructProperties, base structPointer) error { + var state errorState + // Encode fields in tag order so that decoders may use optimizations + // that depend on the ordering. + // https://developers.google.com/protocol-buffers/docs/encoding#order + for _, i := range prop.order { + p := prop.Prop[i] + if p.enc != nil { + err := p.enc(o, p, base) + if err != nil { + if err == ErrNil { + if p.Required && state.err == nil { + state.err = &RequiredNotSetError{p.Name} + } + } else if err == errRepeatedHasNil { + // Give more context to nil values in repeated fields. + return errors.New("repeated field " + p.OrigName + " has nil element") + } else if !state.shouldContinue(err, p) { + return err + } + } + } + } + + // Add unrecognized fields at the end. + if prop.unrecField.IsValid() { + v := *structPointer_Bytes(base, prop.unrecField) + if len(v) > 0 { + o.buf = append(o.buf, v...) + } + } + + return state.err +} + +func size_struct(prop *StructProperties, base structPointer) (n int) { + for _, i := range prop.order { + p := prop.Prop[i] + if p.size != nil { + n += p.size(p, base) + } + } + + // Add unrecognized fields at the end. + if prop.unrecField.IsValid() { + v := *structPointer_Bytes(base, prop.unrecField) + n += len(v) + } + + return +} + +var zeroes [20]byte // longer than any conceivable sizeVarint + +// Encode a struct, preceded by its encoded length (as a varint). +func (o *Buffer) enc_len_struct(prop *StructProperties, base structPointer, state *errorState) error { + return o.enc_len_thing(func() error { return o.enc_struct(prop, base) }, state) +} + +// Encode something, preceded by its encoded length (as a varint). +func (o *Buffer) enc_len_thing(enc func() error, state *errorState) error { + iLen := len(o.buf) + o.buf = append(o.buf, 0, 0, 0, 0) // reserve four bytes for length + iMsg := len(o.buf) + err := enc() + if err != nil && !state.shouldContinue(err, nil) { + return err + } + lMsg := len(o.buf) - iMsg + lLen := sizeVarint(uint64(lMsg)) + switch x := lLen - (iMsg - iLen); { + case x > 0: // actual length is x bytes larger than the space we reserved + // Move msg x bytes right. + o.buf = append(o.buf, zeroes[:x]...) + copy(o.buf[iMsg+x:], o.buf[iMsg:iMsg+lMsg]) + case x < 0: // actual length is x bytes smaller than the space we reserved + // Move msg x bytes left. + copy(o.buf[iMsg+x:], o.buf[iMsg:iMsg+lMsg]) + o.buf = o.buf[:len(o.buf)+x] // x is negative + } + // Encode the length in the reserved space. + o.buf = o.buf[:iLen] + o.EncodeVarint(uint64(lMsg)) + o.buf = o.buf[:len(o.buf)+lMsg] + return state.err +} + +// errorState maintains the first error that occurs and updates that error +// with additional context. +type errorState struct { + err error +} + +// shouldContinue reports whether encoding should continue upon encountering the +// given error. If the error is RequiredNotSetError, shouldContinue returns true +// and, if this is the first appearance of that error, remembers it for future +// reporting. +// +// If prop is not nil, it may update any error with additional context about the +// field with the error. +func (s *errorState) shouldContinue(err error, prop *Properties) bool { + // Ignore unset required fields. + reqNotSet, ok := err.(*RequiredNotSetError) + if !ok { + return false + } + if s.err == nil { + if prop != nil { + err = &RequiredNotSetError{prop.Name + "." + reqNotSet.field} + } + s.err = err + } + return true +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/equal.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/equal.go new file mode 100644 index 0000000000000..d8673a3e97aef --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/equal.go @@ -0,0 +1,256 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2011 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// Protocol buffer comparison. +// TODO: MessageSet. + +package proto + +import ( + "bytes" + "log" + "reflect" + "strings" +) + +/* +Equal returns true iff protocol buffers a and b are equal. +The arguments must both be pointers to protocol buffer structs. + +Equality is defined in this way: + - Two messages are equal iff they are the same type, + corresponding fields are equal, unknown field sets + are equal, and extensions sets are equal. + - Two set scalar fields are equal iff their values are equal. + If the fields are of a floating-point type, remember that + NaN != x for all x, including NaN. + - Two repeated fields are equal iff their lengths are the same, + and their corresponding elements are equal (a "bytes" field, + although represented by []byte, is not a repeated field) + - Two unset fields are equal. + - Two unknown field sets are equal if their current + encoded state is equal. + - Two extension sets are equal iff they have corresponding + elements that are pairwise equal. + - Every other combination of things are not equal. + +The return value is undefined if a and b are not protocol buffers. +*/ +func Equal(a, b Message) bool { + if a == nil || b == nil { + return a == b + } + v1, v2 := reflect.ValueOf(a), reflect.ValueOf(b) + if v1.Type() != v2.Type() { + return false + } + if v1.Kind() == reflect.Ptr { + if v1.IsNil() { + return v2.IsNil() + } + if v2.IsNil() { + return false + } + v1, v2 = v1.Elem(), v2.Elem() + } + if v1.Kind() != reflect.Struct { + return false + } + return equalStruct(v1, v2) +} + +// v1 and v2 are known to have the same type. +func equalStruct(v1, v2 reflect.Value) bool { + for i := 0; i < v1.NumField(); i++ { + f := v1.Type().Field(i) + if strings.HasPrefix(f.Name, "XXX_") { + continue + } + f1, f2 := v1.Field(i), v2.Field(i) + if f.Type.Kind() == reflect.Ptr { + if n1, n2 := f1.IsNil(), f2.IsNil(); n1 && n2 { + // both unset + continue + } else if n1 != n2 { + // set/unset mismatch + return false + } + b1, ok := f1.Interface().(raw) + if ok { + b2 := f2.Interface().(raw) + // RawMessage + if !bytes.Equal(b1.Bytes(), b2.Bytes()) { + return false + } + continue + } + f1, f2 = f1.Elem(), f2.Elem() + } + if !equalAny(f1, f2) { + return false + } + } + + if em1 := v1.FieldByName("XXX_extensions"); em1.IsValid() { + em2 := v2.FieldByName("XXX_extensions") + if !equalExtensions(v1.Type(), em1.Interface().(map[int32]Extension), em2.Interface().(map[int32]Extension)) { + return false + } + } + + uf := v1.FieldByName("XXX_unrecognized") + if !uf.IsValid() { + return true + } + + u1 := uf.Bytes() + u2 := v2.FieldByName("XXX_unrecognized").Bytes() + if !bytes.Equal(u1, u2) { + return false + } + + return true +} + +// v1 and v2 are known to have the same type. +func equalAny(v1, v2 reflect.Value) bool { + if v1.Type() == protoMessageType { + m1, _ := v1.Interface().(Message) + m2, _ := v2.Interface().(Message) + return Equal(m1, m2) + } + switch v1.Kind() { + case reflect.Bool: + return v1.Bool() == v2.Bool() + case reflect.Float32, reflect.Float64: + return v1.Float() == v2.Float() + case reflect.Int32, reflect.Int64: + return v1.Int() == v2.Int() + case reflect.Map: + if v1.Len() != v2.Len() { + return false + } + for _, key := range v1.MapKeys() { + val2 := v2.MapIndex(key) + if !val2.IsValid() { + // This key was not found in the second map. + return false + } + if !equalAny(v1.MapIndex(key), val2) { + return false + } + } + return true + case reflect.Ptr: + return equalAny(v1.Elem(), v2.Elem()) + case reflect.Slice: + if v1.Type().Elem().Kind() == reflect.Uint8 { + // short circuit: []byte + if v1.IsNil() != v2.IsNil() { + return false + } + return bytes.Equal(v1.Interface().([]byte), v2.Interface().([]byte)) + } + + if v1.Len() != v2.Len() { + return false + } + for i := 0; i < v1.Len(); i++ { + if !equalAny(v1.Index(i), v2.Index(i)) { + return false + } + } + return true + case reflect.String: + return v1.Interface().(string) == v2.Interface().(string) + case reflect.Struct: + return equalStruct(v1, v2) + case reflect.Uint32, reflect.Uint64: + return v1.Uint() == v2.Uint() + } + + // unknown type, so not a protocol buffer + log.Printf("proto: don't know how to compare %v", v1) + return false +} + +// base is the struct type that the extensions are based on. +// em1 and em2 are extension maps. +func equalExtensions(base reflect.Type, em1, em2 map[int32]Extension) bool { + if len(em1) != len(em2) { + return false + } + + for extNum, e1 := range em1 { + e2, ok := em2[extNum] + if !ok { + return false + } + + m1, m2 := e1.value, e2.value + + if m1 != nil && m2 != nil { + // Both are unencoded. + if !equalAny(reflect.ValueOf(m1), reflect.ValueOf(m2)) { + return false + } + continue + } + + // At least one is encoded. To do a semantically correct comparison + // we need to unmarshal them first. + var desc *ExtensionDesc + if m := extensionMaps[base]; m != nil { + desc = m[extNum] + } + if desc == nil { + log.Printf("proto: don't know how to compare extension %d of %v", extNum, base) + continue + } + var err error + if m1 == nil { + m1, err = decodeExtension(e1.enc, desc) + } + if m2 == nil && err == nil { + m2, err = decodeExtension(e2.enc, desc) + } + if err != nil { + // The encoded form is invalid. + log.Printf("proto: badly encoded extension %d of %v: %v", extNum, base, err) + return false + } + if !equalAny(reflect.ValueOf(m1), reflect.ValueOf(m2)) { + return false + } + } + + return true +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/equal_test.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/equal_test.go new file mode 100644 index 0000000000000..b322f65ab6146 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/equal_test.go @@ -0,0 +1,191 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2011 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "testing" + + . "github.com/golang/protobuf/proto" + pb "github.com/golang/protobuf/proto/testdata" +) + +// Four identical base messages. +// The init function adds extensions to some of them. +var messageWithoutExtension = &pb.MyMessage{Count: Int32(7)} +var messageWithExtension1a = &pb.MyMessage{Count: Int32(7)} +var messageWithExtension1b = &pb.MyMessage{Count: Int32(7)} +var messageWithExtension2 = &pb.MyMessage{Count: Int32(7)} + +// Two messages with non-message extensions. +var messageWithInt32Extension1 = &pb.MyMessage{Count: Int32(8)} +var messageWithInt32Extension2 = &pb.MyMessage{Count: Int32(8)} + +func init() { + ext1 := &pb.Ext{Data: String("Kirk")} + ext2 := &pb.Ext{Data: String("Picard")} + + // messageWithExtension1a has ext1, but never marshals it. + if err := SetExtension(messageWithExtension1a, pb.E_Ext_More, ext1); err != nil { + panic("SetExtension on 1a failed: " + err.Error()) + } + + // messageWithExtension1b is the unmarshaled form of messageWithExtension1a. + if err := SetExtension(messageWithExtension1b, pb.E_Ext_More, ext1); err != nil { + panic("SetExtension on 1b failed: " + err.Error()) + } + buf, err := Marshal(messageWithExtension1b) + if err != nil { + panic("Marshal of 1b failed: " + err.Error()) + } + messageWithExtension1b.Reset() + if err := Unmarshal(buf, messageWithExtension1b); err != nil { + panic("Unmarshal of 1b failed: " + err.Error()) + } + + // messageWithExtension2 has ext2. + if err := SetExtension(messageWithExtension2, pb.E_Ext_More, ext2); err != nil { + panic("SetExtension on 2 failed: " + err.Error()) + } + + if err := SetExtension(messageWithInt32Extension1, pb.E_Ext_Number, Int32(23)); err != nil { + panic("SetExtension on Int32-1 failed: " + err.Error()) + } + if err := SetExtension(messageWithInt32Extension1, pb.E_Ext_Number, Int32(24)); err != nil { + panic("SetExtension on Int32-2 failed: " + err.Error()) + } +} + +var EqualTests = []struct { + desc string + a, b Message + exp bool +}{ + {"different types", &pb.GoEnum{}, &pb.GoTestField{}, false}, + {"equal empty", &pb.GoEnum{}, &pb.GoEnum{}, true}, + {"nil vs nil", nil, nil, true}, + {"typed nil vs typed nil", (*pb.GoEnum)(nil), (*pb.GoEnum)(nil), true}, + {"typed nil vs empty", (*pb.GoEnum)(nil), &pb.GoEnum{}, false}, + {"different typed nil", (*pb.GoEnum)(nil), (*pb.GoTestField)(nil), false}, + + {"one set field, one unset field", &pb.GoTestField{Label: String("foo")}, &pb.GoTestField{}, false}, + {"one set field zero, one unset field", &pb.GoTest{Param: Int32(0)}, &pb.GoTest{}, false}, + {"different set fields", &pb.GoTestField{Label: String("foo")}, &pb.GoTestField{Label: String("bar")}, false}, + {"equal set", &pb.GoTestField{Label: String("foo")}, &pb.GoTestField{Label: String("foo")}, true}, + + {"repeated, one set", &pb.GoTest{F_Int32Repeated: []int32{2, 3}}, &pb.GoTest{}, false}, + {"repeated, different length", &pb.GoTest{F_Int32Repeated: []int32{2, 3}}, &pb.GoTest{F_Int32Repeated: []int32{2}}, false}, + {"repeated, different value", &pb.GoTest{F_Int32Repeated: []int32{2}}, &pb.GoTest{F_Int32Repeated: []int32{3}}, false}, + {"repeated, equal", &pb.GoTest{F_Int32Repeated: []int32{2, 4}}, &pb.GoTest{F_Int32Repeated: []int32{2, 4}}, true}, + {"repeated, nil equal nil", &pb.GoTest{F_Int32Repeated: nil}, &pb.GoTest{F_Int32Repeated: nil}, true}, + {"repeated, nil equal empty", &pb.GoTest{F_Int32Repeated: nil}, &pb.GoTest{F_Int32Repeated: []int32{}}, true}, + {"repeated, empty equal nil", &pb.GoTest{F_Int32Repeated: []int32{}}, &pb.GoTest{F_Int32Repeated: nil}, true}, + + { + "nested, different", + &pb.GoTest{RequiredField: &pb.GoTestField{Label: String("foo")}}, + &pb.GoTest{RequiredField: &pb.GoTestField{Label: String("bar")}}, + false, + }, + { + "nested, equal", + &pb.GoTest{RequiredField: &pb.GoTestField{Label: String("wow")}}, + &pb.GoTest{RequiredField: &pb.GoTestField{Label: String("wow")}}, + true, + }, + + {"bytes", &pb.OtherMessage{Value: []byte("foo")}, &pb.OtherMessage{Value: []byte("foo")}, true}, + {"bytes, empty", &pb.OtherMessage{Value: []byte{}}, &pb.OtherMessage{Value: []byte{}}, true}, + {"bytes, empty vs nil", &pb.OtherMessage{Value: []byte{}}, &pb.OtherMessage{Value: nil}, false}, + { + "repeated bytes", + &pb.MyMessage{RepBytes: [][]byte{[]byte("sham"), []byte("wow")}}, + &pb.MyMessage{RepBytes: [][]byte{[]byte("sham"), []byte("wow")}}, + true, + }, + + {"extension vs. no extension", messageWithoutExtension, messageWithExtension1a, false}, + {"extension vs. same extension", messageWithExtension1a, messageWithExtension1b, true}, + {"extension vs. different extension", messageWithExtension1a, messageWithExtension2, false}, + + {"int32 extension vs. itself", messageWithInt32Extension1, messageWithInt32Extension1, true}, + {"int32 extension vs. a different int32", messageWithInt32Extension1, messageWithInt32Extension2, false}, + + { + "message with group", + &pb.MyMessage{ + Count: Int32(1), + Somegroup: &pb.MyMessage_SomeGroup{ + GroupField: Int32(5), + }, + }, + &pb.MyMessage{ + Count: Int32(1), + Somegroup: &pb.MyMessage_SomeGroup{ + GroupField: Int32(5), + }, + }, + true, + }, + + { + "map same", + &pb.MessageWithMap{NameMapping: map[int32]string{1: "Ken"}}, + &pb.MessageWithMap{NameMapping: map[int32]string{1: "Ken"}}, + true, + }, + { + "map different entry", + &pb.MessageWithMap{NameMapping: map[int32]string{1: "Ken"}}, + &pb.MessageWithMap{NameMapping: map[int32]string{2: "Rob"}}, + false, + }, + { + "map different key only", + &pb.MessageWithMap{NameMapping: map[int32]string{1: "Ken"}}, + &pb.MessageWithMap{NameMapping: map[int32]string{2: "Ken"}}, + false, + }, + { + "map different value only", + &pb.MessageWithMap{NameMapping: map[int32]string{1: "Ken"}}, + &pb.MessageWithMap{NameMapping: map[int32]string{1: "Rob"}}, + false, + }, +} + +func TestEqual(t *testing.T) { + for _, tc := range EqualTests { + if res := Equal(tc.a, tc.b); res != tc.exp { + t.Errorf("%v: Equal(%v, %v) = %v, want %v", tc.desc, tc.a, tc.b, res, tc.exp) + } + } +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/extensions.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/extensions.go new file mode 100644 index 0000000000000..e591ccef79fe6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/extensions.go @@ -0,0 +1,400 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +/* + * Types and routines for supporting protocol buffer extensions. + */ + +import ( + "errors" + "fmt" + "reflect" + "strconv" + "sync" +) + +// ErrMissingExtension is the error returned by GetExtension if the named extension is not in the message. +var ErrMissingExtension = errors.New("proto: missing extension") + +// ExtensionRange represents a range of message extensions for a protocol buffer. +// Used in code generated by the protocol compiler. +type ExtensionRange struct { + Start, End int32 // both inclusive +} + +// extendableProto is an interface implemented by any protocol buffer that may be extended. +type extendableProto interface { + Message + ExtensionRangeArray() []ExtensionRange + ExtensionMap() map[int32]Extension +} + +var extendableProtoType = reflect.TypeOf((*extendableProto)(nil)).Elem() + +// ExtensionDesc represents an extension specification. +// Used in generated code from the protocol compiler. +type ExtensionDesc struct { + ExtendedType Message // nil pointer to the type that is being extended + ExtensionType interface{} // nil pointer to the extension type + Field int32 // field number + Name string // fully-qualified name of extension, for text formatting + Tag string // protobuf tag style +} + +func (ed *ExtensionDesc) repeated() bool { + t := reflect.TypeOf(ed.ExtensionType) + return t.Kind() == reflect.Slice && t.Elem().Kind() != reflect.Uint8 +} + +// Extension represents an extension in a message. +type Extension struct { + // When an extension is stored in a message using SetExtension + // only desc and value are set. When the message is marshaled + // enc will be set to the encoded form of the message. + // + // When a message is unmarshaled and contains extensions, each + // extension will have only enc set. When such an extension is + // accessed using GetExtension (or GetExtensions) desc and value + // will be set. + desc *ExtensionDesc + value interface{} + enc []byte +} + +// SetRawExtension is for testing only. +func SetRawExtension(base extendableProto, id int32, b []byte) { + base.ExtensionMap()[id] = Extension{enc: b} +} + +// isExtensionField returns true iff the given field number is in an extension range. +func isExtensionField(pb extendableProto, field int32) bool { + for _, er := range pb.ExtensionRangeArray() { + if er.Start <= field && field <= er.End { + return true + } + } + return false +} + +// checkExtensionTypes checks that the given extension is valid for pb. +func checkExtensionTypes(pb extendableProto, extension *ExtensionDesc) error { + // Check the extended type. + if a, b := reflect.TypeOf(pb), reflect.TypeOf(extension.ExtendedType); a != b { + return errors.New("proto: bad extended type; " + b.String() + " does not extend " + a.String()) + } + // Check the range. + if !isExtensionField(pb, extension.Field) { + return errors.New("proto: bad extension number; not in declared ranges") + } + return nil +} + +// extPropKey is sufficient to uniquely identify an extension. +type extPropKey struct { + base reflect.Type + field int32 +} + +var extProp = struct { + sync.RWMutex + m map[extPropKey]*Properties +}{ + m: make(map[extPropKey]*Properties), +} + +func extensionProperties(ed *ExtensionDesc) *Properties { + key := extPropKey{base: reflect.TypeOf(ed.ExtendedType), field: ed.Field} + + extProp.RLock() + if prop, ok := extProp.m[key]; ok { + extProp.RUnlock() + return prop + } + extProp.RUnlock() + + extProp.Lock() + defer extProp.Unlock() + // Check again. + if prop, ok := extProp.m[key]; ok { + return prop + } + + prop := new(Properties) + prop.Init(reflect.TypeOf(ed.ExtensionType), "unknown_name", ed.Tag, nil) + extProp.m[key] = prop + return prop +} + +// encodeExtensionMap encodes any unmarshaled (unencoded) extensions in m. +func encodeExtensionMap(m map[int32]Extension) error { + for k, e := range m { + if e.value == nil || e.desc == nil { + // Extension is only in its encoded form. + continue + } + + // We don't skip extensions that have an encoded form set, + // because the extension value may have been mutated after + // the last time this function was called. + + et := reflect.TypeOf(e.desc.ExtensionType) + props := extensionProperties(e.desc) + + p := NewBuffer(nil) + // If e.value has type T, the encoder expects a *struct{ X T }. + // Pass a *T with a zero field and hope it all works out. + x := reflect.New(et) + x.Elem().Set(reflect.ValueOf(e.value)) + if err := props.enc(p, props, toStructPointer(x)); err != nil { + return err + } + e.enc = p.buf + m[k] = e + } + return nil +} + +func sizeExtensionMap(m map[int32]Extension) (n int) { + for _, e := range m { + if e.value == nil || e.desc == nil { + // Extension is only in its encoded form. + n += len(e.enc) + continue + } + + // We don't skip extensions that have an encoded form set, + // because the extension value may have been mutated after + // the last time this function was called. + + et := reflect.TypeOf(e.desc.ExtensionType) + props := extensionProperties(e.desc) + + // If e.value has type T, the encoder expects a *struct{ X T }. + // Pass a *T with a zero field and hope it all works out. + x := reflect.New(et) + x.Elem().Set(reflect.ValueOf(e.value)) + n += props.size(props, toStructPointer(x)) + } + return +} + +// HasExtension returns whether the given extension is present in pb. +func HasExtension(pb extendableProto, extension *ExtensionDesc) bool { + // TODO: Check types, field numbers, etc.? + _, ok := pb.ExtensionMap()[extension.Field] + return ok +} + +// ClearExtension removes the given extension from pb. +func ClearExtension(pb extendableProto, extension *ExtensionDesc) { + // TODO: Check types, field numbers, etc.? + delete(pb.ExtensionMap(), extension.Field) +} + +// GetExtension parses and returns the given extension of pb. +// If the extension is not present and has no default value it returns ErrMissingExtension. +func GetExtension(pb extendableProto, extension *ExtensionDesc) (interface{}, error) { + if err := checkExtensionTypes(pb, extension); err != nil { + return nil, err + } + + emap := pb.ExtensionMap() + e, ok := emap[extension.Field] + if !ok { + // defaultExtensionValue returns the default value or + // ErrMissingExtension if there is no default. + return defaultExtensionValue(extension) + } + + if e.value != nil { + // Already decoded. Check the descriptor, though. + if e.desc != extension { + // This shouldn't happen. If it does, it means that + // GetExtension was called twice with two different + // descriptors with the same field number. + return nil, errors.New("proto: descriptor conflict") + } + return e.value, nil + } + + v, err := decodeExtension(e.enc, extension) + if err != nil { + return nil, err + } + + // Remember the decoded version and drop the encoded version. + // That way it is safe to mutate what we return. + e.value = v + e.desc = extension + e.enc = nil + emap[extension.Field] = e + return e.value, nil +} + +// defaultExtensionValue returns the default value for extension. +// If no default for an extension is defined ErrMissingExtension is returned. +func defaultExtensionValue(extension *ExtensionDesc) (interface{}, error) { + t := reflect.TypeOf(extension.ExtensionType) + props := extensionProperties(extension) + + sf, _, err := fieldDefault(t, props) + if err != nil { + return nil, err + } + + if sf == nil || sf.value == nil { + // There is no default value. + return nil, ErrMissingExtension + } + + if t.Kind() != reflect.Ptr { + // We do not need to return a Ptr, we can directly return sf.value. + return sf.value, nil + } + + // We need to return an interface{} that is a pointer to sf.value. + value := reflect.New(t).Elem() + value.Set(reflect.New(value.Type().Elem())) + if sf.kind == reflect.Int32 { + // We may have an int32 or an enum, but the underlying data is int32. + // Since we can't set an int32 into a non int32 reflect.value directly + // set it as a int32. + value.Elem().SetInt(int64(sf.value.(int32))) + } else { + value.Elem().Set(reflect.ValueOf(sf.value)) + } + return value.Interface(), nil +} + +// decodeExtension decodes an extension encoded in b. +func decodeExtension(b []byte, extension *ExtensionDesc) (interface{}, error) { + o := NewBuffer(b) + + t := reflect.TypeOf(extension.ExtensionType) + rep := extension.repeated() + + props := extensionProperties(extension) + + // t is a pointer to a struct, pointer to basic type or a slice. + // Allocate a "field" to store the pointer/slice itself; the + // pointer/slice will be stored here. We pass + // the address of this field to props.dec. + // This passes a zero field and a *t and lets props.dec + // interpret it as a *struct{ x t }. + value := reflect.New(t).Elem() + + for { + // Discard wire type and field number varint. It isn't needed. + if _, err := o.DecodeVarint(); err != nil { + return nil, err + } + + if err := props.dec(o, props, toStructPointer(value.Addr())); err != nil { + return nil, err + } + + if !rep || o.index >= len(o.buf) { + break + } + } + return value.Interface(), nil +} + +// GetExtensions returns a slice of the extensions present in pb that are also listed in es. +// The returned slice has the same length as es; missing extensions will appear as nil elements. +func GetExtensions(pb Message, es []*ExtensionDesc) (extensions []interface{}, err error) { + epb, ok := pb.(extendableProto) + if !ok { + err = errors.New("proto: not an extendable proto") + return + } + extensions = make([]interface{}, len(es)) + for i, e := range es { + extensions[i], err = GetExtension(epb, e) + if err == ErrMissingExtension { + err = nil + } + if err != nil { + return + } + } + return +} + +// SetExtension sets the specified extension of pb to the specified value. +func SetExtension(pb extendableProto, extension *ExtensionDesc, value interface{}) error { + if err := checkExtensionTypes(pb, extension); err != nil { + return err + } + typ := reflect.TypeOf(extension.ExtensionType) + if typ != reflect.TypeOf(value) { + return errors.New("proto: bad extension value type") + } + // nil extension values need to be caught early, because the + // encoder can't distinguish an ErrNil due to a nil extension + // from an ErrNil due to a missing field. Extensions are + // always optional, so the encoder would just swallow the error + // and drop all the extensions from the encoded message. + if reflect.ValueOf(value).IsNil() { + return fmt.Errorf("proto: SetExtension called with nil value of type %T", value) + } + + pb.ExtensionMap()[extension.Field] = Extension{desc: extension, value: value} + return nil +} + +// A global registry of extensions. +// The generated code will register the generated descriptors by calling RegisterExtension. + +var extensionMaps = make(map[reflect.Type]map[int32]*ExtensionDesc) + +// RegisterExtension is called from the generated code. +func RegisterExtension(desc *ExtensionDesc) { + st := reflect.TypeOf(desc.ExtendedType).Elem() + m := extensionMaps[st] + if m == nil { + m = make(map[int32]*ExtensionDesc) + extensionMaps[st] = m + } + if _, ok := m[desc.Field]; ok { + panic("proto: duplicate extension registered: " + st.String() + " " + strconv.Itoa(int(desc.Field))) + } + m[desc.Field] = desc +} + +// RegisteredExtensions returns a map of the registered extensions of a +// protocol buffer struct, indexed by the extension number. +// The argument pb should be a nil pointer to the struct type. +func RegisteredExtensions(pb Message) map[int32]*ExtensionDesc { + return extensionMaps[reflect.TypeOf(pb).Elem()] +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/extensions_test.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/extensions_test.go new file mode 100644 index 0000000000000..72552767d8d09 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/extensions_test.go @@ -0,0 +1,292 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2014 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "fmt" + "reflect" + "testing" + + "github.com/golang/protobuf/proto" + pb "github.com/golang/protobuf/proto/testdata" +) + +func TestGetExtensionsWithMissingExtensions(t *testing.T) { + msg := &pb.MyMessage{} + ext1 := &pb.Ext{} + if err := proto.SetExtension(msg, pb.E_Ext_More, ext1); err != nil { + t.Fatalf("Could not set ext1: %s", ext1) + } + exts, err := proto.GetExtensions(msg, []*proto.ExtensionDesc{ + pb.E_Ext_More, + pb.E_Ext_Text, + }) + if err != nil { + t.Fatalf("GetExtensions() failed: %s", err) + } + if exts[0] != ext1 { + t.Errorf("ext1 not in returned extensions: %T %v", exts[0], exts[0]) + } + if exts[1] != nil { + t.Errorf("ext2 in returned extensions: %T %v", exts[1], exts[1]) + } +} + +func TestGetExtensionStability(t *testing.T) { + check := func(m *pb.MyMessage) bool { + ext1, err := proto.GetExtension(m, pb.E_Ext_More) + if err != nil { + t.Fatalf("GetExtension() failed: %s", err) + } + ext2, err := proto.GetExtension(m, pb.E_Ext_More) + if err != nil { + t.Fatalf("GetExtension() failed: %s", err) + } + return ext1 == ext2 + } + msg := &pb.MyMessage{Count: proto.Int32(4)} + ext0 := &pb.Ext{} + if err := proto.SetExtension(msg, pb.E_Ext_More, ext0); err != nil { + t.Fatalf("Could not set ext1: %s", ext0) + } + if !check(msg) { + t.Errorf("GetExtension() not stable before marshaling") + } + bb, err := proto.Marshal(msg) + if err != nil { + t.Fatalf("Marshal() failed: %s", err) + } + msg1 := &pb.MyMessage{} + err = proto.Unmarshal(bb, msg1) + if err != nil { + t.Fatalf("Unmarshal() failed: %s", err) + } + if !check(msg1) { + t.Errorf("GetExtension() not stable after unmarshaling") + } +} + +func TestGetExtensionDefaults(t *testing.T) { + var setFloat64 float64 = 1 + var setFloat32 float32 = 2 + var setInt32 int32 = 3 + var setInt64 int64 = 4 + var setUint32 uint32 = 5 + var setUint64 uint64 = 6 + var setBool = true + var setBool2 = false + var setString = "Goodnight string" + var setBytes = []byte("Goodnight bytes") + var setEnum = pb.DefaultsMessage_TWO + + type testcase struct { + ext *proto.ExtensionDesc // Extension we are testing. + want interface{} // Expected value of extension, or nil (meaning that GetExtension will fail). + def interface{} // Expected value of extension after ClearExtension(). + } + tests := []testcase{ + {pb.E_NoDefaultDouble, setFloat64, nil}, + {pb.E_NoDefaultFloat, setFloat32, nil}, + {pb.E_NoDefaultInt32, setInt32, nil}, + {pb.E_NoDefaultInt64, setInt64, nil}, + {pb.E_NoDefaultUint32, setUint32, nil}, + {pb.E_NoDefaultUint64, setUint64, nil}, + {pb.E_NoDefaultSint32, setInt32, nil}, + {pb.E_NoDefaultSint64, setInt64, nil}, + {pb.E_NoDefaultFixed32, setUint32, nil}, + {pb.E_NoDefaultFixed64, setUint64, nil}, + {pb.E_NoDefaultSfixed32, setInt32, nil}, + {pb.E_NoDefaultSfixed64, setInt64, nil}, + {pb.E_NoDefaultBool, setBool, nil}, + {pb.E_NoDefaultBool, setBool2, nil}, + {pb.E_NoDefaultString, setString, nil}, + {pb.E_NoDefaultBytes, setBytes, nil}, + {pb.E_NoDefaultEnum, setEnum, nil}, + {pb.E_DefaultDouble, setFloat64, float64(3.1415)}, + {pb.E_DefaultFloat, setFloat32, float32(3.14)}, + {pb.E_DefaultInt32, setInt32, int32(42)}, + {pb.E_DefaultInt64, setInt64, int64(43)}, + {pb.E_DefaultUint32, setUint32, uint32(44)}, + {pb.E_DefaultUint64, setUint64, uint64(45)}, + {pb.E_DefaultSint32, setInt32, int32(46)}, + {pb.E_DefaultSint64, setInt64, int64(47)}, + {pb.E_DefaultFixed32, setUint32, uint32(48)}, + {pb.E_DefaultFixed64, setUint64, uint64(49)}, + {pb.E_DefaultSfixed32, setInt32, int32(50)}, + {pb.E_DefaultSfixed64, setInt64, int64(51)}, + {pb.E_DefaultBool, setBool, true}, + {pb.E_DefaultBool, setBool2, true}, + {pb.E_DefaultString, setString, "Hello, string"}, + {pb.E_DefaultBytes, setBytes, []byte("Hello, bytes")}, + {pb.E_DefaultEnum, setEnum, pb.DefaultsMessage_ONE}, + } + + checkVal := func(test testcase, msg *pb.DefaultsMessage, valWant interface{}) error { + val, err := proto.GetExtension(msg, test.ext) + if err != nil { + if valWant != nil { + return fmt.Errorf("GetExtension(): %s", err) + } + if want := proto.ErrMissingExtension; err != want { + return fmt.Errorf("Unexpected error: got %v, want %v", err, want) + } + return nil + } + + // All proto2 extension values are either a pointer to a value or a slice of values. + ty := reflect.TypeOf(val) + tyWant := reflect.TypeOf(test.ext.ExtensionType) + if got, want := ty, tyWant; got != want { + return fmt.Errorf("unexpected reflect.TypeOf(): got %v want %v", got, want) + } + tye := ty.Elem() + tyeWant := tyWant.Elem() + if got, want := tye, tyeWant; got != want { + return fmt.Errorf("unexpected reflect.TypeOf().Elem(): got %v want %v", got, want) + } + + // Check the name of the type of the value. + // If it is an enum it will be type int32 with the name of the enum. + if got, want := tye.Name(), tye.Name(); got != want { + return fmt.Errorf("unexpected reflect.TypeOf().Elem().Name(): got %v want %v", got, want) + } + + // Check that value is what we expect. + // If we have a pointer in val, get the value it points to. + valExp := val + if ty.Kind() == reflect.Ptr { + valExp = reflect.ValueOf(val).Elem().Interface() + } + if got, want := valExp, valWant; !reflect.DeepEqual(got, want) { + return fmt.Errorf("unexpected reflect.DeepEqual(): got %v want %v", got, want) + } + + return nil + } + + setTo := func(test testcase) interface{} { + setTo := reflect.ValueOf(test.want) + if typ := reflect.TypeOf(test.ext.ExtensionType); typ.Kind() == reflect.Ptr { + setTo = reflect.New(typ).Elem() + setTo.Set(reflect.New(setTo.Type().Elem())) + setTo.Elem().Set(reflect.ValueOf(test.want)) + } + return setTo.Interface() + } + + for _, test := range tests { + msg := &pb.DefaultsMessage{} + name := test.ext.Name + + // Check the initial value. + if err := checkVal(test, msg, test.def); err != nil { + t.Errorf("%s: %v", name, err) + } + + // Set the per-type value and check value. + name = fmt.Sprintf("%s (set to %T %v)", name, test.want, test.want) + if err := proto.SetExtension(msg, test.ext, setTo(test)); err != nil { + t.Errorf("%s: SetExtension(): %v", name, err) + continue + } + if err := checkVal(test, msg, test.want); err != nil { + t.Errorf("%s: %v", name, err) + continue + } + + // Set and check the value. + name += " (cleared)" + proto.ClearExtension(msg, test.ext) + if err := checkVal(test, msg, test.def); err != nil { + t.Errorf("%s: %v", name, err) + } + } +} + +func TestExtensionsRoundTrip(t *testing.T) { + msg := &pb.MyMessage{} + ext1 := &pb.Ext{ + Data: proto.String("hi"), + } + ext2 := &pb.Ext{ + Data: proto.String("there"), + } + exists := proto.HasExtension(msg, pb.E_Ext_More) + if exists { + t.Error("Extension More present unexpectedly") + } + if err := proto.SetExtension(msg, pb.E_Ext_More, ext1); err != nil { + t.Error(err) + } + if err := proto.SetExtension(msg, pb.E_Ext_More, ext2); err != nil { + t.Error(err) + } + e, err := proto.GetExtension(msg, pb.E_Ext_More) + if err != nil { + t.Error(err) + } + x, ok := e.(*pb.Ext) + if !ok { + t.Errorf("e has type %T, expected testdata.Ext", e) + } else if *x.Data != "there" { + t.Errorf("SetExtension failed to overwrite, got %+v, not 'there'", x) + } + proto.ClearExtension(msg, pb.E_Ext_More) + if _, err = proto.GetExtension(msg, pb.E_Ext_More); err != proto.ErrMissingExtension { + t.Errorf("got %v, expected ErrMissingExtension", e) + } + if _, err := proto.GetExtension(msg, pb.E_X215); err == nil { + t.Error("expected bad extension error, got nil") + } + if err := proto.SetExtension(msg, pb.E_X215, 12); err == nil { + t.Error("expected extension err") + } + if err := proto.SetExtension(msg, pb.E_Ext_More, 12); err == nil { + t.Error("expected some sort of type mismatch error, got nil") + } +} + +func TestNilExtension(t *testing.T) { + msg := &pb.MyMessage{ + Count: proto.Int32(1), + } + if err := proto.SetExtension(msg, pb.E_Ext_Text, proto.String("hello")); err != nil { + t.Fatal(err) + } + if err := proto.SetExtension(msg, pb.E_Ext_More, (*pb.Ext)(nil)); err == nil { + t.Error("expected SetExtension to fail due to a nil extension") + } else if want := "proto: SetExtension called with nil value of type *testdata.Ext"; err.Error() != want { + t.Errorf("expected error %v, got %v", want, err) + } + // Note: if the behavior of Marshal is ever changed to ignore nil extensions, update + // this test to verify that E_Ext_Text is properly propagated through marshal->unmarshal. +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/lib.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/lib.go new file mode 100644 index 0000000000000..95f7975dda814 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/lib.go @@ -0,0 +1,841 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +/* +Package proto converts data structures to and from the wire format of +protocol buffers. It works in concert with the Go source code generated +for .proto files by the protocol compiler. + +A summary of the properties of the protocol buffer interface +for a protocol buffer variable v: + + - Names are turned from camel_case to CamelCase for export. + - There are no methods on v to set fields; just treat + them as structure fields. + - There are getters that return a field's value if set, + and return the field's default value if unset. + The getters work even if the receiver is a nil message. + - The zero value for a struct is its correct initialization state. + All desired fields must be set before marshaling. + - A Reset() method will restore a protobuf struct to its zero state. + - Non-repeated fields are pointers to the values; nil means unset. + That is, optional or required field int32 f becomes F *int32. + - Repeated fields are slices. + - Helper functions are available to aid the setting of fields. + msg.Foo = proto.String("hello") // set field + - Constants are defined to hold the default values of all fields that + have them. They have the form Default_StructName_FieldName. + Because the getter methods handle defaulted values, + direct use of these constants should be rare. + - Enums are given type names and maps from names to values. + Enum values are prefixed by the enclosing message's name, or by the + enum's type name if it is a top-level enum. Enum types have a String + method, and a Enum method to assist in message construction. + - Nested messages, groups and enums have type names prefixed with the name of + the surrounding message type. + - Extensions are given descriptor names that start with E_, + followed by an underscore-delimited list of the nested messages + that contain it (if any) followed by the CamelCased name of the + extension field itself. HasExtension, ClearExtension, GetExtension + and SetExtension are functions for manipulating extensions. + - Marshal and Unmarshal are functions to encode and decode the wire format. + +The simplest way to describe this is to see an example. +Given file test.proto, containing + + package example; + + enum FOO { X = 17; } + + message Test { + required string label = 1; + optional int32 type = 2 [default=77]; + repeated int64 reps = 3; + optional group OptionalGroup = 4 { + required string RequiredField = 5; + } + } + +The resulting file, test.pb.go, is: + + package example + + import proto "github.com/golang/protobuf/proto" + import math "math" + + type FOO int32 + const ( + FOO_X FOO = 17 + ) + var FOO_name = map[int32]string{ + 17: "X", + } + var FOO_value = map[string]int32{ + "X": 17, + } + + func (x FOO) Enum() *FOO { + p := new(FOO) + *p = x + return p + } + func (x FOO) String() string { + return proto.EnumName(FOO_name, int32(x)) + } + func (x *FOO) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(FOO_value, data) + if err != nil { + return err + } + *x = FOO(value) + return nil + } + + type Test struct { + Label *string `protobuf:"bytes,1,req,name=label" json:"label,omitempty"` + Type *int32 `protobuf:"varint,2,opt,name=type,def=77" json:"type,omitempty"` + Reps []int64 `protobuf:"varint,3,rep,name=reps" json:"reps,omitempty"` + Optionalgroup *Test_OptionalGroup `protobuf:"group,4,opt,name=OptionalGroup" json:"optionalgroup,omitempty"` + XXX_unrecognized []byte `json:"-"` + } + func (m *Test) Reset() { *m = Test{} } + func (m *Test) String() string { return proto.CompactTextString(m) } + func (*Test) ProtoMessage() {} + const Default_Test_Type int32 = 77 + + func (m *Test) GetLabel() string { + if m != nil && m.Label != nil { + return *m.Label + } + return "" + } + + func (m *Test) GetType() int32 { + if m != nil && m.Type != nil { + return *m.Type + } + return Default_Test_Type + } + + func (m *Test) GetOptionalgroup() *Test_OptionalGroup { + if m != nil { + return m.Optionalgroup + } + return nil + } + + type Test_OptionalGroup struct { + RequiredField *string `protobuf:"bytes,5,req" json:"RequiredField,omitempty"` + } + func (m *Test_OptionalGroup) Reset() { *m = Test_OptionalGroup{} } + func (m *Test_OptionalGroup) String() string { return proto.CompactTextString(m) } + + func (m *Test_OptionalGroup) GetRequiredField() string { + if m != nil && m.RequiredField != nil { + return *m.RequiredField + } + return "" + } + + func init() { + proto.RegisterEnum("example.FOO", FOO_name, FOO_value) + } + +To create and play with a Test object: + +package main + + import ( + "log" + + "github.com/golang/protobuf/proto" + pb "./example.pb" + ) + + func main() { + test := &pb.Test{ + Label: proto.String("hello"), + Type: proto.Int32(17), + Optionalgroup: &pb.Test_OptionalGroup{ + RequiredField: proto.String("good bye"), + }, + } + data, err := proto.Marshal(test) + if err != nil { + log.Fatal("marshaling error: ", err) + } + newTest := &pb.Test{} + err = proto.Unmarshal(data, newTest) + if err != nil { + log.Fatal("unmarshaling error: ", err) + } + // Now test and newTest contain the same data. + if test.GetLabel() != newTest.GetLabel() { + log.Fatalf("data mismatch %q != %q", test.GetLabel(), newTest.GetLabel()) + } + // etc. + } +*/ +package proto + +import ( + "encoding/json" + "fmt" + "log" + "reflect" + "sort" + "strconv" + "sync" +) + +// Message is implemented by generated protocol buffer messages. +type Message interface { + Reset() + String() string + ProtoMessage() +} + +// Stats records allocation details about the protocol buffer encoders +// and decoders. Useful for tuning the library itself. +type Stats struct { + Emalloc uint64 // mallocs in encode + Dmalloc uint64 // mallocs in decode + Encode uint64 // number of encodes + Decode uint64 // number of decodes + Chit uint64 // number of cache hits + Cmiss uint64 // number of cache misses + Size uint64 // number of sizes +} + +// Set to true to enable stats collection. +const collectStats = false + +var stats Stats + +// GetStats returns a copy of the global Stats structure. +func GetStats() Stats { return stats } + +// A Buffer is a buffer manager for marshaling and unmarshaling +// protocol buffers. It may be reused between invocations to +// reduce memory usage. It is not necessary to use a Buffer; +// the global functions Marshal and Unmarshal create a +// temporary Buffer and are fine for most applications. +type Buffer struct { + buf []byte // encode/decode byte stream + index int // write point + + // pools of basic types to amortize allocation. + bools []bool + uint32s []uint32 + uint64s []uint64 + + // extra pools, only used with pointer_reflect.go + int32s []int32 + int64s []int64 + float32s []float32 + float64s []float64 +} + +// NewBuffer allocates a new Buffer and initializes its internal data to +// the contents of the argument slice. +func NewBuffer(e []byte) *Buffer { + return &Buffer{buf: e} +} + +// Reset resets the Buffer, ready for marshaling a new protocol buffer. +func (p *Buffer) Reset() { + p.buf = p.buf[0:0] // for reading/writing + p.index = 0 // for reading +} + +// SetBuf replaces the internal buffer with the slice, +// ready for unmarshaling the contents of the slice. +func (p *Buffer) SetBuf(s []byte) { + p.buf = s + p.index = 0 +} + +// Bytes returns the contents of the Buffer. +func (p *Buffer) Bytes() []byte { return p.buf } + +/* + * Helper routines for simplifying the creation of optional fields of basic type. + */ + +// Bool is a helper routine that allocates a new bool value +// to store v and returns a pointer to it. +func Bool(v bool) *bool { + return &v +} + +// Int32 is a helper routine that allocates a new int32 value +// to store v and returns a pointer to it. +func Int32(v int32) *int32 { + return &v +} + +// Int is a helper routine that allocates a new int32 value +// to store v and returns a pointer to it, but unlike Int32 +// its argument value is an int. +func Int(v int) *int32 { + p := new(int32) + *p = int32(v) + return p +} + +// Int64 is a helper routine that allocates a new int64 value +// to store v and returns a pointer to it. +func Int64(v int64) *int64 { + return &v +} + +// Float32 is a helper routine that allocates a new float32 value +// to store v and returns a pointer to it. +func Float32(v float32) *float32 { + return &v +} + +// Float64 is a helper routine that allocates a new float64 value +// to store v and returns a pointer to it. +func Float64(v float64) *float64 { + return &v +} + +// Uint32 is a helper routine that allocates a new uint32 value +// to store v and returns a pointer to it. +func Uint32(v uint32) *uint32 { + return &v +} + +// Uint64 is a helper routine that allocates a new uint64 value +// to store v and returns a pointer to it. +func Uint64(v uint64) *uint64 { + return &v +} + +// String is a helper routine that allocates a new string value +// to store v and returns a pointer to it. +func String(v string) *string { + return &v +} + +// EnumName is a helper function to simplify printing protocol buffer enums +// by name. Given an enum map and a value, it returns a useful string. +func EnumName(m map[int32]string, v int32) string { + s, ok := m[v] + if ok { + return s + } + return strconv.Itoa(int(v)) +} + +// UnmarshalJSONEnum is a helper function to simplify recovering enum int values +// from their JSON-encoded representation. Given a map from the enum's symbolic +// names to its int values, and a byte buffer containing the JSON-encoded +// value, it returns an int32 that can be cast to the enum type by the caller. +// +// The function can deal with both JSON representations, numeric and symbolic. +func UnmarshalJSONEnum(m map[string]int32, data []byte, enumName string) (int32, error) { + if data[0] == '"' { + // New style: enums are strings. + var repr string + if err := json.Unmarshal(data, &repr); err != nil { + return -1, err + } + val, ok := m[repr] + if !ok { + return 0, fmt.Errorf("unrecognized enum %s value %q", enumName, repr) + } + return val, nil + } + // Old style: enums are ints. + var val int32 + if err := json.Unmarshal(data, &val); err != nil { + return 0, fmt.Errorf("cannot unmarshal %#q into enum %s", data, enumName) + } + return val, nil +} + +// DebugPrint dumps the encoded data in b in a debugging format with a header +// including the string s. Used in testing but made available for general debugging. +func (p *Buffer) DebugPrint(s string, b []byte) { + var u uint64 + + obuf := p.buf + index := p.index + p.buf = b + p.index = 0 + depth := 0 + + fmt.Printf("\n--- %s ---\n", s) + +out: + for { + for i := 0; i < depth; i++ { + fmt.Print(" ") + } + + index := p.index + if index == len(p.buf) { + break + } + + op, err := p.DecodeVarint() + if err != nil { + fmt.Printf("%3d: fetching op err %v\n", index, err) + break out + } + tag := op >> 3 + wire := op & 7 + + switch wire { + default: + fmt.Printf("%3d: t=%3d unknown wire=%d\n", + index, tag, wire) + break out + + case WireBytes: + var r []byte + + r, err = p.DecodeRawBytes(false) + if err != nil { + break out + } + fmt.Printf("%3d: t=%3d bytes [%d]", index, tag, len(r)) + if len(r) <= 6 { + for i := 0; i < len(r); i++ { + fmt.Printf(" %.2x", r[i]) + } + } else { + for i := 0; i < 3; i++ { + fmt.Printf(" %.2x", r[i]) + } + fmt.Printf(" ..") + for i := len(r) - 3; i < len(r); i++ { + fmt.Printf(" %.2x", r[i]) + } + } + fmt.Printf("\n") + + case WireFixed32: + u, err = p.DecodeFixed32() + if err != nil { + fmt.Printf("%3d: t=%3d fix32 err %v\n", index, tag, err) + break out + } + fmt.Printf("%3d: t=%3d fix32 %d\n", index, tag, u) + + case WireFixed64: + u, err = p.DecodeFixed64() + if err != nil { + fmt.Printf("%3d: t=%3d fix64 err %v\n", index, tag, err) + break out + } + fmt.Printf("%3d: t=%3d fix64 %d\n", index, tag, u) + break + + case WireVarint: + u, err = p.DecodeVarint() + if err != nil { + fmt.Printf("%3d: t=%3d varint err %v\n", index, tag, err) + break out + } + fmt.Printf("%3d: t=%3d varint %d\n", index, tag, u) + + case WireStartGroup: + if err != nil { + fmt.Printf("%3d: t=%3d start err %v\n", index, tag, err) + break out + } + fmt.Printf("%3d: t=%3d start\n", index, tag) + depth++ + + case WireEndGroup: + depth-- + if err != nil { + fmt.Printf("%3d: t=%3d end err %v\n", index, tag, err) + break out + } + fmt.Printf("%3d: t=%3d end\n", index, tag) + } + } + + if depth != 0 { + fmt.Printf("%3d: start-end not balanced %d\n", p.index, depth) + } + fmt.Printf("\n") + + p.buf = obuf + p.index = index +} + +// SetDefaults sets unset protocol buffer fields to their default values. +// It only modifies fields that are both unset and have defined defaults. +// It recursively sets default values in any non-nil sub-messages. +func SetDefaults(pb Message) { + setDefaults(reflect.ValueOf(pb), true, false) +} + +// v is a pointer to a struct. +func setDefaults(v reflect.Value, recur, zeros bool) { + v = v.Elem() + + defaultMu.RLock() + dm, ok := defaults[v.Type()] + defaultMu.RUnlock() + if !ok { + dm = buildDefaultMessage(v.Type()) + defaultMu.Lock() + defaults[v.Type()] = dm + defaultMu.Unlock() + } + + for _, sf := range dm.scalars { + f := v.Field(sf.index) + if !f.IsNil() { + // field already set + continue + } + dv := sf.value + if dv == nil && !zeros { + // no explicit default, and don't want to set zeros + continue + } + fptr := f.Addr().Interface() // **T + // TODO: Consider batching the allocations we do here. + switch sf.kind { + case reflect.Bool: + b := new(bool) + if dv != nil { + *b = dv.(bool) + } + *(fptr.(**bool)) = b + case reflect.Float32: + f := new(float32) + if dv != nil { + *f = dv.(float32) + } + *(fptr.(**float32)) = f + case reflect.Float64: + f := new(float64) + if dv != nil { + *f = dv.(float64) + } + *(fptr.(**float64)) = f + case reflect.Int32: + // might be an enum + if ft := f.Type(); ft != int32PtrType { + // enum + f.Set(reflect.New(ft.Elem())) + if dv != nil { + f.Elem().SetInt(int64(dv.(int32))) + } + } else { + // int32 field + i := new(int32) + if dv != nil { + *i = dv.(int32) + } + *(fptr.(**int32)) = i + } + case reflect.Int64: + i := new(int64) + if dv != nil { + *i = dv.(int64) + } + *(fptr.(**int64)) = i + case reflect.String: + s := new(string) + if dv != nil { + *s = dv.(string) + } + *(fptr.(**string)) = s + case reflect.Uint8: + // exceptional case: []byte + var b []byte + if dv != nil { + db := dv.([]byte) + b = make([]byte, len(db)) + copy(b, db) + } else { + b = []byte{} + } + *(fptr.(*[]byte)) = b + case reflect.Uint32: + u := new(uint32) + if dv != nil { + *u = dv.(uint32) + } + *(fptr.(**uint32)) = u + case reflect.Uint64: + u := new(uint64) + if dv != nil { + *u = dv.(uint64) + } + *(fptr.(**uint64)) = u + default: + log.Printf("proto: can't set default for field %v (sf.kind=%v)", f, sf.kind) + } + } + + for _, ni := range dm.nested { + f := v.Field(ni) + // f is *T or []*T or map[T]*T + switch f.Kind() { + case reflect.Ptr: + if f.IsNil() { + continue + } + setDefaults(f, recur, zeros) + + case reflect.Slice: + for i := 0; i < f.Len(); i++ { + e := f.Index(i) + if e.IsNil() { + continue + } + setDefaults(e, recur, zeros) + } + + case reflect.Map: + for _, k := range f.MapKeys() { + e := f.MapIndex(k) + if e.IsNil() { + continue + } + setDefaults(e, recur, zeros) + } + } + } +} + +var ( + // defaults maps a protocol buffer struct type to a slice of the fields, + // with its scalar fields set to their proto-declared non-zero default values. + defaultMu sync.RWMutex + defaults = make(map[reflect.Type]defaultMessage) + + int32PtrType = reflect.TypeOf((*int32)(nil)) +) + +// defaultMessage represents information about the default values of a message. +type defaultMessage struct { + scalars []scalarField + nested []int // struct field index of nested messages +} + +type scalarField struct { + index int // struct field index + kind reflect.Kind // element type (the T in *T or []T) + value interface{} // the proto-declared default value, or nil +} + +// t is a struct type. +func buildDefaultMessage(t reflect.Type) (dm defaultMessage) { + sprop := GetProperties(t) + for _, prop := range sprop.Prop { + fi, ok := sprop.decoderTags.get(prop.Tag) + if !ok { + // XXX_unrecognized + continue + } + ft := t.Field(fi).Type + + sf, nested, err := fieldDefault(ft, prop) + switch { + case err != nil: + log.Print(err) + case nested: + dm.nested = append(dm.nested, fi) + case sf != nil: + sf.index = fi + dm.scalars = append(dm.scalars, *sf) + } + } + + return dm +} + +// fieldDefault returns the scalarField for field type ft. +// sf will be nil if the field can not have a default. +// nestedMessage will be true if this is a nested message. +// Note that sf.index is not set on return. +func fieldDefault(ft reflect.Type, prop *Properties) (sf *scalarField, nestedMessage bool, err error) { + var canHaveDefault bool + switch ft.Kind() { + case reflect.Ptr: + if ft.Elem().Kind() == reflect.Struct { + nestedMessage = true + } else { + canHaveDefault = true // proto2 scalar field + } + + case reflect.Slice: + switch ft.Elem().Kind() { + case reflect.Ptr: + nestedMessage = true // repeated message + case reflect.Uint8: + canHaveDefault = true // bytes field + } + + case reflect.Map: + if ft.Elem().Kind() == reflect.Ptr { + nestedMessage = true // map with message values + } + } + + if !canHaveDefault { + if nestedMessage { + return nil, true, nil + } + return nil, false, nil + } + + // We now know that ft is a pointer or slice. + sf = &scalarField{kind: ft.Elem().Kind()} + + // scalar fields without defaults + if !prop.HasDefault { + return sf, false, nil + } + + // a scalar field: either *T or []byte + switch ft.Elem().Kind() { + case reflect.Bool: + x, err := strconv.ParseBool(prop.Default) + if err != nil { + return nil, false, fmt.Errorf("proto: bad default bool %q: %v", prop.Default, err) + } + sf.value = x + case reflect.Float32: + x, err := strconv.ParseFloat(prop.Default, 32) + if err != nil { + return nil, false, fmt.Errorf("proto: bad default float32 %q: %v", prop.Default, err) + } + sf.value = float32(x) + case reflect.Float64: + x, err := strconv.ParseFloat(prop.Default, 64) + if err != nil { + return nil, false, fmt.Errorf("proto: bad default float64 %q: %v", prop.Default, err) + } + sf.value = x + case reflect.Int32: + x, err := strconv.ParseInt(prop.Default, 10, 32) + if err != nil { + return nil, false, fmt.Errorf("proto: bad default int32 %q: %v", prop.Default, err) + } + sf.value = int32(x) + case reflect.Int64: + x, err := strconv.ParseInt(prop.Default, 10, 64) + if err != nil { + return nil, false, fmt.Errorf("proto: bad default int64 %q: %v", prop.Default, err) + } + sf.value = x + case reflect.String: + sf.value = prop.Default + case reflect.Uint8: + // []byte (not *uint8) + sf.value = []byte(prop.Default) + case reflect.Uint32: + x, err := strconv.ParseUint(prop.Default, 10, 32) + if err != nil { + return nil, false, fmt.Errorf("proto: bad default uint32 %q: %v", prop.Default, err) + } + sf.value = uint32(x) + case reflect.Uint64: + x, err := strconv.ParseUint(prop.Default, 10, 64) + if err != nil { + return nil, false, fmt.Errorf("proto: bad default uint64 %q: %v", prop.Default, err) + } + sf.value = x + default: + return nil, false, fmt.Errorf("proto: unhandled def kind %v", ft.Elem().Kind()) + } + + return sf, false, nil +} + +// Map fields may have key types of non-float scalars, strings and enums. +// The easiest way to sort them in some deterministic order is to use fmt. +// If this turns out to be inefficient we can always consider other options, +// such as doing a Schwartzian transform. + +func mapKeys(vs []reflect.Value) sort.Interface { + s := mapKeySorter{ + vs: vs, + // default Less function: textual comparison + less: func(a, b reflect.Value) bool { + return fmt.Sprint(a.Interface()) < fmt.Sprint(b.Interface()) + }, + } + + // Type specialization per https://developers.google.com/protocol-buffers/docs/proto#maps; + // numeric keys are sorted numerically. + if len(vs) == 0 { + return s + } + switch vs[0].Kind() { + case reflect.Int32, reflect.Int64: + s.less = func(a, b reflect.Value) bool { return a.Int() < b.Int() } + case reflect.Uint32, reflect.Uint64: + s.less = func(a, b reflect.Value) bool { return a.Uint() < b.Uint() } + } + + return s +} + +type mapKeySorter struct { + vs []reflect.Value + less func(a, b reflect.Value) bool +} + +func (s mapKeySorter) Len() int { return len(s.vs) } +func (s mapKeySorter) Swap(i, j int) { s.vs[i], s.vs[j] = s.vs[j], s.vs[i] } +func (s mapKeySorter) Less(i, j int) bool { + return s.less(s.vs[i], s.vs[j]) +} + +// isProto3Zero reports whether v is a zero proto3 value. +func isProto3Zero(v reflect.Value) bool { + switch v.Kind() { + case reflect.Bool: + return !v.Bool() + case reflect.Int32, reflect.Int64: + return v.Int() == 0 + case reflect.Uint32, reflect.Uint64: + return v.Uint() == 0 + case reflect.Float32, reflect.Float64: + return v.Float() == 0 + case reflect.String: + return v.String() == "" + } + return false +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/message_set.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/message_set.go new file mode 100644 index 0000000000000..9d912bce19bb6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/message_set.go @@ -0,0 +1,287 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +/* + * Support for message sets. + */ + +import ( + "bytes" + "encoding/json" + "errors" + "fmt" + "reflect" + "sort" +) + +// ErrNoMessageTypeId occurs when a protocol buffer does not have a message type ID. +// A message type ID is required for storing a protocol buffer in a message set. +var ErrNoMessageTypeId = errors.New("proto does not have a message type ID") + +// The first two types (_MessageSet_Item and MessageSet) +// model what the protocol compiler produces for the following protocol message: +// message MessageSet { +// repeated group Item = 1 { +// required int32 type_id = 2; +// required string message = 3; +// }; +// } +// That is the MessageSet wire format. We can't use a proto to generate these +// because that would introduce a circular dependency between it and this package. +// +// When a proto1 proto has a field that looks like: +// optional message info = 3; +// the protocol compiler produces a field in the generated struct that looks like: +// Info *_proto_.MessageSet `protobuf:"bytes,3,opt,name=info"` +// The package is automatically inserted so there is no need for that proto file to +// import this package. + +type _MessageSet_Item struct { + TypeId *int32 `protobuf:"varint,2,req,name=type_id"` + Message []byte `protobuf:"bytes,3,req,name=message"` +} + +type MessageSet struct { + Item []*_MessageSet_Item `protobuf:"group,1,rep"` + XXX_unrecognized []byte + // TODO: caching? +} + +// Make sure MessageSet is a Message. +var _ Message = (*MessageSet)(nil) + +// messageTypeIder is an interface satisfied by a protocol buffer type +// that may be stored in a MessageSet. +type messageTypeIder interface { + MessageTypeId() int32 +} + +func (ms *MessageSet) find(pb Message) *_MessageSet_Item { + mti, ok := pb.(messageTypeIder) + if !ok { + return nil + } + id := mti.MessageTypeId() + for _, item := range ms.Item { + if *item.TypeId == id { + return item + } + } + return nil +} + +func (ms *MessageSet) Has(pb Message) bool { + if ms.find(pb) != nil { + return true + } + return false +} + +func (ms *MessageSet) Unmarshal(pb Message) error { + if item := ms.find(pb); item != nil { + return Unmarshal(item.Message, pb) + } + if _, ok := pb.(messageTypeIder); !ok { + return ErrNoMessageTypeId + } + return nil // TODO: return error instead? +} + +func (ms *MessageSet) Marshal(pb Message) error { + msg, err := Marshal(pb) + if err != nil { + return err + } + if item := ms.find(pb); item != nil { + // reuse existing item + item.Message = msg + return nil + } + + mti, ok := pb.(messageTypeIder) + if !ok { + return ErrNoMessageTypeId + } + + mtid := mti.MessageTypeId() + ms.Item = append(ms.Item, &_MessageSet_Item{ + TypeId: &mtid, + Message: msg, + }) + return nil +} + +func (ms *MessageSet) Reset() { *ms = MessageSet{} } +func (ms *MessageSet) String() string { return CompactTextString(ms) } +func (*MessageSet) ProtoMessage() {} + +// Support for the message_set_wire_format message option. + +func skipVarint(buf []byte) []byte { + i := 0 + for ; buf[i]&0x80 != 0; i++ { + } + return buf[i+1:] +} + +// MarshalMessageSet encodes the extension map represented by m in the message set wire format. +// It is called by generated Marshal methods on protocol buffer messages with the message_set_wire_format option. +func MarshalMessageSet(m map[int32]Extension) ([]byte, error) { + if err := encodeExtensionMap(m); err != nil { + return nil, err + } + + // Sort extension IDs to provide a deterministic encoding. + // See also enc_map in encode.go. + ids := make([]int, 0, len(m)) + for id := range m { + ids = append(ids, int(id)) + } + sort.Ints(ids) + + ms := &MessageSet{Item: make([]*_MessageSet_Item, 0, len(m))} + for _, id := range ids { + e := m[int32(id)] + // Remove the wire type and field number varint, as well as the length varint. + msg := skipVarint(skipVarint(e.enc)) + + ms.Item = append(ms.Item, &_MessageSet_Item{ + TypeId: Int32(int32(id)), + Message: msg, + }) + } + return Marshal(ms) +} + +// UnmarshalMessageSet decodes the extension map encoded in buf in the message set wire format. +// It is called by generated Unmarshal methods on protocol buffer messages with the message_set_wire_format option. +func UnmarshalMessageSet(buf []byte, m map[int32]Extension) error { + ms := new(MessageSet) + if err := Unmarshal(buf, ms); err != nil { + return err + } + for _, item := range ms.Item { + id := *item.TypeId + msg := item.Message + + // Restore wire type and field number varint, plus length varint. + // Be careful to preserve duplicate items. + b := EncodeVarint(uint64(id)<<3 | WireBytes) + if ext, ok := m[id]; ok { + // Existing data; rip off the tag and length varint + // so we join the new data correctly. + // We can assume that ext.enc is set because we are unmarshaling. + o := ext.enc[len(b):] // skip wire type and field number + _, n := DecodeVarint(o) // calculate length of length varint + o = o[n:] // skip length varint + msg = append(o, msg...) // join old data and new data + } + b = append(b, EncodeVarint(uint64(len(msg)))...) + b = append(b, msg...) + + m[id] = Extension{enc: b} + } + return nil +} + +// MarshalMessageSetJSON encodes the extension map represented by m in JSON format. +// It is called by generated MarshalJSON methods on protocol buffer messages with the message_set_wire_format option. +func MarshalMessageSetJSON(m map[int32]Extension) ([]byte, error) { + var b bytes.Buffer + b.WriteByte('{') + + // Process the map in key order for deterministic output. + ids := make([]int32, 0, len(m)) + for id := range m { + ids = append(ids, id) + } + sort.Sort(int32Slice(ids)) // int32Slice defined in text.go + + for i, id := range ids { + ext := m[id] + if i > 0 { + b.WriteByte(',') + } + + msd, ok := messageSetMap[id] + if !ok { + // Unknown type; we can't render it, so skip it. + continue + } + fmt.Fprintf(&b, `"[%s]":`, msd.name) + + x := ext.value + if x == nil { + x = reflect.New(msd.t.Elem()).Interface() + if err := Unmarshal(ext.enc, x.(Message)); err != nil { + return nil, err + } + } + d, err := json.Marshal(x) + if err != nil { + return nil, err + } + b.Write(d) + } + b.WriteByte('}') + return b.Bytes(), nil +} + +// UnmarshalMessageSetJSON decodes the extension map encoded in buf in JSON format. +// It is called by generated UnmarshalJSON methods on protocol buffer messages with the message_set_wire_format option. +func UnmarshalMessageSetJSON(buf []byte, m map[int32]Extension) error { + // Common-case fast path. + if len(buf) == 0 || bytes.Equal(buf, []byte("{}")) { + return nil + } + + // This is fairly tricky, and it's not clear that it is needed. + return errors.New("TODO: UnmarshalMessageSetJSON not yet implemented") +} + +// A global registry of types that can be used in a MessageSet. + +var messageSetMap = make(map[int32]messageSetDesc) + +type messageSetDesc struct { + t reflect.Type // pointer to struct + name string +} + +// RegisterMessageSetType is called from the generated code. +func RegisterMessageSetType(m Message, fieldNum int32, name string) { + messageSetMap[fieldNum] = messageSetDesc{ + t: reflect.TypeOf(m), + name: name, + } +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/message_set_test.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/message_set_test.go new file mode 100644 index 0000000000000..7c29bccf4b1d9 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/message_set_test.go @@ -0,0 +1,66 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2014 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +import ( + "bytes" + "testing" +) + +func TestUnmarshalMessageSetWithDuplicate(t *testing.T) { + // Check that a repeated message set entry will be concatenated. + in := &MessageSet{ + Item: []*_MessageSet_Item{ + {TypeId: Int32(12345), Message: []byte("hoo")}, + {TypeId: Int32(12345), Message: []byte("hah")}, + }, + } + b, err := Marshal(in) + if err != nil { + t.Fatalf("Marshal: %v", err) + } + t.Logf("Marshaled bytes: %q", b) + + m := make(map[int32]Extension) + if err := UnmarshalMessageSet(b, m); err != nil { + t.Fatalf("UnmarshalMessageSet: %v", err) + } + ext, ok := m[12345] + if !ok { + t.Fatalf("Didn't retrieve extension 12345; map is %v", m) + } + // Skip wire type/field number and length varints. + got := skipVarint(skipVarint(ext.enc)) + if want := []byte("hoohah"); !bytes.Equal(got, want) { + t.Errorf("Combined extension is %q, want %q", got, want) + } +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/pointer_reflect.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/pointer_reflect.go new file mode 100644 index 0000000000000..749919d250a1a --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/pointer_reflect.go @@ -0,0 +1,479 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2012 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// +build appengine + +// This file contains an implementation of proto field accesses using package reflect. +// It is slower than the code in pointer_unsafe.go but it avoids package unsafe and can +// be used on App Engine. + +package proto + +import ( + "math" + "reflect" +) + +// A structPointer is a pointer to a struct. +type structPointer struct { + v reflect.Value +} + +// toStructPointer returns a structPointer equivalent to the given reflect value. +// The reflect value must itself be a pointer to a struct. +func toStructPointer(v reflect.Value) structPointer { + return structPointer{v} +} + +// IsNil reports whether p is nil. +func structPointer_IsNil(p structPointer) bool { + return p.v.IsNil() +} + +// Interface returns the struct pointer as an interface value. +func structPointer_Interface(p structPointer, _ reflect.Type) interface{} { + return p.v.Interface() +} + +// A field identifies a field in a struct, accessible from a structPointer. +// In this implementation, a field is identified by the sequence of field indices +// passed to reflect's FieldByIndex. +type field []int + +// toField returns a field equivalent to the given reflect field. +func toField(f *reflect.StructField) field { + return f.Index +} + +// invalidField is an invalid field identifier. +var invalidField = field(nil) + +// IsValid reports whether the field identifier is valid. +func (f field) IsValid() bool { return f != nil } + +// field returns the given field in the struct as a reflect value. +func structPointer_field(p structPointer, f field) reflect.Value { + // Special case: an extension map entry with a value of type T + // passes a *T to the struct-handling code with a zero field, + // expecting that it will be treated as equivalent to *struct{ X T }, + // which has the same memory layout. We have to handle that case + // specially, because reflect will panic if we call FieldByIndex on a + // non-struct. + if f == nil { + return p.v.Elem() + } + + return p.v.Elem().FieldByIndex(f) +} + +// ifield returns the given field in the struct as an interface value. +func structPointer_ifield(p structPointer, f field) interface{} { + return structPointer_field(p, f).Addr().Interface() +} + +// Bytes returns the address of a []byte field in the struct. +func structPointer_Bytes(p structPointer, f field) *[]byte { + return structPointer_ifield(p, f).(*[]byte) +} + +// BytesSlice returns the address of a [][]byte field in the struct. +func structPointer_BytesSlice(p structPointer, f field) *[][]byte { + return structPointer_ifield(p, f).(*[][]byte) +} + +// Bool returns the address of a *bool field in the struct. +func structPointer_Bool(p structPointer, f field) **bool { + return structPointer_ifield(p, f).(**bool) +} + +// BoolVal returns the address of a bool field in the struct. +func structPointer_BoolVal(p structPointer, f field) *bool { + return structPointer_ifield(p, f).(*bool) +} + +// BoolSlice returns the address of a []bool field in the struct. +func structPointer_BoolSlice(p structPointer, f field) *[]bool { + return structPointer_ifield(p, f).(*[]bool) +} + +// String returns the address of a *string field in the struct. +func structPointer_String(p structPointer, f field) **string { + return structPointer_ifield(p, f).(**string) +} + +// StringVal returns the address of a string field in the struct. +func structPointer_StringVal(p structPointer, f field) *string { + return structPointer_ifield(p, f).(*string) +} + +// StringSlice returns the address of a []string field in the struct. +func structPointer_StringSlice(p structPointer, f field) *[]string { + return structPointer_ifield(p, f).(*[]string) +} + +// ExtMap returns the address of an extension map field in the struct. +func structPointer_ExtMap(p structPointer, f field) *map[int32]Extension { + return structPointer_ifield(p, f).(*map[int32]Extension) +} + +// NewAt returns the reflect.Value for a pointer to a field in the struct. +func structPointer_NewAt(p structPointer, f field, typ reflect.Type) reflect.Value { + return structPointer_field(p, f).Addr() +} + +// SetStructPointer writes a *struct field in the struct. +func structPointer_SetStructPointer(p structPointer, f field, q structPointer) { + structPointer_field(p, f).Set(q.v) +} + +// GetStructPointer reads a *struct field in the struct. +func structPointer_GetStructPointer(p structPointer, f field) structPointer { + return structPointer{structPointer_field(p, f)} +} + +// StructPointerSlice the address of a []*struct field in the struct. +func structPointer_StructPointerSlice(p structPointer, f field) structPointerSlice { + return structPointerSlice{structPointer_field(p, f)} +} + +// A structPointerSlice represents the address of a slice of pointers to structs +// (themselves messages or groups). That is, v.Type() is *[]*struct{...}. +type structPointerSlice struct { + v reflect.Value +} + +func (p structPointerSlice) Len() int { return p.v.Len() } +func (p structPointerSlice) Index(i int) structPointer { return structPointer{p.v.Index(i)} } +func (p structPointerSlice) Append(q structPointer) { + p.v.Set(reflect.Append(p.v, q.v)) +} + +var ( + int32Type = reflect.TypeOf(int32(0)) + uint32Type = reflect.TypeOf(uint32(0)) + float32Type = reflect.TypeOf(float32(0)) + int64Type = reflect.TypeOf(int64(0)) + uint64Type = reflect.TypeOf(uint64(0)) + float64Type = reflect.TypeOf(float64(0)) +) + +// A word32 represents a field of type *int32, *uint32, *float32, or *enum. +// That is, v.Type() is *int32, *uint32, *float32, or *enum and v is assignable. +type word32 struct { + v reflect.Value +} + +// IsNil reports whether p is nil. +func word32_IsNil(p word32) bool { + return p.v.IsNil() +} + +// Set sets p to point at a newly allocated word with bits set to x. +func word32_Set(p word32, o *Buffer, x uint32) { + t := p.v.Type().Elem() + switch t { + case int32Type: + if len(o.int32s) == 0 { + o.int32s = make([]int32, uint32PoolSize) + } + o.int32s[0] = int32(x) + p.v.Set(reflect.ValueOf(&o.int32s[0])) + o.int32s = o.int32s[1:] + return + case uint32Type: + if len(o.uint32s) == 0 { + o.uint32s = make([]uint32, uint32PoolSize) + } + o.uint32s[0] = x + p.v.Set(reflect.ValueOf(&o.uint32s[0])) + o.uint32s = o.uint32s[1:] + return + case float32Type: + if len(o.float32s) == 0 { + o.float32s = make([]float32, uint32PoolSize) + } + o.float32s[0] = math.Float32frombits(x) + p.v.Set(reflect.ValueOf(&o.float32s[0])) + o.float32s = o.float32s[1:] + return + } + + // must be enum + p.v.Set(reflect.New(t)) + p.v.Elem().SetInt(int64(int32(x))) +} + +// Get gets the bits pointed at by p, as a uint32. +func word32_Get(p word32) uint32 { + elem := p.v.Elem() + switch elem.Kind() { + case reflect.Int32: + return uint32(elem.Int()) + case reflect.Uint32: + return uint32(elem.Uint()) + case reflect.Float32: + return math.Float32bits(float32(elem.Float())) + } + panic("unreachable") +} + +// Word32 returns a reference to a *int32, *uint32, *float32, or *enum field in the struct. +func structPointer_Word32(p structPointer, f field) word32 { + return word32{structPointer_field(p, f)} +} + +// A word32Val represents a field of type int32, uint32, float32, or enum. +// That is, v.Type() is int32, uint32, float32, or enum and v is assignable. +type word32Val struct { + v reflect.Value +} + +// Set sets *p to x. +func word32Val_Set(p word32Val, x uint32) { + switch p.v.Type() { + case int32Type: + p.v.SetInt(int64(x)) + return + case uint32Type: + p.v.SetUint(uint64(x)) + return + case float32Type: + p.v.SetFloat(float64(math.Float32frombits(x))) + return + } + + // must be enum + p.v.SetInt(int64(int32(x))) +} + +// Get gets the bits pointed at by p, as a uint32. +func word32Val_Get(p word32Val) uint32 { + elem := p.v + switch elem.Kind() { + case reflect.Int32: + return uint32(elem.Int()) + case reflect.Uint32: + return uint32(elem.Uint()) + case reflect.Float32: + return math.Float32bits(float32(elem.Float())) + } + panic("unreachable") +} + +// Word32Val returns a reference to a int32, uint32, float32, or enum field in the struct. +func structPointer_Word32Val(p structPointer, f field) word32Val { + return word32Val{structPointer_field(p, f)} +} + +// A word32Slice is a slice of 32-bit values. +// That is, v.Type() is []int32, []uint32, []float32, or []enum. +type word32Slice struct { + v reflect.Value +} + +func (p word32Slice) Append(x uint32) { + n, m := p.v.Len(), p.v.Cap() + if n < m { + p.v.SetLen(n + 1) + } else { + t := p.v.Type().Elem() + p.v.Set(reflect.Append(p.v, reflect.Zero(t))) + } + elem := p.v.Index(n) + switch elem.Kind() { + case reflect.Int32: + elem.SetInt(int64(int32(x))) + case reflect.Uint32: + elem.SetUint(uint64(x)) + case reflect.Float32: + elem.SetFloat(float64(math.Float32frombits(x))) + } +} + +func (p word32Slice) Len() int { + return p.v.Len() +} + +func (p word32Slice) Index(i int) uint32 { + elem := p.v.Index(i) + switch elem.Kind() { + case reflect.Int32: + return uint32(elem.Int()) + case reflect.Uint32: + return uint32(elem.Uint()) + case reflect.Float32: + return math.Float32bits(float32(elem.Float())) + } + panic("unreachable") +} + +// Word32Slice returns a reference to a []int32, []uint32, []float32, or []enum field in the struct. +func structPointer_Word32Slice(p structPointer, f field) word32Slice { + return word32Slice{structPointer_field(p, f)} +} + +// word64 is like word32 but for 64-bit values. +type word64 struct { + v reflect.Value +} + +func word64_Set(p word64, o *Buffer, x uint64) { + t := p.v.Type().Elem() + switch t { + case int64Type: + if len(o.int64s) == 0 { + o.int64s = make([]int64, uint64PoolSize) + } + o.int64s[0] = int64(x) + p.v.Set(reflect.ValueOf(&o.int64s[0])) + o.int64s = o.int64s[1:] + return + case uint64Type: + if len(o.uint64s) == 0 { + o.uint64s = make([]uint64, uint64PoolSize) + } + o.uint64s[0] = x + p.v.Set(reflect.ValueOf(&o.uint64s[0])) + o.uint64s = o.uint64s[1:] + return + case float64Type: + if len(o.float64s) == 0 { + o.float64s = make([]float64, uint64PoolSize) + } + o.float64s[0] = math.Float64frombits(x) + p.v.Set(reflect.ValueOf(&o.float64s[0])) + o.float64s = o.float64s[1:] + return + } + panic("unreachable") +} + +func word64_IsNil(p word64) bool { + return p.v.IsNil() +} + +func word64_Get(p word64) uint64 { + elem := p.v.Elem() + switch elem.Kind() { + case reflect.Int64: + return uint64(elem.Int()) + case reflect.Uint64: + return elem.Uint() + case reflect.Float64: + return math.Float64bits(elem.Float()) + } + panic("unreachable") +} + +func structPointer_Word64(p structPointer, f field) word64 { + return word64{structPointer_field(p, f)} +} + +// word64Val is like word32Val but for 64-bit values. +type word64Val struct { + v reflect.Value +} + +func word64Val_Set(p word64Val, o *Buffer, x uint64) { + switch p.v.Type() { + case int64Type: + p.v.SetInt(int64(x)) + return + case uint64Type: + p.v.SetUint(x) + return + case float64Type: + p.v.SetFloat(math.Float64frombits(x)) + return + } + panic("unreachable") +} + +func word64Val_Get(p word64Val) uint64 { + elem := p.v + switch elem.Kind() { + case reflect.Int64: + return uint64(elem.Int()) + case reflect.Uint64: + return elem.Uint() + case reflect.Float64: + return math.Float64bits(elem.Float()) + } + panic("unreachable") +} + +func structPointer_Word64Val(p structPointer, f field) word64Val { + return word64Val{structPointer_field(p, f)} +} + +type word64Slice struct { + v reflect.Value +} + +func (p word64Slice) Append(x uint64) { + n, m := p.v.Len(), p.v.Cap() + if n < m { + p.v.SetLen(n + 1) + } else { + t := p.v.Type().Elem() + p.v.Set(reflect.Append(p.v, reflect.Zero(t))) + } + elem := p.v.Index(n) + switch elem.Kind() { + case reflect.Int64: + elem.SetInt(int64(int64(x))) + case reflect.Uint64: + elem.SetUint(uint64(x)) + case reflect.Float64: + elem.SetFloat(float64(math.Float64frombits(x))) + } +} + +func (p word64Slice) Len() int { + return p.v.Len() +} + +func (p word64Slice) Index(i int) uint64 { + elem := p.v.Index(i) + switch elem.Kind() { + case reflect.Int64: + return uint64(elem.Int()) + case reflect.Uint64: + return uint64(elem.Uint()) + case reflect.Float64: + return math.Float64bits(float64(elem.Float())) + } + panic("unreachable") +} + +func structPointer_Word64Slice(p structPointer, f field) word64Slice { + return word64Slice{structPointer_field(p, f)} +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/pointer_unsafe.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/pointer_unsafe.go new file mode 100644 index 0000000000000..e9be0fe92ee70 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/pointer_unsafe.go @@ -0,0 +1,266 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2012 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// +build !appengine + +// This file contains the implementation of the proto field accesses using package unsafe. + +package proto + +import ( + "reflect" + "unsafe" +) + +// NOTE: These type_Foo functions would more idiomatically be methods, +// but Go does not allow methods on pointer types, and we must preserve +// some pointer type for the garbage collector. We use these +// funcs with clunky names as our poor approximation to methods. +// +// An alternative would be +// type structPointer struct { p unsafe.Pointer } +// but that does not registerize as well. + +// A structPointer is a pointer to a struct. +type structPointer unsafe.Pointer + +// toStructPointer returns a structPointer equivalent to the given reflect value. +func toStructPointer(v reflect.Value) structPointer { + return structPointer(unsafe.Pointer(v.Pointer())) +} + +// IsNil reports whether p is nil. +func structPointer_IsNil(p structPointer) bool { + return p == nil +} + +// Interface returns the struct pointer, assumed to have element type t, +// as an interface value. +func structPointer_Interface(p structPointer, t reflect.Type) interface{} { + return reflect.NewAt(t, unsafe.Pointer(p)).Interface() +} + +// A field identifies a field in a struct, accessible from a structPointer. +// In this implementation, a field is identified by its byte offset from the start of the struct. +type field uintptr + +// toField returns a field equivalent to the given reflect field. +func toField(f *reflect.StructField) field { + return field(f.Offset) +} + +// invalidField is an invalid field identifier. +const invalidField = ^field(0) + +// IsValid reports whether the field identifier is valid. +func (f field) IsValid() bool { + return f != ^field(0) +} + +// Bytes returns the address of a []byte field in the struct. +func structPointer_Bytes(p structPointer, f field) *[]byte { + return (*[]byte)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// BytesSlice returns the address of a [][]byte field in the struct. +func structPointer_BytesSlice(p structPointer, f field) *[][]byte { + return (*[][]byte)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// Bool returns the address of a *bool field in the struct. +func structPointer_Bool(p structPointer, f field) **bool { + return (**bool)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// BoolVal returns the address of a bool field in the struct. +func structPointer_BoolVal(p structPointer, f field) *bool { + return (*bool)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// BoolSlice returns the address of a []bool field in the struct. +func structPointer_BoolSlice(p structPointer, f field) *[]bool { + return (*[]bool)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// String returns the address of a *string field in the struct. +func structPointer_String(p structPointer, f field) **string { + return (**string)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// StringVal returns the address of a string field in the struct. +func structPointer_StringVal(p structPointer, f field) *string { + return (*string)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// StringSlice returns the address of a []string field in the struct. +func structPointer_StringSlice(p structPointer, f field) *[]string { + return (*[]string)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// ExtMap returns the address of an extension map field in the struct. +func structPointer_ExtMap(p structPointer, f field) *map[int32]Extension { + return (*map[int32]Extension)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// NewAt returns the reflect.Value for a pointer to a field in the struct. +func structPointer_NewAt(p structPointer, f field, typ reflect.Type) reflect.Value { + return reflect.NewAt(typ, unsafe.Pointer(uintptr(p)+uintptr(f))) +} + +// SetStructPointer writes a *struct field in the struct. +func structPointer_SetStructPointer(p structPointer, f field, q structPointer) { + *(*structPointer)(unsafe.Pointer(uintptr(p) + uintptr(f))) = q +} + +// GetStructPointer reads a *struct field in the struct. +func structPointer_GetStructPointer(p structPointer, f field) structPointer { + return *(*structPointer)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// StructPointerSlice the address of a []*struct field in the struct. +func structPointer_StructPointerSlice(p structPointer, f field) *structPointerSlice { + return (*structPointerSlice)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// A structPointerSlice represents a slice of pointers to structs (themselves submessages or groups). +type structPointerSlice []structPointer + +func (v *structPointerSlice) Len() int { return len(*v) } +func (v *structPointerSlice) Index(i int) structPointer { return (*v)[i] } +func (v *structPointerSlice) Append(p structPointer) { *v = append(*v, p) } + +// A word32 is the address of a "pointer to 32-bit value" field. +type word32 **uint32 + +// IsNil reports whether *v is nil. +func word32_IsNil(p word32) bool { + return *p == nil +} + +// Set sets *v to point at a newly allocated word set to x. +func word32_Set(p word32, o *Buffer, x uint32) { + if len(o.uint32s) == 0 { + o.uint32s = make([]uint32, uint32PoolSize) + } + o.uint32s[0] = x + *p = &o.uint32s[0] + o.uint32s = o.uint32s[1:] +} + +// Get gets the value pointed at by *v. +func word32_Get(p word32) uint32 { + return **p +} + +// Word32 returns the address of a *int32, *uint32, *float32, or *enum field in the struct. +func structPointer_Word32(p structPointer, f field) word32 { + return word32((**uint32)(unsafe.Pointer(uintptr(p) + uintptr(f)))) +} + +// A word32Val is the address of a 32-bit value field. +type word32Val *uint32 + +// Set sets *p to x. +func word32Val_Set(p word32Val, x uint32) { + *p = x +} + +// Get gets the value pointed at by p. +func word32Val_Get(p word32Val) uint32 { + return *p +} + +// Word32Val returns the address of a *int32, *uint32, *float32, or *enum field in the struct. +func structPointer_Word32Val(p structPointer, f field) word32Val { + return word32Val((*uint32)(unsafe.Pointer(uintptr(p) + uintptr(f)))) +} + +// A word32Slice is a slice of 32-bit values. +type word32Slice []uint32 + +func (v *word32Slice) Append(x uint32) { *v = append(*v, x) } +func (v *word32Slice) Len() int { return len(*v) } +func (v *word32Slice) Index(i int) uint32 { return (*v)[i] } + +// Word32Slice returns the address of a []int32, []uint32, []float32, or []enum field in the struct. +func structPointer_Word32Slice(p structPointer, f field) *word32Slice { + return (*word32Slice)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} + +// word64 is like word32 but for 64-bit values. +type word64 **uint64 + +func word64_Set(p word64, o *Buffer, x uint64) { + if len(o.uint64s) == 0 { + o.uint64s = make([]uint64, uint64PoolSize) + } + o.uint64s[0] = x + *p = &o.uint64s[0] + o.uint64s = o.uint64s[1:] +} + +func word64_IsNil(p word64) bool { + return *p == nil +} + +func word64_Get(p word64) uint64 { + return **p +} + +func structPointer_Word64(p structPointer, f field) word64 { + return word64((**uint64)(unsafe.Pointer(uintptr(p) + uintptr(f)))) +} + +// word64Val is like word32Val but for 64-bit values. +type word64Val *uint64 + +func word64Val_Set(p word64Val, o *Buffer, x uint64) { + *p = x +} + +func word64Val_Get(p word64Val) uint64 { + return *p +} + +func structPointer_Word64Val(p structPointer, f field) word64Val { + return word64Val((*uint64)(unsafe.Pointer(uintptr(p) + uintptr(f)))) +} + +// word64Slice is like word32Slice but for 64-bit values. +type word64Slice []uint64 + +func (v *word64Slice) Append(x uint64) { *v = append(*v, x) } +func (v *word64Slice) Len() int { return len(*v) } +func (v *word64Slice) Index(i int) uint64 { return (*v)[i] } + +func structPointer_Word64Slice(p structPointer, f field) *word64Slice { + return (*word64Slice)(unsafe.Pointer(uintptr(p) + uintptr(f))) +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/properties.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/properties.go new file mode 100644 index 0000000000000..d74844ab2a289 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/properties.go @@ -0,0 +1,742 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +/* + * Routines for encoding data into the wire format for protocol buffers. + */ + +import ( + "fmt" + "os" + "reflect" + "sort" + "strconv" + "strings" + "sync" +) + +const debug bool = false + +// Constants that identify the encoding of a value on the wire. +const ( + WireVarint = 0 + WireFixed64 = 1 + WireBytes = 2 + WireStartGroup = 3 + WireEndGroup = 4 + WireFixed32 = 5 +) + +const startSize = 10 // initial slice/string sizes + +// Encoders are defined in encode.go +// An encoder outputs the full representation of a field, including its +// tag and encoder type. +type encoder func(p *Buffer, prop *Properties, base structPointer) error + +// A valueEncoder encodes a single integer in a particular encoding. +type valueEncoder func(o *Buffer, x uint64) error + +// Sizers are defined in encode.go +// A sizer returns the encoded size of a field, including its tag and encoder +// type. +type sizer func(prop *Properties, base structPointer) int + +// A valueSizer returns the encoded size of a single integer in a particular +// encoding. +type valueSizer func(x uint64) int + +// Decoders are defined in decode.go +// A decoder creates a value from its wire representation. +// Unrecognized subelements are saved in unrec. +type decoder func(p *Buffer, prop *Properties, base structPointer) error + +// A valueDecoder decodes a single integer in a particular encoding. +type valueDecoder func(o *Buffer) (x uint64, err error) + +// tagMap is an optimization over map[int]int for typical protocol buffer +// use-cases. Encoded protocol buffers are often in tag order with small tag +// numbers. +type tagMap struct { + fastTags []int + slowTags map[int]int +} + +// tagMapFastLimit is the upper bound on the tag number that will be stored in +// the tagMap slice rather than its map. +const tagMapFastLimit = 1024 + +func (p *tagMap) get(t int) (int, bool) { + if t > 0 && t < tagMapFastLimit { + if t >= len(p.fastTags) { + return 0, false + } + fi := p.fastTags[t] + return fi, fi >= 0 + } + fi, ok := p.slowTags[t] + return fi, ok +} + +func (p *tagMap) put(t int, fi int) { + if t > 0 && t < tagMapFastLimit { + for len(p.fastTags) < t+1 { + p.fastTags = append(p.fastTags, -1) + } + p.fastTags[t] = fi + return + } + if p.slowTags == nil { + p.slowTags = make(map[int]int) + } + p.slowTags[t] = fi +} + +// StructProperties represents properties for all the fields of a struct. +// decoderTags and decoderOrigNames should only be used by the decoder. +type StructProperties struct { + Prop []*Properties // properties for each field + reqCount int // required count + decoderTags tagMap // map from proto tag to struct field number + decoderOrigNames map[string]int // map from original name to struct field number + order []int // list of struct field numbers in tag order + unrecField field // field id of the XXX_unrecognized []byte field + extendable bool // is this an extendable proto +} + +// Implement the sorting interface so we can sort the fields in tag order, as recommended by the spec. +// See encode.go, (*Buffer).enc_struct. + +func (sp *StructProperties) Len() int { return len(sp.order) } +func (sp *StructProperties) Less(i, j int) bool { + return sp.Prop[sp.order[i]].Tag < sp.Prop[sp.order[j]].Tag +} +func (sp *StructProperties) Swap(i, j int) { sp.order[i], sp.order[j] = sp.order[j], sp.order[i] } + +// Properties represents the protocol-specific behavior of a single struct field. +type Properties struct { + Name string // name of the field, for error messages + OrigName string // original name before protocol compiler (always set) + Wire string + WireType int + Tag int + Required bool + Optional bool + Repeated bool + Packed bool // relevant for repeated primitives only + Enum string // set for enum types only + proto3 bool // whether this is known to be a proto3 field; set for []byte only + + Default string // default value + HasDefault bool // whether an explicit default was provided + def_uint64 uint64 + + enc encoder + valEnc valueEncoder // set for bool and numeric types only + field field + tagcode []byte // encoding of EncodeVarint((Tag<<3)|WireType) + tagbuf [8]byte + stype reflect.Type // set for struct types only + sprop *StructProperties // set for struct types only + isMarshaler bool + isUnmarshaler bool + + mtype reflect.Type // set for map types only + mkeyprop *Properties // set for map types only + mvalprop *Properties // set for map types only + + size sizer + valSize valueSizer // set for bool and numeric types only + + dec decoder + valDec valueDecoder // set for bool and numeric types only + + // If this is a packable field, this will be the decoder for the packed version of the field. + packedDec decoder +} + +// String formats the properties in the protobuf struct field tag style. +func (p *Properties) String() string { + s := p.Wire + s = "," + s += strconv.Itoa(p.Tag) + if p.Required { + s += ",req" + } + if p.Optional { + s += ",opt" + } + if p.Repeated { + s += ",rep" + } + if p.Packed { + s += ",packed" + } + if p.OrigName != p.Name { + s += ",name=" + p.OrigName + } + if p.proto3 { + s += ",proto3" + } + if len(p.Enum) > 0 { + s += ",enum=" + p.Enum + } + if p.HasDefault { + s += ",def=" + p.Default + } + return s +} + +// Parse populates p by parsing a string in the protobuf struct field tag style. +func (p *Properties) Parse(s string) { + // "bytes,49,opt,name=foo,def=hello!" + fields := strings.Split(s, ",") // breaks def=, but handled below. + if len(fields) < 2 { + fmt.Fprintf(os.Stderr, "proto: tag has too few fields: %q\n", s) + return + } + + p.Wire = fields[0] + switch p.Wire { + case "varint": + p.WireType = WireVarint + p.valEnc = (*Buffer).EncodeVarint + p.valDec = (*Buffer).DecodeVarint + p.valSize = sizeVarint + case "fixed32": + p.WireType = WireFixed32 + p.valEnc = (*Buffer).EncodeFixed32 + p.valDec = (*Buffer).DecodeFixed32 + p.valSize = sizeFixed32 + case "fixed64": + p.WireType = WireFixed64 + p.valEnc = (*Buffer).EncodeFixed64 + p.valDec = (*Buffer).DecodeFixed64 + p.valSize = sizeFixed64 + case "zigzag32": + p.WireType = WireVarint + p.valEnc = (*Buffer).EncodeZigzag32 + p.valDec = (*Buffer).DecodeZigzag32 + p.valSize = sizeZigzag32 + case "zigzag64": + p.WireType = WireVarint + p.valEnc = (*Buffer).EncodeZigzag64 + p.valDec = (*Buffer).DecodeZigzag64 + p.valSize = sizeZigzag64 + case "bytes", "group": + p.WireType = WireBytes + // no numeric converter for non-numeric types + default: + fmt.Fprintf(os.Stderr, "proto: tag has unknown wire type: %q\n", s) + return + } + + var err error + p.Tag, err = strconv.Atoi(fields[1]) + if err != nil { + return + } + + for i := 2; i < len(fields); i++ { + f := fields[i] + switch { + case f == "req": + p.Required = true + case f == "opt": + p.Optional = true + case f == "rep": + p.Repeated = true + case f == "packed": + p.Packed = true + case strings.HasPrefix(f, "name="): + p.OrigName = f[5:] + case strings.HasPrefix(f, "enum="): + p.Enum = f[5:] + case f == "proto3": + p.proto3 = true + case strings.HasPrefix(f, "def="): + p.HasDefault = true + p.Default = f[4:] // rest of string + if i+1 < len(fields) { + // Commas aren't escaped, and def is always last. + p.Default += "," + strings.Join(fields[i+1:], ",") + break + } + } + } +} + +func logNoSliceEnc(t1, t2 reflect.Type) { + fmt.Fprintf(os.Stderr, "proto: no slice oenc for %T = []%T\n", t1, t2) +} + +var protoMessageType = reflect.TypeOf((*Message)(nil)).Elem() + +// Initialize the fields for encoding and decoding. +func (p *Properties) setEncAndDec(typ reflect.Type, f *reflect.StructField, lockGetProp bool) { + p.enc = nil + p.dec = nil + p.size = nil + + switch t1 := typ; t1.Kind() { + default: + fmt.Fprintf(os.Stderr, "proto: no coders for %v\n", t1) + + // proto3 scalar types + + case reflect.Bool: + p.enc = (*Buffer).enc_proto3_bool + p.dec = (*Buffer).dec_proto3_bool + p.size = size_proto3_bool + case reflect.Int32: + p.enc = (*Buffer).enc_proto3_int32 + p.dec = (*Buffer).dec_proto3_int32 + p.size = size_proto3_int32 + case reflect.Uint32: + p.enc = (*Buffer).enc_proto3_uint32 + p.dec = (*Buffer).dec_proto3_int32 // can reuse + p.size = size_proto3_uint32 + case reflect.Int64, reflect.Uint64: + p.enc = (*Buffer).enc_proto3_int64 + p.dec = (*Buffer).dec_proto3_int64 + p.size = size_proto3_int64 + case reflect.Float32: + p.enc = (*Buffer).enc_proto3_uint32 // can just treat them as bits + p.dec = (*Buffer).dec_proto3_int32 + p.size = size_proto3_uint32 + case reflect.Float64: + p.enc = (*Buffer).enc_proto3_int64 // can just treat them as bits + p.dec = (*Buffer).dec_proto3_int64 + p.size = size_proto3_int64 + case reflect.String: + p.enc = (*Buffer).enc_proto3_string + p.dec = (*Buffer).dec_proto3_string + p.size = size_proto3_string + + case reflect.Ptr: + switch t2 := t1.Elem(); t2.Kind() { + default: + fmt.Fprintf(os.Stderr, "proto: no encoder function for %v -> %v\n", t1, t2) + break + case reflect.Bool: + p.enc = (*Buffer).enc_bool + p.dec = (*Buffer).dec_bool + p.size = size_bool + case reflect.Int32: + p.enc = (*Buffer).enc_int32 + p.dec = (*Buffer).dec_int32 + p.size = size_int32 + case reflect.Uint32: + p.enc = (*Buffer).enc_uint32 + p.dec = (*Buffer).dec_int32 // can reuse + p.size = size_uint32 + case reflect.Int64, reflect.Uint64: + p.enc = (*Buffer).enc_int64 + p.dec = (*Buffer).dec_int64 + p.size = size_int64 + case reflect.Float32: + p.enc = (*Buffer).enc_uint32 // can just treat them as bits + p.dec = (*Buffer).dec_int32 + p.size = size_uint32 + case reflect.Float64: + p.enc = (*Buffer).enc_int64 // can just treat them as bits + p.dec = (*Buffer).dec_int64 + p.size = size_int64 + case reflect.String: + p.enc = (*Buffer).enc_string + p.dec = (*Buffer).dec_string + p.size = size_string + case reflect.Struct: + p.stype = t1.Elem() + p.isMarshaler = isMarshaler(t1) + p.isUnmarshaler = isUnmarshaler(t1) + if p.Wire == "bytes" { + p.enc = (*Buffer).enc_struct_message + p.dec = (*Buffer).dec_struct_message + p.size = size_struct_message + } else { + p.enc = (*Buffer).enc_struct_group + p.dec = (*Buffer).dec_struct_group + p.size = size_struct_group + } + } + + case reflect.Slice: + switch t2 := t1.Elem(); t2.Kind() { + default: + logNoSliceEnc(t1, t2) + break + case reflect.Bool: + if p.Packed { + p.enc = (*Buffer).enc_slice_packed_bool + p.size = size_slice_packed_bool + } else { + p.enc = (*Buffer).enc_slice_bool + p.size = size_slice_bool + } + p.dec = (*Buffer).dec_slice_bool + p.packedDec = (*Buffer).dec_slice_packed_bool + case reflect.Int32: + if p.Packed { + p.enc = (*Buffer).enc_slice_packed_int32 + p.size = size_slice_packed_int32 + } else { + p.enc = (*Buffer).enc_slice_int32 + p.size = size_slice_int32 + } + p.dec = (*Buffer).dec_slice_int32 + p.packedDec = (*Buffer).dec_slice_packed_int32 + case reflect.Uint32: + if p.Packed { + p.enc = (*Buffer).enc_slice_packed_uint32 + p.size = size_slice_packed_uint32 + } else { + p.enc = (*Buffer).enc_slice_uint32 + p.size = size_slice_uint32 + } + p.dec = (*Buffer).dec_slice_int32 + p.packedDec = (*Buffer).dec_slice_packed_int32 + case reflect.Int64, reflect.Uint64: + if p.Packed { + p.enc = (*Buffer).enc_slice_packed_int64 + p.size = size_slice_packed_int64 + } else { + p.enc = (*Buffer).enc_slice_int64 + p.size = size_slice_int64 + } + p.dec = (*Buffer).dec_slice_int64 + p.packedDec = (*Buffer).dec_slice_packed_int64 + case reflect.Uint8: + p.enc = (*Buffer).enc_slice_byte + p.dec = (*Buffer).dec_slice_byte + p.size = size_slice_byte + // This is a []byte, which is either a bytes field, + // or the value of a map field. In the latter case, + // we always encode an empty []byte, so we should not + // use the proto3 enc/size funcs. + // f == nil iff this is the key/value of a map field. + if p.proto3 && f != nil { + p.enc = (*Buffer).enc_proto3_slice_byte + p.size = size_proto3_slice_byte + } + case reflect.Float32, reflect.Float64: + switch t2.Bits() { + case 32: + // can just treat them as bits + if p.Packed { + p.enc = (*Buffer).enc_slice_packed_uint32 + p.size = size_slice_packed_uint32 + } else { + p.enc = (*Buffer).enc_slice_uint32 + p.size = size_slice_uint32 + } + p.dec = (*Buffer).dec_slice_int32 + p.packedDec = (*Buffer).dec_slice_packed_int32 + case 64: + // can just treat them as bits + if p.Packed { + p.enc = (*Buffer).enc_slice_packed_int64 + p.size = size_slice_packed_int64 + } else { + p.enc = (*Buffer).enc_slice_int64 + p.size = size_slice_int64 + } + p.dec = (*Buffer).dec_slice_int64 + p.packedDec = (*Buffer).dec_slice_packed_int64 + default: + logNoSliceEnc(t1, t2) + break + } + case reflect.String: + p.enc = (*Buffer).enc_slice_string + p.dec = (*Buffer).dec_slice_string + p.size = size_slice_string + case reflect.Ptr: + switch t3 := t2.Elem(); t3.Kind() { + default: + fmt.Fprintf(os.Stderr, "proto: no ptr oenc for %T -> %T -> %T\n", t1, t2, t3) + break + case reflect.Struct: + p.stype = t2.Elem() + p.isMarshaler = isMarshaler(t2) + p.isUnmarshaler = isUnmarshaler(t2) + if p.Wire == "bytes" { + p.enc = (*Buffer).enc_slice_struct_message + p.dec = (*Buffer).dec_slice_struct_message + p.size = size_slice_struct_message + } else { + p.enc = (*Buffer).enc_slice_struct_group + p.dec = (*Buffer).dec_slice_struct_group + p.size = size_slice_struct_group + } + } + case reflect.Slice: + switch t2.Elem().Kind() { + default: + fmt.Fprintf(os.Stderr, "proto: no slice elem oenc for %T -> %T -> %T\n", t1, t2, t2.Elem()) + break + case reflect.Uint8: + p.enc = (*Buffer).enc_slice_slice_byte + p.dec = (*Buffer).dec_slice_slice_byte + p.size = size_slice_slice_byte + } + } + + case reflect.Map: + p.enc = (*Buffer).enc_new_map + p.dec = (*Buffer).dec_new_map + p.size = size_new_map + + p.mtype = t1 + p.mkeyprop = &Properties{} + p.mkeyprop.init(reflect.PtrTo(p.mtype.Key()), "Key", f.Tag.Get("protobuf_key"), nil, lockGetProp) + p.mvalprop = &Properties{} + vtype := p.mtype.Elem() + if vtype.Kind() != reflect.Ptr && vtype.Kind() != reflect.Slice { + // The value type is not a message (*T) or bytes ([]byte), + // so we need encoders for the pointer to this type. + vtype = reflect.PtrTo(vtype) + } + p.mvalprop.init(vtype, "Value", f.Tag.Get("protobuf_val"), nil, lockGetProp) + } + + // precalculate tag code + wire := p.WireType + if p.Packed { + wire = WireBytes + } + x := uint32(p.Tag)<<3 | uint32(wire) + i := 0 + for i = 0; x > 127; i++ { + p.tagbuf[i] = 0x80 | uint8(x&0x7F) + x >>= 7 + } + p.tagbuf[i] = uint8(x) + p.tagcode = p.tagbuf[0 : i+1] + + if p.stype != nil { + if lockGetProp { + p.sprop = GetProperties(p.stype) + } else { + p.sprop = getPropertiesLocked(p.stype) + } + } +} + +var ( + marshalerType = reflect.TypeOf((*Marshaler)(nil)).Elem() + unmarshalerType = reflect.TypeOf((*Unmarshaler)(nil)).Elem() +) + +// isMarshaler reports whether type t implements Marshaler. +func isMarshaler(t reflect.Type) bool { + // We're checking for (likely) pointer-receiver methods + // so if t is not a pointer, something is very wrong. + // The calls above only invoke isMarshaler on pointer types. + if t.Kind() != reflect.Ptr { + panic("proto: misuse of isMarshaler") + } + return t.Implements(marshalerType) +} + +// isUnmarshaler reports whether type t implements Unmarshaler. +func isUnmarshaler(t reflect.Type) bool { + // We're checking for (likely) pointer-receiver methods + // so if t is not a pointer, something is very wrong. + // The calls above only invoke isUnmarshaler on pointer types. + if t.Kind() != reflect.Ptr { + panic("proto: misuse of isUnmarshaler") + } + return t.Implements(unmarshalerType) +} + +// Init populates the properties from a protocol buffer struct tag. +func (p *Properties) Init(typ reflect.Type, name, tag string, f *reflect.StructField) { + p.init(typ, name, tag, f, true) +} + +func (p *Properties) init(typ reflect.Type, name, tag string, f *reflect.StructField, lockGetProp bool) { + // "bytes,49,opt,def=hello!" + p.Name = name + p.OrigName = name + if f != nil { + p.field = toField(f) + } + if tag == "" { + return + } + p.Parse(tag) + p.setEncAndDec(typ, f, lockGetProp) +} + +var ( + propertiesMu sync.RWMutex + propertiesMap = make(map[reflect.Type]*StructProperties) +) + +// GetProperties returns the list of properties for the type represented by t. +// t must represent a generated struct type of a protocol message. +func GetProperties(t reflect.Type) *StructProperties { + if t.Kind() != reflect.Struct { + panic("proto: type must have kind struct") + } + + // Most calls to GetProperties in a long-running program will be + // retrieving details for types we have seen before. + propertiesMu.RLock() + sprop, ok := propertiesMap[t] + propertiesMu.RUnlock() + if ok { + if collectStats { + stats.Chit++ + } + return sprop + } + + propertiesMu.Lock() + sprop = getPropertiesLocked(t) + propertiesMu.Unlock() + return sprop +} + +// getPropertiesLocked requires that propertiesMu is held. +func getPropertiesLocked(t reflect.Type) *StructProperties { + if prop, ok := propertiesMap[t]; ok { + if collectStats { + stats.Chit++ + } + return prop + } + if collectStats { + stats.Cmiss++ + } + + prop := new(StructProperties) + // in case of recursive protos, fill this in now. + propertiesMap[t] = prop + + // build properties + prop.extendable = reflect.PtrTo(t).Implements(extendableProtoType) + prop.unrecField = invalidField + prop.Prop = make([]*Properties, t.NumField()) + prop.order = make([]int, t.NumField()) + + for i := 0; i < t.NumField(); i++ { + f := t.Field(i) + p := new(Properties) + name := f.Name + p.init(f.Type, name, f.Tag.Get("protobuf"), &f, false) + + if f.Name == "XXX_extensions" { // special case + p.enc = (*Buffer).enc_map + p.dec = nil // not needed + p.size = size_map + } + if f.Name == "XXX_unrecognized" { // special case + prop.unrecField = toField(&f) + } + prop.Prop[i] = p + prop.order[i] = i + if debug { + print(i, " ", f.Name, " ", t.String(), " ") + if p.Tag > 0 { + print(p.String()) + } + print("\n") + } + if p.enc == nil && !strings.HasPrefix(f.Name, "XXX_") { + fmt.Fprintln(os.Stderr, "proto: no encoder for", f.Name, f.Type.String(), "[GetProperties]") + } + } + + // Re-order prop.order. + sort.Sort(prop) + + // build required counts + // build tags + reqCount := 0 + prop.decoderOrigNames = make(map[string]int) + for i, p := range prop.Prop { + if strings.HasPrefix(p.Name, "XXX_") { + // Internal fields should not appear in tags/origNames maps. + // They are handled specially when encoding and decoding. + continue + } + if p.Required { + reqCount++ + } + prop.decoderTags.put(p.Tag, i) + prop.decoderOrigNames[p.OrigName] = i + } + prop.reqCount = reqCount + + return prop +} + +// Return the Properties object for the x[0]'th field of the structure. +func propByIndex(t reflect.Type, x []int) *Properties { + if len(x) != 1 { + fmt.Fprintf(os.Stderr, "proto: field index dimension %d (not 1) for type %s\n", len(x), t) + return nil + } + prop := GetProperties(t) + return prop.Prop[x[0]] +} + +// Get the address and type of a pointer to a struct from an interface. +func getbase(pb Message) (t reflect.Type, b structPointer, err error) { + if pb == nil { + err = ErrNil + return + } + // get the reflect type of the pointer to the struct. + t = reflect.TypeOf(pb) + // get the address of the struct. + value := reflect.ValueOf(pb) + b = toStructPointer(value) + return +} + +// A global registry of enum types. +// The generated code will register the generated maps by calling RegisterEnum. + +var enumValueMaps = make(map[string]map[string]int32) + +// RegisterEnum is called from the generated code to install the enum descriptor +// maps into the global table to aid parsing text format protocol buffers. +func RegisterEnum(typeName string, unusedNameMap map[int32]string, valueMap map[string]int32) { + if _, ok := enumValueMaps[typeName]; ok { + panic("proto: duplicate enum registered: " + typeName) + } + enumValueMaps[typeName] = valueMap +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/proto3_proto/proto3.pb.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/proto3_proto/proto3.pb.go new file mode 100644 index 0000000000000..37c77820921d3 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/proto3_proto/proto3.pb.go @@ -0,0 +1,122 @@ +// Code generated by protoc-gen-go. +// source: proto3_proto/proto3.proto +// DO NOT EDIT! + +/* +Package proto3_proto is a generated protocol buffer package. + +It is generated from these files: + proto3_proto/proto3.proto + +It has these top-level messages: + Message + Nested + MessageWithMap +*/ +package proto3_proto + +import proto "github.com/golang/protobuf/proto" +import testdata "github.com/golang/protobuf/proto/testdata" + +// Reference imports to suppress errors if they are not otherwise used. +var _ = proto.Marshal + +type Message_Humour int32 + +const ( + Message_UNKNOWN Message_Humour = 0 + Message_PUNS Message_Humour = 1 + Message_SLAPSTICK Message_Humour = 2 + Message_BILL_BAILEY Message_Humour = 3 +) + +var Message_Humour_name = map[int32]string{ + 0: "UNKNOWN", + 1: "PUNS", + 2: "SLAPSTICK", + 3: "BILL_BAILEY", +} +var Message_Humour_value = map[string]int32{ + "UNKNOWN": 0, + "PUNS": 1, + "SLAPSTICK": 2, + "BILL_BAILEY": 3, +} + +func (x Message_Humour) String() string { + return proto.EnumName(Message_Humour_name, int32(x)) +} + +type Message struct { + Name string `protobuf:"bytes,1,opt,name=name" json:"name,omitempty"` + Hilarity Message_Humour `protobuf:"varint,2,opt,name=hilarity,enum=proto3_proto.Message_Humour" json:"hilarity,omitempty"` + HeightInCm uint32 `protobuf:"varint,3,opt,name=height_in_cm" json:"height_in_cm,omitempty"` + Data []byte `protobuf:"bytes,4,opt,name=data,proto3" json:"data,omitempty"` + ResultCount int64 `protobuf:"varint,7,opt,name=result_count" json:"result_count,omitempty"` + TrueScotsman bool `protobuf:"varint,8,opt,name=true_scotsman" json:"true_scotsman,omitempty"` + Score float32 `protobuf:"fixed32,9,opt,name=score" json:"score,omitempty"` + Key []uint64 `protobuf:"varint,5,rep,name=key" json:"key,omitempty"` + Nested *Nested `protobuf:"bytes,6,opt,name=nested" json:"nested,omitempty"` + Terrain map[string]*Nested `protobuf:"bytes,10,rep,name=terrain" json:"terrain,omitempty" protobuf_key:"bytes,1,opt,name=key" protobuf_val:"bytes,2,opt,name=value"` + Proto2Field *testdata.SubDefaults `protobuf:"bytes,11,opt,name=proto2_field" json:"proto2_field,omitempty"` + Proto2Value map[string]*testdata.SubDefaults `protobuf:"bytes,13,rep,name=proto2_value" json:"proto2_value,omitempty" protobuf_key:"bytes,1,opt,name=key" protobuf_val:"bytes,2,opt,name=value"` +} + +func (m *Message) Reset() { *m = Message{} } +func (m *Message) String() string { return proto.CompactTextString(m) } +func (*Message) ProtoMessage() {} + +func (m *Message) GetNested() *Nested { + if m != nil { + return m.Nested + } + return nil +} + +func (m *Message) GetTerrain() map[string]*Nested { + if m != nil { + return m.Terrain + } + return nil +} + +func (m *Message) GetProto2Field() *testdata.SubDefaults { + if m != nil { + return m.Proto2Field + } + return nil +} + +func (m *Message) GetProto2Value() map[string]*testdata.SubDefaults { + if m != nil { + return m.Proto2Value + } + return nil +} + +type Nested struct { + Bunny string `protobuf:"bytes,1,opt,name=bunny" json:"bunny,omitempty"` +} + +func (m *Nested) Reset() { *m = Nested{} } +func (m *Nested) String() string { return proto.CompactTextString(m) } +func (*Nested) ProtoMessage() {} + +type MessageWithMap struct { + ByteMapping map[bool][]byte `protobuf:"bytes,1,rep,name=byte_mapping" json:"byte_mapping,omitempty" protobuf_key:"varint,1,opt,name=key" protobuf_val:"bytes,2,opt,name=value,proto3"` +} + +func (m *MessageWithMap) Reset() { *m = MessageWithMap{} } +func (m *MessageWithMap) String() string { return proto.CompactTextString(m) } +func (*MessageWithMap) ProtoMessage() {} + +func (m *MessageWithMap) GetByteMapping() map[bool][]byte { + if m != nil { + return m.ByteMapping + } + return nil +} + +func init() { + proto.RegisterEnum("proto3_proto.Message_Humour", Message_Humour_name, Message_Humour_value) +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/proto3_proto/proto3.proto b/Godeps/_workspace/src/github.com/golang/protobuf/proto/proto3_proto/proto3.proto new file mode 100644 index 0000000000000..e2311d9294df6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/proto3_proto/proto3.proto @@ -0,0 +1,68 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2014 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +syntax = "proto3"; + +import "testdata/test.proto"; + +package proto3_proto; + +message Message { + enum Humour { + UNKNOWN = 0; + PUNS = 1; + SLAPSTICK = 2; + BILL_BAILEY = 3; + } + + string name = 1; + Humour hilarity = 2; + uint32 height_in_cm = 3; + bytes data = 4; + int64 result_count = 7; + bool true_scotsman = 8; + float score = 9; + + repeated uint64 key = 5; + Nested nested = 6; + + map terrain = 10; + testdata.SubDefaults proto2_field = 11; + map proto2_value = 13; +} + +message Nested { + string bunny = 1; +} + +message MessageWithMap { + map byte_mapping = 1; +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/proto3_test.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/proto3_test.go new file mode 100644 index 0000000000000..462f8055c3093 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/proto3_test.go @@ -0,0 +1,125 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2014 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "testing" + + "github.com/golang/protobuf/proto" + pb "github.com/golang/protobuf/proto/proto3_proto" + tpb "github.com/golang/protobuf/proto/testdata" +) + +func TestProto3ZeroValues(t *testing.T) { + tests := []struct { + desc string + m proto.Message + }{ + {"zero message", &pb.Message{}}, + {"empty bytes field", &pb.Message{Data: []byte{}}}, + } + for _, test := range tests { + b, err := proto.Marshal(test.m) + if err != nil { + t.Errorf("%s: proto.Marshal: %v", test.desc, err) + continue + } + if len(b) > 0 { + t.Errorf("%s: Encoding is non-empty: %q", test.desc, b) + } + } +} + +func TestRoundTripProto3(t *testing.T) { + m := &pb.Message{ + Name: "David", // (2 | 1<<3): 0x0a 0x05 "David" + Hilarity: pb.Message_PUNS, // (0 | 2<<3): 0x10 0x01 + HeightInCm: 178, // (0 | 3<<3): 0x18 0xb2 0x01 + Data: []byte("roboto"), // (2 | 4<<3): 0x20 0x06 "roboto" + ResultCount: 47, // (0 | 7<<3): 0x38 0x2f + TrueScotsman: true, // (0 | 8<<3): 0x40 0x01 + Score: 8.1, // (5 | 9<<3): 0x4d <8.1> + + Key: []uint64{1, 0xdeadbeef}, + Nested: &pb.Nested{ + Bunny: "Monty", + }, + } + t.Logf(" m: %v", m) + + b, err := proto.Marshal(m) + if err != nil { + t.Fatalf("proto.Marshal: %v", err) + } + t.Logf(" b: %q", b) + + m2 := new(pb.Message) + if err := proto.Unmarshal(b, m2); err != nil { + t.Fatalf("proto.Unmarshal: %v", err) + } + t.Logf("m2: %v", m2) + + if !proto.Equal(m, m2) { + t.Errorf("proto.Equal returned false:\n m: %v\nm2: %v", m, m2) + } +} + +func TestProto3SetDefaults(t *testing.T) { + in := &pb.Message{ + Terrain: map[string]*pb.Nested{ + "meadow": new(pb.Nested), + }, + Proto2Field: new(tpb.SubDefaults), + Proto2Value: map[string]*tpb.SubDefaults{ + "badlands": new(tpb.SubDefaults), + }, + } + + got := proto.Clone(in).(*pb.Message) + proto.SetDefaults(got) + + // There are no defaults in proto3. Everything should be the zero value, but + // we need to remember to set defaults for nested proto2 messages. + want := &pb.Message{ + Terrain: map[string]*pb.Nested{ + "meadow": new(pb.Nested), + }, + Proto2Field: &tpb.SubDefaults{N: proto.Int64(7)}, + Proto2Value: map[string]*tpb.SubDefaults{ + "badlands": &tpb.SubDefaults{N: proto.Int64(7)}, + }, + } + + if !proto.Equal(got, want) { + t.Errorf("with in = %v\nproto.SetDefaults(in) =>\ngot %v\nwant %v", in, got, want) + } +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/size2_test.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/size2_test.go new file mode 100644 index 0000000000000..a2729c39a1b83 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/size2_test.go @@ -0,0 +1,63 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2012 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +import ( + "testing" +) + +// This is a separate file and package from size_test.go because that one uses +// generated messages and thus may not be in package proto without having a circular +// dependency, whereas this file tests unexported details of size.go. + +func TestVarintSize(t *testing.T) { + // Check the edge cases carefully. + testCases := []struct { + n uint64 + size int + }{ + {0, 1}, + {1, 1}, + {127, 1}, + {128, 2}, + {16383, 2}, + {16384, 3}, + {1<<63 - 1, 9}, + {1 << 63, 10}, + } + for _, tc := range testCases { + size := sizeVarint(tc.n) + if size != tc.size { + t.Errorf("sizeVarint(%d) = %d, want %d", tc.n, size, tc.size) + } + } +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/size_test.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/size_test.go new file mode 100644 index 0000000000000..db5614fd1d011 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/size_test.go @@ -0,0 +1,142 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2012 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "log" + "strings" + "testing" + + . "github.com/golang/protobuf/proto" + proto3pb "github.com/golang/protobuf/proto/proto3_proto" + pb "github.com/golang/protobuf/proto/testdata" +) + +var messageWithExtension1 = &pb.MyMessage{Count: Int32(7)} + +// messageWithExtension2 is in equal_test.go. +var messageWithExtension3 = &pb.MyMessage{Count: Int32(8)} + +func init() { + if err := SetExtension(messageWithExtension1, pb.E_Ext_More, &pb.Ext{Data: String("Abbott")}); err != nil { + log.Panicf("SetExtension: %v", err) + } + if err := SetExtension(messageWithExtension3, pb.E_Ext_More, &pb.Ext{Data: String("Costello")}); err != nil { + log.Panicf("SetExtension: %v", err) + } + + // Force messageWithExtension3 to have the extension encoded. + Marshal(messageWithExtension3) + +} + +var SizeTests = []struct { + desc string + pb Message +}{ + {"empty", &pb.OtherMessage{}}, + // Basic types. + {"bool", &pb.Defaults{F_Bool: Bool(true)}}, + {"int32", &pb.Defaults{F_Int32: Int32(12)}}, + {"negative int32", &pb.Defaults{F_Int32: Int32(-1)}}, + {"small int64", &pb.Defaults{F_Int64: Int64(1)}}, + {"big int64", &pb.Defaults{F_Int64: Int64(1 << 20)}}, + {"negative int64", &pb.Defaults{F_Int64: Int64(-1)}}, + {"fixed32", &pb.Defaults{F_Fixed32: Uint32(71)}}, + {"fixed64", &pb.Defaults{F_Fixed64: Uint64(72)}}, + {"uint32", &pb.Defaults{F_Uint32: Uint32(123)}}, + {"uint64", &pb.Defaults{F_Uint64: Uint64(124)}}, + {"float", &pb.Defaults{F_Float: Float32(12.6)}}, + {"double", &pb.Defaults{F_Double: Float64(13.9)}}, + {"string", &pb.Defaults{F_String: String("niles")}}, + {"bytes", &pb.Defaults{F_Bytes: []byte("wowsa")}}, + {"bytes, empty", &pb.Defaults{F_Bytes: []byte{}}}, + {"sint32", &pb.Defaults{F_Sint32: Int32(65)}}, + {"sint64", &pb.Defaults{F_Sint64: Int64(67)}}, + {"enum", &pb.Defaults{F_Enum: pb.Defaults_BLUE.Enum()}}, + // Repeated. + {"empty repeated bool", &pb.MoreRepeated{Bools: []bool{}}}, + {"repeated bool", &pb.MoreRepeated{Bools: []bool{false, true, true, false}}}, + {"packed repeated bool", &pb.MoreRepeated{BoolsPacked: []bool{false, true, true, false, true, true, true}}}, + {"repeated int32", &pb.MoreRepeated{Ints: []int32{1, 12203, 1729, -1}}}, + {"repeated int32 packed", &pb.MoreRepeated{IntsPacked: []int32{1, 12203, 1729}}}, + {"repeated int64 packed", &pb.MoreRepeated{Int64SPacked: []int64{ + // Need enough large numbers to verify that the header is counting the number of bytes + // for the field, not the number of elements. + 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, + 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, 1 << 62, + }}}, + {"repeated string", &pb.MoreRepeated{Strings: []string{"r", "ken", "gri"}}}, + {"repeated fixed", &pb.MoreRepeated{Fixeds: []uint32{1, 2, 3, 4}}}, + // Nested. + {"nested", &pb.OldMessage{Nested: &pb.OldMessage_Nested{Name: String("whatever")}}}, + {"group", &pb.GroupOld{G: &pb.GroupOld_G{X: Int32(12345)}}}, + // Other things. + {"unrecognized", &pb.MoreRepeated{XXX_unrecognized: []byte{13<<3 | 0, 4}}}, + {"extension (unencoded)", messageWithExtension1}, + {"extension (encoded)", messageWithExtension3}, + // proto3 message + {"proto3 empty", &proto3pb.Message{}}, + {"proto3 bool", &proto3pb.Message{TrueScotsman: true}}, + {"proto3 int64", &proto3pb.Message{ResultCount: 1}}, + {"proto3 uint32", &proto3pb.Message{HeightInCm: 123}}, + {"proto3 float", &proto3pb.Message{Score: 12.6}}, + {"proto3 string", &proto3pb.Message{Name: "Snezana"}}, + {"proto3 bytes", &proto3pb.Message{Data: []byte("wowsa")}}, + {"proto3 bytes, empty", &proto3pb.Message{Data: []byte{}}}, + {"proto3 enum", &proto3pb.Message{Hilarity: proto3pb.Message_PUNS}}, + {"proto3 map field with empty bytes", &proto3pb.MessageWithMap{ByteMapping: map[bool][]byte{false: []byte{}}}}, + + {"map field", &pb.MessageWithMap{NameMapping: map[int32]string{1: "Rob", 7: "Andrew"}}}, + {"map field with message", &pb.MessageWithMap{MsgMapping: map[int64]*pb.FloatingPoint{0x7001: &pb.FloatingPoint{F: Float64(2.0)}}}}, + {"map field with bytes", &pb.MessageWithMap{ByteMapping: map[bool][]byte{true: []byte("this time for sure")}}}, + {"map field with empty bytes", &pb.MessageWithMap{ByteMapping: map[bool][]byte{true: []byte{}}}}, + + {"map field with big entry", &pb.MessageWithMap{NameMapping: map[int32]string{8: strings.Repeat("x", 125)}}}, + {"map field with big key and val", &pb.MessageWithMap{StrToStr: map[string]string{strings.Repeat("x", 70): strings.Repeat("y", 70)}}}, + {"map field with big numeric key", &pb.MessageWithMap{NameMapping: map[int32]string{0xf00d: "om nom nom"}}}, +} + +func TestSize(t *testing.T) { + for _, tc := range SizeTests { + size := Size(tc.pb) + b, err := Marshal(tc.pb) + if err != nil { + t.Errorf("%v: Marshal failed: %v", tc.desc, err) + continue + } + if size != len(b) { + t.Errorf("%v: Size(%v) = %d, want %d", tc.desc, tc.pb, size, len(b)) + t.Logf("%v: bytes: %#v", tc.desc, b) + } + } +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/testdata/Makefile b/Godeps/_workspace/src/github.com/golang/protobuf/proto/testdata/Makefile new file mode 100644 index 0000000000000..fc288628a7528 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/testdata/Makefile @@ -0,0 +1,50 @@ +# Go support for Protocol Buffers - Google's data interchange format +# +# Copyright 2010 The Go Authors. All rights reserved. +# https://github.com/golang/protobuf +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are +# met: +# +# * Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +# * Redistributions in binary form must reproduce the above +# copyright notice, this list of conditions and the following disclaimer +# in the documentation and/or other materials provided with the +# distribution. +# * Neither the name of Google Inc. nor the names of its +# contributors may be used to endorse or promote products derived from +# this software without specific prior written permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + + +include ../../Make.protobuf + +all: regenerate + +regenerate: + rm -f test.pb.go + make test.pb.go + +# The following rules are just aids to development. Not needed for typical testing. + +diff: regenerate + git diff test.pb.go + +restore: + cp test.pb.go.golden test.pb.go + +preserve: + cp test.pb.go test.pb.go.golden diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/testdata/golden_test.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/testdata/golden_test.go new file mode 100644 index 0000000000000..7172d0e96985d --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/testdata/golden_test.go @@ -0,0 +1,86 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2012 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// Verify that the compiler output for test.proto is unchanged. + +package testdata + +import ( + "crypto/sha1" + "fmt" + "io/ioutil" + "os" + "os/exec" + "path/filepath" + "testing" +) + +// sum returns in string form (for easy comparison) the SHA-1 hash of the named file. +func sum(t *testing.T, name string) string { + data, err := ioutil.ReadFile(name) + if err != nil { + t.Fatal(err) + } + t.Logf("sum(%q): length is %d", name, len(data)) + hash := sha1.New() + _, err = hash.Write(data) + if err != nil { + t.Fatal(err) + } + return fmt.Sprintf("% x", hash.Sum(nil)) +} + +func run(t *testing.T, name string, args ...string) { + cmd := exec.Command(name, args...) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + err := cmd.Run() + if err != nil { + t.Fatal(err) + } +} + +func TestGolden(t *testing.T) { + // Compute the original checksum. + goldenSum := sum(t, "test.pb.go") + // Run the proto compiler. + run(t, "protoc", "--go_out="+os.TempDir(), "test.proto") + newFile := filepath.Join(os.TempDir(), "test.pb.go") + defer os.Remove(newFile) + // Compute the new checksum. + newSum := sum(t, newFile) + // Verify + if newSum != goldenSum { + run(t, "diff", "-u", "test.pb.go", newFile) + t.Fatal("Code generated by protoc-gen-go has changed; update test.pb.go") + } +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/testdata/test.pb.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/testdata/test.pb.go new file mode 100644 index 0000000000000..13674a4491900 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/testdata/test.pb.go @@ -0,0 +1,2746 @@ +// Code generated by protoc-gen-go. +// source: test.proto +// DO NOT EDIT! + +/* +Package testdata is a generated protocol buffer package. + +It is generated from these files: + test.proto + +It has these top-level messages: + GoEnum + GoTestField + GoTest + GoSkipTest + NonPackedTest + PackedTest + MaxTag + OldMessage + NewMessage + InnerMessage + OtherMessage + MyMessage + Ext + DefaultsMessage + MyMessageSet + Empty + MessageList + Strings + Defaults + SubDefaults + RepeatedEnum + MoreRepeated + GroupOld + GroupNew + FloatingPoint + MessageWithMap +*/ +package testdata + +import proto "github.com/golang/protobuf/proto" +import math "math" + +// Reference imports to suppress errors if they are not otherwise used. +var _ = proto.Marshal +var _ = math.Inf + +type FOO int32 + +const ( + FOO_FOO1 FOO = 1 +) + +var FOO_name = map[int32]string{ + 1: "FOO1", +} +var FOO_value = map[string]int32{ + "FOO1": 1, +} + +func (x FOO) Enum() *FOO { + p := new(FOO) + *p = x + return p +} +func (x FOO) String() string { + return proto.EnumName(FOO_name, int32(x)) +} +func (x *FOO) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(FOO_value, data, "FOO") + if err != nil { + return err + } + *x = FOO(value) + return nil +} + +// An enum, for completeness. +type GoTest_KIND int32 + +const ( + GoTest_VOID GoTest_KIND = 0 + // Basic types + GoTest_BOOL GoTest_KIND = 1 + GoTest_BYTES GoTest_KIND = 2 + GoTest_FINGERPRINT GoTest_KIND = 3 + GoTest_FLOAT GoTest_KIND = 4 + GoTest_INT GoTest_KIND = 5 + GoTest_STRING GoTest_KIND = 6 + GoTest_TIME GoTest_KIND = 7 + // Groupings + GoTest_TUPLE GoTest_KIND = 8 + GoTest_ARRAY GoTest_KIND = 9 + GoTest_MAP GoTest_KIND = 10 + // Table types + GoTest_TABLE GoTest_KIND = 11 + // Functions + GoTest_FUNCTION GoTest_KIND = 12 +) + +var GoTest_KIND_name = map[int32]string{ + 0: "VOID", + 1: "BOOL", + 2: "BYTES", + 3: "FINGERPRINT", + 4: "FLOAT", + 5: "INT", + 6: "STRING", + 7: "TIME", + 8: "TUPLE", + 9: "ARRAY", + 10: "MAP", + 11: "TABLE", + 12: "FUNCTION", +} +var GoTest_KIND_value = map[string]int32{ + "VOID": 0, + "BOOL": 1, + "BYTES": 2, + "FINGERPRINT": 3, + "FLOAT": 4, + "INT": 5, + "STRING": 6, + "TIME": 7, + "TUPLE": 8, + "ARRAY": 9, + "MAP": 10, + "TABLE": 11, + "FUNCTION": 12, +} + +func (x GoTest_KIND) Enum() *GoTest_KIND { + p := new(GoTest_KIND) + *p = x + return p +} +func (x GoTest_KIND) String() string { + return proto.EnumName(GoTest_KIND_name, int32(x)) +} +func (x *GoTest_KIND) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(GoTest_KIND_value, data, "GoTest_KIND") + if err != nil { + return err + } + *x = GoTest_KIND(value) + return nil +} + +type MyMessage_Color int32 + +const ( + MyMessage_RED MyMessage_Color = 0 + MyMessage_GREEN MyMessage_Color = 1 + MyMessage_BLUE MyMessage_Color = 2 +) + +var MyMessage_Color_name = map[int32]string{ + 0: "RED", + 1: "GREEN", + 2: "BLUE", +} +var MyMessage_Color_value = map[string]int32{ + "RED": 0, + "GREEN": 1, + "BLUE": 2, +} + +func (x MyMessage_Color) Enum() *MyMessage_Color { + p := new(MyMessage_Color) + *p = x + return p +} +func (x MyMessage_Color) String() string { + return proto.EnumName(MyMessage_Color_name, int32(x)) +} +func (x *MyMessage_Color) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(MyMessage_Color_value, data, "MyMessage_Color") + if err != nil { + return err + } + *x = MyMessage_Color(value) + return nil +} + +type DefaultsMessage_DefaultsEnum int32 + +const ( + DefaultsMessage_ZERO DefaultsMessage_DefaultsEnum = 0 + DefaultsMessage_ONE DefaultsMessage_DefaultsEnum = 1 + DefaultsMessage_TWO DefaultsMessage_DefaultsEnum = 2 +) + +var DefaultsMessage_DefaultsEnum_name = map[int32]string{ + 0: "ZERO", + 1: "ONE", + 2: "TWO", +} +var DefaultsMessage_DefaultsEnum_value = map[string]int32{ + "ZERO": 0, + "ONE": 1, + "TWO": 2, +} + +func (x DefaultsMessage_DefaultsEnum) Enum() *DefaultsMessage_DefaultsEnum { + p := new(DefaultsMessage_DefaultsEnum) + *p = x + return p +} +func (x DefaultsMessage_DefaultsEnum) String() string { + return proto.EnumName(DefaultsMessage_DefaultsEnum_name, int32(x)) +} +func (x *DefaultsMessage_DefaultsEnum) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(DefaultsMessage_DefaultsEnum_value, data, "DefaultsMessage_DefaultsEnum") + if err != nil { + return err + } + *x = DefaultsMessage_DefaultsEnum(value) + return nil +} + +type Defaults_Color int32 + +const ( + Defaults_RED Defaults_Color = 0 + Defaults_GREEN Defaults_Color = 1 + Defaults_BLUE Defaults_Color = 2 +) + +var Defaults_Color_name = map[int32]string{ + 0: "RED", + 1: "GREEN", + 2: "BLUE", +} +var Defaults_Color_value = map[string]int32{ + "RED": 0, + "GREEN": 1, + "BLUE": 2, +} + +func (x Defaults_Color) Enum() *Defaults_Color { + p := new(Defaults_Color) + *p = x + return p +} +func (x Defaults_Color) String() string { + return proto.EnumName(Defaults_Color_name, int32(x)) +} +func (x *Defaults_Color) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(Defaults_Color_value, data, "Defaults_Color") + if err != nil { + return err + } + *x = Defaults_Color(value) + return nil +} + +type RepeatedEnum_Color int32 + +const ( + RepeatedEnum_RED RepeatedEnum_Color = 1 +) + +var RepeatedEnum_Color_name = map[int32]string{ + 1: "RED", +} +var RepeatedEnum_Color_value = map[string]int32{ + "RED": 1, +} + +func (x RepeatedEnum_Color) Enum() *RepeatedEnum_Color { + p := new(RepeatedEnum_Color) + *p = x + return p +} +func (x RepeatedEnum_Color) String() string { + return proto.EnumName(RepeatedEnum_Color_name, int32(x)) +} +func (x *RepeatedEnum_Color) UnmarshalJSON(data []byte) error { + value, err := proto.UnmarshalJSONEnum(RepeatedEnum_Color_value, data, "RepeatedEnum_Color") + if err != nil { + return err + } + *x = RepeatedEnum_Color(value) + return nil +} + +type GoEnum struct { + Foo *FOO `protobuf:"varint,1,req,name=foo,enum=testdata.FOO" json:"foo,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoEnum) Reset() { *m = GoEnum{} } +func (m *GoEnum) String() string { return proto.CompactTextString(m) } +func (*GoEnum) ProtoMessage() {} + +func (m *GoEnum) GetFoo() FOO { + if m != nil && m.Foo != nil { + return *m.Foo + } + return FOO_FOO1 +} + +type GoTestField struct { + Label *string `protobuf:"bytes,1,req" json:"Label,omitempty"` + Type *string `protobuf:"bytes,2,req" json:"Type,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTestField) Reset() { *m = GoTestField{} } +func (m *GoTestField) String() string { return proto.CompactTextString(m) } +func (*GoTestField) ProtoMessage() {} + +func (m *GoTestField) GetLabel() string { + if m != nil && m.Label != nil { + return *m.Label + } + return "" +} + +func (m *GoTestField) GetType() string { + if m != nil && m.Type != nil { + return *m.Type + } + return "" +} + +type GoTest struct { + // Some typical parameters + Kind *GoTest_KIND `protobuf:"varint,1,req,enum=testdata.GoTest_KIND" json:"Kind,omitempty"` + Table *string `protobuf:"bytes,2,opt" json:"Table,omitempty"` + Param *int32 `protobuf:"varint,3,opt" json:"Param,omitempty"` + // Required, repeated and optional foreign fields. + RequiredField *GoTestField `protobuf:"bytes,4,req" json:"RequiredField,omitempty"` + RepeatedField []*GoTestField `protobuf:"bytes,5,rep" json:"RepeatedField,omitempty"` + OptionalField *GoTestField `protobuf:"bytes,6,opt" json:"OptionalField,omitempty"` + // Required fields of all basic types + F_BoolRequired *bool `protobuf:"varint,10,req,name=F_Bool_required" json:"F_Bool_required,omitempty"` + F_Int32Required *int32 `protobuf:"varint,11,req,name=F_Int32_required" json:"F_Int32_required,omitempty"` + F_Int64Required *int64 `protobuf:"varint,12,req,name=F_Int64_required" json:"F_Int64_required,omitempty"` + F_Fixed32Required *uint32 `protobuf:"fixed32,13,req,name=F_Fixed32_required" json:"F_Fixed32_required,omitempty"` + F_Fixed64Required *uint64 `protobuf:"fixed64,14,req,name=F_Fixed64_required" json:"F_Fixed64_required,omitempty"` + F_Uint32Required *uint32 `protobuf:"varint,15,req,name=F_Uint32_required" json:"F_Uint32_required,omitempty"` + F_Uint64Required *uint64 `protobuf:"varint,16,req,name=F_Uint64_required" json:"F_Uint64_required,omitempty"` + F_FloatRequired *float32 `protobuf:"fixed32,17,req,name=F_Float_required" json:"F_Float_required,omitempty"` + F_DoubleRequired *float64 `protobuf:"fixed64,18,req,name=F_Double_required" json:"F_Double_required,omitempty"` + F_StringRequired *string `protobuf:"bytes,19,req,name=F_String_required" json:"F_String_required,omitempty"` + F_BytesRequired []byte `protobuf:"bytes,101,req,name=F_Bytes_required" json:"F_Bytes_required,omitempty"` + F_Sint32Required *int32 `protobuf:"zigzag32,102,req,name=F_Sint32_required" json:"F_Sint32_required,omitempty"` + F_Sint64Required *int64 `protobuf:"zigzag64,103,req,name=F_Sint64_required" json:"F_Sint64_required,omitempty"` + // Repeated fields of all basic types + F_BoolRepeated []bool `protobuf:"varint,20,rep,name=F_Bool_repeated" json:"F_Bool_repeated,omitempty"` + F_Int32Repeated []int32 `protobuf:"varint,21,rep,name=F_Int32_repeated" json:"F_Int32_repeated,omitempty"` + F_Int64Repeated []int64 `protobuf:"varint,22,rep,name=F_Int64_repeated" json:"F_Int64_repeated,omitempty"` + F_Fixed32Repeated []uint32 `protobuf:"fixed32,23,rep,name=F_Fixed32_repeated" json:"F_Fixed32_repeated,omitempty"` + F_Fixed64Repeated []uint64 `protobuf:"fixed64,24,rep,name=F_Fixed64_repeated" json:"F_Fixed64_repeated,omitempty"` + F_Uint32Repeated []uint32 `protobuf:"varint,25,rep,name=F_Uint32_repeated" json:"F_Uint32_repeated,omitempty"` + F_Uint64Repeated []uint64 `protobuf:"varint,26,rep,name=F_Uint64_repeated" json:"F_Uint64_repeated,omitempty"` + F_FloatRepeated []float32 `protobuf:"fixed32,27,rep,name=F_Float_repeated" json:"F_Float_repeated,omitempty"` + F_DoubleRepeated []float64 `protobuf:"fixed64,28,rep,name=F_Double_repeated" json:"F_Double_repeated,omitempty"` + F_StringRepeated []string `protobuf:"bytes,29,rep,name=F_String_repeated" json:"F_String_repeated,omitempty"` + F_BytesRepeated [][]byte `protobuf:"bytes,201,rep,name=F_Bytes_repeated" json:"F_Bytes_repeated,omitempty"` + F_Sint32Repeated []int32 `protobuf:"zigzag32,202,rep,name=F_Sint32_repeated" json:"F_Sint32_repeated,omitempty"` + F_Sint64Repeated []int64 `protobuf:"zigzag64,203,rep,name=F_Sint64_repeated" json:"F_Sint64_repeated,omitempty"` + // Optional fields of all basic types + F_BoolOptional *bool `protobuf:"varint,30,opt,name=F_Bool_optional" json:"F_Bool_optional,omitempty"` + F_Int32Optional *int32 `protobuf:"varint,31,opt,name=F_Int32_optional" json:"F_Int32_optional,omitempty"` + F_Int64Optional *int64 `protobuf:"varint,32,opt,name=F_Int64_optional" json:"F_Int64_optional,omitempty"` + F_Fixed32Optional *uint32 `protobuf:"fixed32,33,opt,name=F_Fixed32_optional" json:"F_Fixed32_optional,omitempty"` + F_Fixed64Optional *uint64 `protobuf:"fixed64,34,opt,name=F_Fixed64_optional" json:"F_Fixed64_optional,omitempty"` + F_Uint32Optional *uint32 `protobuf:"varint,35,opt,name=F_Uint32_optional" json:"F_Uint32_optional,omitempty"` + F_Uint64Optional *uint64 `protobuf:"varint,36,opt,name=F_Uint64_optional" json:"F_Uint64_optional,omitempty"` + F_FloatOptional *float32 `protobuf:"fixed32,37,opt,name=F_Float_optional" json:"F_Float_optional,omitempty"` + F_DoubleOptional *float64 `protobuf:"fixed64,38,opt,name=F_Double_optional" json:"F_Double_optional,omitempty"` + F_StringOptional *string `protobuf:"bytes,39,opt,name=F_String_optional" json:"F_String_optional,omitempty"` + F_BytesOptional []byte `protobuf:"bytes,301,opt,name=F_Bytes_optional" json:"F_Bytes_optional,omitempty"` + F_Sint32Optional *int32 `protobuf:"zigzag32,302,opt,name=F_Sint32_optional" json:"F_Sint32_optional,omitempty"` + F_Sint64Optional *int64 `protobuf:"zigzag64,303,opt,name=F_Sint64_optional" json:"F_Sint64_optional,omitempty"` + // Default-valued fields of all basic types + F_BoolDefaulted *bool `protobuf:"varint,40,opt,name=F_Bool_defaulted,def=1" json:"F_Bool_defaulted,omitempty"` + F_Int32Defaulted *int32 `protobuf:"varint,41,opt,name=F_Int32_defaulted,def=32" json:"F_Int32_defaulted,omitempty"` + F_Int64Defaulted *int64 `protobuf:"varint,42,opt,name=F_Int64_defaulted,def=64" json:"F_Int64_defaulted,omitempty"` + F_Fixed32Defaulted *uint32 `protobuf:"fixed32,43,opt,name=F_Fixed32_defaulted,def=320" json:"F_Fixed32_defaulted,omitempty"` + F_Fixed64Defaulted *uint64 `protobuf:"fixed64,44,opt,name=F_Fixed64_defaulted,def=640" json:"F_Fixed64_defaulted,omitempty"` + F_Uint32Defaulted *uint32 `protobuf:"varint,45,opt,name=F_Uint32_defaulted,def=3200" json:"F_Uint32_defaulted,omitempty"` + F_Uint64Defaulted *uint64 `protobuf:"varint,46,opt,name=F_Uint64_defaulted,def=6400" json:"F_Uint64_defaulted,omitempty"` + F_FloatDefaulted *float32 `protobuf:"fixed32,47,opt,name=F_Float_defaulted,def=314159" json:"F_Float_defaulted,omitempty"` + F_DoubleDefaulted *float64 `protobuf:"fixed64,48,opt,name=F_Double_defaulted,def=271828" json:"F_Double_defaulted,omitempty"` + F_StringDefaulted *string `protobuf:"bytes,49,opt,name=F_String_defaulted,def=hello, \"world!\"\n" json:"F_String_defaulted,omitempty"` + F_BytesDefaulted []byte `protobuf:"bytes,401,opt,name=F_Bytes_defaulted,def=Bignose" json:"F_Bytes_defaulted,omitempty"` + F_Sint32Defaulted *int32 `protobuf:"zigzag32,402,opt,name=F_Sint32_defaulted,def=-32" json:"F_Sint32_defaulted,omitempty"` + F_Sint64Defaulted *int64 `protobuf:"zigzag64,403,opt,name=F_Sint64_defaulted,def=-64" json:"F_Sint64_defaulted,omitempty"` + // Packed repeated fields (no string or bytes). + F_BoolRepeatedPacked []bool `protobuf:"varint,50,rep,packed,name=F_Bool_repeated_packed" json:"F_Bool_repeated_packed,omitempty"` + F_Int32RepeatedPacked []int32 `protobuf:"varint,51,rep,packed,name=F_Int32_repeated_packed" json:"F_Int32_repeated_packed,omitempty"` + F_Int64RepeatedPacked []int64 `protobuf:"varint,52,rep,packed,name=F_Int64_repeated_packed" json:"F_Int64_repeated_packed,omitempty"` + F_Fixed32RepeatedPacked []uint32 `protobuf:"fixed32,53,rep,packed,name=F_Fixed32_repeated_packed" json:"F_Fixed32_repeated_packed,omitempty"` + F_Fixed64RepeatedPacked []uint64 `protobuf:"fixed64,54,rep,packed,name=F_Fixed64_repeated_packed" json:"F_Fixed64_repeated_packed,omitempty"` + F_Uint32RepeatedPacked []uint32 `protobuf:"varint,55,rep,packed,name=F_Uint32_repeated_packed" json:"F_Uint32_repeated_packed,omitempty"` + F_Uint64RepeatedPacked []uint64 `protobuf:"varint,56,rep,packed,name=F_Uint64_repeated_packed" json:"F_Uint64_repeated_packed,omitempty"` + F_FloatRepeatedPacked []float32 `protobuf:"fixed32,57,rep,packed,name=F_Float_repeated_packed" json:"F_Float_repeated_packed,omitempty"` + F_DoubleRepeatedPacked []float64 `protobuf:"fixed64,58,rep,packed,name=F_Double_repeated_packed" json:"F_Double_repeated_packed,omitempty"` + F_Sint32RepeatedPacked []int32 `protobuf:"zigzag32,502,rep,packed,name=F_Sint32_repeated_packed" json:"F_Sint32_repeated_packed,omitempty"` + F_Sint64RepeatedPacked []int64 `protobuf:"zigzag64,503,rep,packed,name=F_Sint64_repeated_packed" json:"F_Sint64_repeated_packed,omitempty"` + Requiredgroup *GoTest_RequiredGroup `protobuf:"group,70,req,name=RequiredGroup" json:"requiredgroup,omitempty"` + Repeatedgroup []*GoTest_RepeatedGroup `protobuf:"group,80,rep,name=RepeatedGroup" json:"repeatedgroup,omitempty"` + Optionalgroup *GoTest_OptionalGroup `protobuf:"group,90,opt,name=OptionalGroup" json:"optionalgroup,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTest) Reset() { *m = GoTest{} } +func (m *GoTest) String() string { return proto.CompactTextString(m) } +func (*GoTest) ProtoMessage() {} + +const Default_GoTest_F_BoolDefaulted bool = true +const Default_GoTest_F_Int32Defaulted int32 = 32 +const Default_GoTest_F_Int64Defaulted int64 = 64 +const Default_GoTest_F_Fixed32Defaulted uint32 = 320 +const Default_GoTest_F_Fixed64Defaulted uint64 = 640 +const Default_GoTest_F_Uint32Defaulted uint32 = 3200 +const Default_GoTest_F_Uint64Defaulted uint64 = 6400 +const Default_GoTest_F_FloatDefaulted float32 = 314159 +const Default_GoTest_F_DoubleDefaulted float64 = 271828 +const Default_GoTest_F_StringDefaulted string = "hello, \"world!\"\n" + +var Default_GoTest_F_BytesDefaulted []byte = []byte("Bignose") + +const Default_GoTest_F_Sint32Defaulted int32 = -32 +const Default_GoTest_F_Sint64Defaulted int64 = -64 + +func (m *GoTest) GetKind() GoTest_KIND { + if m != nil && m.Kind != nil { + return *m.Kind + } + return GoTest_VOID +} + +func (m *GoTest) GetTable() string { + if m != nil && m.Table != nil { + return *m.Table + } + return "" +} + +func (m *GoTest) GetParam() int32 { + if m != nil && m.Param != nil { + return *m.Param + } + return 0 +} + +func (m *GoTest) GetRequiredField() *GoTestField { + if m != nil { + return m.RequiredField + } + return nil +} + +func (m *GoTest) GetRepeatedField() []*GoTestField { + if m != nil { + return m.RepeatedField + } + return nil +} + +func (m *GoTest) GetOptionalField() *GoTestField { + if m != nil { + return m.OptionalField + } + return nil +} + +func (m *GoTest) GetF_BoolRequired() bool { + if m != nil && m.F_BoolRequired != nil { + return *m.F_BoolRequired + } + return false +} + +func (m *GoTest) GetF_Int32Required() int32 { + if m != nil && m.F_Int32Required != nil { + return *m.F_Int32Required + } + return 0 +} + +func (m *GoTest) GetF_Int64Required() int64 { + if m != nil && m.F_Int64Required != nil { + return *m.F_Int64Required + } + return 0 +} + +func (m *GoTest) GetF_Fixed32Required() uint32 { + if m != nil && m.F_Fixed32Required != nil { + return *m.F_Fixed32Required + } + return 0 +} + +func (m *GoTest) GetF_Fixed64Required() uint64 { + if m != nil && m.F_Fixed64Required != nil { + return *m.F_Fixed64Required + } + return 0 +} + +func (m *GoTest) GetF_Uint32Required() uint32 { + if m != nil && m.F_Uint32Required != nil { + return *m.F_Uint32Required + } + return 0 +} + +func (m *GoTest) GetF_Uint64Required() uint64 { + if m != nil && m.F_Uint64Required != nil { + return *m.F_Uint64Required + } + return 0 +} + +func (m *GoTest) GetF_FloatRequired() float32 { + if m != nil && m.F_FloatRequired != nil { + return *m.F_FloatRequired + } + return 0 +} + +func (m *GoTest) GetF_DoubleRequired() float64 { + if m != nil && m.F_DoubleRequired != nil { + return *m.F_DoubleRequired + } + return 0 +} + +func (m *GoTest) GetF_StringRequired() string { + if m != nil && m.F_StringRequired != nil { + return *m.F_StringRequired + } + return "" +} + +func (m *GoTest) GetF_BytesRequired() []byte { + if m != nil { + return m.F_BytesRequired + } + return nil +} + +func (m *GoTest) GetF_Sint32Required() int32 { + if m != nil && m.F_Sint32Required != nil { + return *m.F_Sint32Required + } + return 0 +} + +func (m *GoTest) GetF_Sint64Required() int64 { + if m != nil && m.F_Sint64Required != nil { + return *m.F_Sint64Required + } + return 0 +} + +func (m *GoTest) GetF_BoolRepeated() []bool { + if m != nil { + return m.F_BoolRepeated + } + return nil +} + +func (m *GoTest) GetF_Int32Repeated() []int32 { + if m != nil { + return m.F_Int32Repeated + } + return nil +} + +func (m *GoTest) GetF_Int64Repeated() []int64 { + if m != nil { + return m.F_Int64Repeated + } + return nil +} + +func (m *GoTest) GetF_Fixed32Repeated() []uint32 { + if m != nil { + return m.F_Fixed32Repeated + } + return nil +} + +func (m *GoTest) GetF_Fixed64Repeated() []uint64 { + if m != nil { + return m.F_Fixed64Repeated + } + return nil +} + +func (m *GoTest) GetF_Uint32Repeated() []uint32 { + if m != nil { + return m.F_Uint32Repeated + } + return nil +} + +func (m *GoTest) GetF_Uint64Repeated() []uint64 { + if m != nil { + return m.F_Uint64Repeated + } + return nil +} + +func (m *GoTest) GetF_FloatRepeated() []float32 { + if m != nil { + return m.F_FloatRepeated + } + return nil +} + +func (m *GoTest) GetF_DoubleRepeated() []float64 { + if m != nil { + return m.F_DoubleRepeated + } + return nil +} + +func (m *GoTest) GetF_StringRepeated() []string { + if m != nil { + return m.F_StringRepeated + } + return nil +} + +func (m *GoTest) GetF_BytesRepeated() [][]byte { + if m != nil { + return m.F_BytesRepeated + } + return nil +} + +func (m *GoTest) GetF_Sint32Repeated() []int32 { + if m != nil { + return m.F_Sint32Repeated + } + return nil +} + +func (m *GoTest) GetF_Sint64Repeated() []int64 { + if m != nil { + return m.F_Sint64Repeated + } + return nil +} + +func (m *GoTest) GetF_BoolOptional() bool { + if m != nil && m.F_BoolOptional != nil { + return *m.F_BoolOptional + } + return false +} + +func (m *GoTest) GetF_Int32Optional() int32 { + if m != nil && m.F_Int32Optional != nil { + return *m.F_Int32Optional + } + return 0 +} + +func (m *GoTest) GetF_Int64Optional() int64 { + if m != nil && m.F_Int64Optional != nil { + return *m.F_Int64Optional + } + return 0 +} + +func (m *GoTest) GetF_Fixed32Optional() uint32 { + if m != nil && m.F_Fixed32Optional != nil { + return *m.F_Fixed32Optional + } + return 0 +} + +func (m *GoTest) GetF_Fixed64Optional() uint64 { + if m != nil && m.F_Fixed64Optional != nil { + return *m.F_Fixed64Optional + } + return 0 +} + +func (m *GoTest) GetF_Uint32Optional() uint32 { + if m != nil && m.F_Uint32Optional != nil { + return *m.F_Uint32Optional + } + return 0 +} + +func (m *GoTest) GetF_Uint64Optional() uint64 { + if m != nil && m.F_Uint64Optional != nil { + return *m.F_Uint64Optional + } + return 0 +} + +func (m *GoTest) GetF_FloatOptional() float32 { + if m != nil && m.F_FloatOptional != nil { + return *m.F_FloatOptional + } + return 0 +} + +func (m *GoTest) GetF_DoubleOptional() float64 { + if m != nil && m.F_DoubleOptional != nil { + return *m.F_DoubleOptional + } + return 0 +} + +func (m *GoTest) GetF_StringOptional() string { + if m != nil && m.F_StringOptional != nil { + return *m.F_StringOptional + } + return "" +} + +func (m *GoTest) GetF_BytesOptional() []byte { + if m != nil { + return m.F_BytesOptional + } + return nil +} + +func (m *GoTest) GetF_Sint32Optional() int32 { + if m != nil && m.F_Sint32Optional != nil { + return *m.F_Sint32Optional + } + return 0 +} + +func (m *GoTest) GetF_Sint64Optional() int64 { + if m != nil && m.F_Sint64Optional != nil { + return *m.F_Sint64Optional + } + return 0 +} + +func (m *GoTest) GetF_BoolDefaulted() bool { + if m != nil && m.F_BoolDefaulted != nil { + return *m.F_BoolDefaulted + } + return Default_GoTest_F_BoolDefaulted +} + +func (m *GoTest) GetF_Int32Defaulted() int32 { + if m != nil && m.F_Int32Defaulted != nil { + return *m.F_Int32Defaulted + } + return Default_GoTest_F_Int32Defaulted +} + +func (m *GoTest) GetF_Int64Defaulted() int64 { + if m != nil && m.F_Int64Defaulted != nil { + return *m.F_Int64Defaulted + } + return Default_GoTest_F_Int64Defaulted +} + +func (m *GoTest) GetF_Fixed32Defaulted() uint32 { + if m != nil && m.F_Fixed32Defaulted != nil { + return *m.F_Fixed32Defaulted + } + return Default_GoTest_F_Fixed32Defaulted +} + +func (m *GoTest) GetF_Fixed64Defaulted() uint64 { + if m != nil && m.F_Fixed64Defaulted != nil { + return *m.F_Fixed64Defaulted + } + return Default_GoTest_F_Fixed64Defaulted +} + +func (m *GoTest) GetF_Uint32Defaulted() uint32 { + if m != nil && m.F_Uint32Defaulted != nil { + return *m.F_Uint32Defaulted + } + return Default_GoTest_F_Uint32Defaulted +} + +func (m *GoTest) GetF_Uint64Defaulted() uint64 { + if m != nil && m.F_Uint64Defaulted != nil { + return *m.F_Uint64Defaulted + } + return Default_GoTest_F_Uint64Defaulted +} + +func (m *GoTest) GetF_FloatDefaulted() float32 { + if m != nil && m.F_FloatDefaulted != nil { + return *m.F_FloatDefaulted + } + return Default_GoTest_F_FloatDefaulted +} + +func (m *GoTest) GetF_DoubleDefaulted() float64 { + if m != nil && m.F_DoubleDefaulted != nil { + return *m.F_DoubleDefaulted + } + return Default_GoTest_F_DoubleDefaulted +} + +func (m *GoTest) GetF_StringDefaulted() string { + if m != nil && m.F_StringDefaulted != nil { + return *m.F_StringDefaulted + } + return Default_GoTest_F_StringDefaulted +} + +func (m *GoTest) GetF_BytesDefaulted() []byte { + if m != nil && m.F_BytesDefaulted != nil { + return m.F_BytesDefaulted + } + return append([]byte(nil), Default_GoTest_F_BytesDefaulted...) +} + +func (m *GoTest) GetF_Sint32Defaulted() int32 { + if m != nil && m.F_Sint32Defaulted != nil { + return *m.F_Sint32Defaulted + } + return Default_GoTest_F_Sint32Defaulted +} + +func (m *GoTest) GetF_Sint64Defaulted() int64 { + if m != nil && m.F_Sint64Defaulted != nil { + return *m.F_Sint64Defaulted + } + return Default_GoTest_F_Sint64Defaulted +} + +func (m *GoTest) GetF_BoolRepeatedPacked() []bool { + if m != nil { + return m.F_BoolRepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Int32RepeatedPacked() []int32 { + if m != nil { + return m.F_Int32RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Int64RepeatedPacked() []int64 { + if m != nil { + return m.F_Int64RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Fixed32RepeatedPacked() []uint32 { + if m != nil { + return m.F_Fixed32RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Fixed64RepeatedPacked() []uint64 { + if m != nil { + return m.F_Fixed64RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Uint32RepeatedPacked() []uint32 { + if m != nil { + return m.F_Uint32RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Uint64RepeatedPacked() []uint64 { + if m != nil { + return m.F_Uint64RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_FloatRepeatedPacked() []float32 { + if m != nil { + return m.F_FloatRepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_DoubleRepeatedPacked() []float64 { + if m != nil { + return m.F_DoubleRepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Sint32RepeatedPacked() []int32 { + if m != nil { + return m.F_Sint32RepeatedPacked + } + return nil +} + +func (m *GoTest) GetF_Sint64RepeatedPacked() []int64 { + if m != nil { + return m.F_Sint64RepeatedPacked + } + return nil +} + +func (m *GoTest) GetRequiredgroup() *GoTest_RequiredGroup { + if m != nil { + return m.Requiredgroup + } + return nil +} + +func (m *GoTest) GetRepeatedgroup() []*GoTest_RepeatedGroup { + if m != nil { + return m.Repeatedgroup + } + return nil +} + +func (m *GoTest) GetOptionalgroup() *GoTest_OptionalGroup { + if m != nil { + return m.Optionalgroup + } + return nil +} + +// Required, repeated, and optional groups. +type GoTest_RequiredGroup struct { + RequiredField *string `protobuf:"bytes,71,req" json:"RequiredField,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTest_RequiredGroup) Reset() { *m = GoTest_RequiredGroup{} } +func (m *GoTest_RequiredGroup) String() string { return proto.CompactTextString(m) } +func (*GoTest_RequiredGroup) ProtoMessage() {} + +func (m *GoTest_RequiredGroup) GetRequiredField() string { + if m != nil && m.RequiredField != nil { + return *m.RequiredField + } + return "" +} + +type GoTest_RepeatedGroup struct { + RequiredField *string `protobuf:"bytes,81,req" json:"RequiredField,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTest_RepeatedGroup) Reset() { *m = GoTest_RepeatedGroup{} } +func (m *GoTest_RepeatedGroup) String() string { return proto.CompactTextString(m) } +func (*GoTest_RepeatedGroup) ProtoMessage() {} + +func (m *GoTest_RepeatedGroup) GetRequiredField() string { + if m != nil && m.RequiredField != nil { + return *m.RequiredField + } + return "" +} + +type GoTest_OptionalGroup struct { + RequiredField *string `protobuf:"bytes,91,req" json:"RequiredField,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoTest_OptionalGroup) Reset() { *m = GoTest_OptionalGroup{} } +func (m *GoTest_OptionalGroup) String() string { return proto.CompactTextString(m) } +func (*GoTest_OptionalGroup) ProtoMessage() {} + +func (m *GoTest_OptionalGroup) GetRequiredField() string { + if m != nil && m.RequiredField != nil { + return *m.RequiredField + } + return "" +} + +// For testing skipping of unrecognized fields. +// Numbers are all big, larger than tag numbers in GoTestField, +// the message used in the corresponding test. +type GoSkipTest struct { + SkipInt32 *int32 `protobuf:"varint,11,req,name=skip_int32" json:"skip_int32,omitempty"` + SkipFixed32 *uint32 `protobuf:"fixed32,12,req,name=skip_fixed32" json:"skip_fixed32,omitempty"` + SkipFixed64 *uint64 `protobuf:"fixed64,13,req,name=skip_fixed64" json:"skip_fixed64,omitempty"` + SkipString *string `protobuf:"bytes,14,req,name=skip_string" json:"skip_string,omitempty"` + Skipgroup *GoSkipTest_SkipGroup `protobuf:"group,15,req,name=SkipGroup" json:"skipgroup,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoSkipTest) Reset() { *m = GoSkipTest{} } +func (m *GoSkipTest) String() string { return proto.CompactTextString(m) } +func (*GoSkipTest) ProtoMessage() {} + +func (m *GoSkipTest) GetSkipInt32() int32 { + if m != nil && m.SkipInt32 != nil { + return *m.SkipInt32 + } + return 0 +} + +func (m *GoSkipTest) GetSkipFixed32() uint32 { + if m != nil && m.SkipFixed32 != nil { + return *m.SkipFixed32 + } + return 0 +} + +func (m *GoSkipTest) GetSkipFixed64() uint64 { + if m != nil && m.SkipFixed64 != nil { + return *m.SkipFixed64 + } + return 0 +} + +func (m *GoSkipTest) GetSkipString() string { + if m != nil && m.SkipString != nil { + return *m.SkipString + } + return "" +} + +func (m *GoSkipTest) GetSkipgroup() *GoSkipTest_SkipGroup { + if m != nil { + return m.Skipgroup + } + return nil +} + +type GoSkipTest_SkipGroup struct { + GroupInt32 *int32 `protobuf:"varint,16,req,name=group_int32" json:"group_int32,omitempty"` + GroupString *string `protobuf:"bytes,17,req,name=group_string" json:"group_string,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GoSkipTest_SkipGroup) Reset() { *m = GoSkipTest_SkipGroup{} } +func (m *GoSkipTest_SkipGroup) String() string { return proto.CompactTextString(m) } +func (*GoSkipTest_SkipGroup) ProtoMessage() {} + +func (m *GoSkipTest_SkipGroup) GetGroupInt32() int32 { + if m != nil && m.GroupInt32 != nil { + return *m.GroupInt32 + } + return 0 +} + +func (m *GoSkipTest_SkipGroup) GetGroupString() string { + if m != nil && m.GroupString != nil { + return *m.GroupString + } + return "" +} + +// For testing packed/non-packed decoder switching. +// A serialized instance of one should be deserializable as the other. +type NonPackedTest struct { + A []int32 `protobuf:"varint,1,rep,name=a" json:"a,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *NonPackedTest) Reset() { *m = NonPackedTest{} } +func (m *NonPackedTest) String() string { return proto.CompactTextString(m) } +func (*NonPackedTest) ProtoMessage() {} + +func (m *NonPackedTest) GetA() []int32 { + if m != nil { + return m.A + } + return nil +} + +type PackedTest struct { + B []int32 `protobuf:"varint,1,rep,packed,name=b" json:"b,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *PackedTest) Reset() { *m = PackedTest{} } +func (m *PackedTest) String() string { return proto.CompactTextString(m) } +func (*PackedTest) ProtoMessage() {} + +func (m *PackedTest) GetB() []int32 { + if m != nil { + return m.B + } + return nil +} + +type MaxTag struct { + // Maximum possible tag number. + LastField *string `protobuf:"bytes,536870911,opt,name=last_field" json:"last_field,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MaxTag) Reset() { *m = MaxTag{} } +func (m *MaxTag) String() string { return proto.CompactTextString(m) } +func (*MaxTag) ProtoMessage() {} + +func (m *MaxTag) GetLastField() string { + if m != nil && m.LastField != nil { + return *m.LastField + } + return "" +} + +type OldMessage struct { + Nested *OldMessage_Nested `protobuf:"bytes,1,opt,name=nested" json:"nested,omitempty"` + Num *int32 `protobuf:"varint,2,opt,name=num" json:"num,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *OldMessage) Reset() { *m = OldMessage{} } +func (m *OldMessage) String() string { return proto.CompactTextString(m) } +func (*OldMessage) ProtoMessage() {} + +func (m *OldMessage) GetNested() *OldMessage_Nested { + if m != nil { + return m.Nested + } + return nil +} + +func (m *OldMessage) GetNum() int32 { + if m != nil && m.Num != nil { + return *m.Num + } + return 0 +} + +type OldMessage_Nested struct { + Name *string `protobuf:"bytes,1,opt,name=name" json:"name,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *OldMessage_Nested) Reset() { *m = OldMessage_Nested{} } +func (m *OldMessage_Nested) String() string { return proto.CompactTextString(m) } +func (*OldMessage_Nested) ProtoMessage() {} + +func (m *OldMessage_Nested) GetName() string { + if m != nil && m.Name != nil { + return *m.Name + } + return "" +} + +// NewMessage is wire compatible with OldMessage; +// imagine it as a future version. +type NewMessage struct { + Nested *NewMessage_Nested `protobuf:"bytes,1,opt,name=nested" json:"nested,omitempty"` + // This is an int32 in OldMessage. + Num *int64 `protobuf:"varint,2,opt,name=num" json:"num,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *NewMessage) Reset() { *m = NewMessage{} } +func (m *NewMessage) String() string { return proto.CompactTextString(m) } +func (*NewMessage) ProtoMessage() {} + +func (m *NewMessage) GetNested() *NewMessage_Nested { + if m != nil { + return m.Nested + } + return nil +} + +func (m *NewMessage) GetNum() int64 { + if m != nil && m.Num != nil { + return *m.Num + } + return 0 +} + +type NewMessage_Nested struct { + Name *string `protobuf:"bytes,1,opt,name=name" json:"name,omitempty"` + FoodGroup *string `protobuf:"bytes,2,opt,name=food_group" json:"food_group,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *NewMessage_Nested) Reset() { *m = NewMessage_Nested{} } +func (m *NewMessage_Nested) String() string { return proto.CompactTextString(m) } +func (*NewMessage_Nested) ProtoMessage() {} + +func (m *NewMessage_Nested) GetName() string { + if m != nil && m.Name != nil { + return *m.Name + } + return "" +} + +func (m *NewMessage_Nested) GetFoodGroup() string { + if m != nil && m.FoodGroup != nil { + return *m.FoodGroup + } + return "" +} + +type InnerMessage struct { + Host *string `protobuf:"bytes,1,req,name=host" json:"host,omitempty"` + Port *int32 `protobuf:"varint,2,opt,name=port,def=4000" json:"port,omitempty"` + Connected *bool `protobuf:"varint,3,opt,name=connected" json:"connected,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *InnerMessage) Reset() { *m = InnerMessage{} } +func (m *InnerMessage) String() string { return proto.CompactTextString(m) } +func (*InnerMessage) ProtoMessage() {} + +const Default_InnerMessage_Port int32 = 4000 + +func (m *InnerMessage) GetHost() string { + if m != nil && m.Host != nil { + return *m.Host + } + return "" +} + +func (m *InnerMessage) GetPort() int32 { + if m != nil && m.Port != nil { + return *m.Port + } + return Default_InnerMessage_Port +} + +func (m *InnerMessage) GetConnected() bool { + if m != nil && m.Connected != nil { + return *m.Connected + } + return false +} + +type OtherMessage struct { + Key *int64 `protobuf:"varint,1,opt,name=key" json:"key,omitempty"` + Value []byte `protobuf:"bytes,2,opt,name=value" json:"value,omitempty"` + Weight *float32 `protobuf:"fixed32,3,opt,name=weight" json:"weight,omitempty"` + Inner *InnerMessage `protobuf:"bytes,4,opt,name=inner" json:"inner,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *OtherMessage) Reset() { *m = OtherMessage{} } +func (m *OtherMessage) String() string { return proto.CompactTextString(m) } +func (*OtherMessage) ProtoMessage() {} + +func (m *OtherMessage) GetKey() int64 { + if m != nil && m.Key != nil { + return *m.Key + } + return 0 +} + +func (m *OtherMessage) GetValue() []byte { + if m != nil { + return m.Value + } + return nil +} + +func (m *OtherMessage) GetWeight() float32 { + if m != nil && m.Weight != nil { + return *m.Weight + } + return 0 +} + +func (m *OtherMessage) GetInner() *InnerMessage { + if m != nil { + return m.Inner + } + return nil +} + +type MyMessage struct { + Count *int32 `protobuf:"varint,1,req,name=count" json:"count,omitempty"` + Name *string `protobuf:"bytes,2,opt,name=name" json:"name,omitempty"` + Quote *string `protobuf:"bytes,3,opt,name=quote" json:"quote,omitempty"` + Pet []string `protobuf:"bytes,4,rep,name=pet" json:"pet,omitempty"` + Inner *InnerMessage `protobuf:"bytes,5,opt,name=inner" json:"inner,omitempty"` + Others []*OtherMessage `protobuf:"bytes,6,rep,name=others" json:"others,omitempty"` + RepInner []*InnerMessage `protobuf:"bytes,12,rep,name=rep_inner" json:"rep_inner,omitempty"` + Bikeshed *MyMessage_Color `protobuf:"varint,7,opt,name=bikeshed,enum=testdata.MyMessage_Color" json:"bikeshed,omitempty"` + Somegroup *MyMessage_SomeGroup `protobuf:"group,8,opt,name=SomeGroup" json:"somegroup,omitempty"` + // This field becomes [][]byte in the generated code. + RepBytes [][]byte `protobuf:"bytes,10,rep,name=rep_bytes" json:"rep_bytes,omitempty"` + Bigfloat *float64 `protobuf:"fixed64,11,opt,name=bigfloat" json:"bigfloat,omitempty"` + XXX_extensions map[int32]proto.Extension `json:"-"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MyMessage) Reset() { *m = MyMessage{} } +func (m *MyMessage) String() string { return proto.CompactTextString(m) } +func (*MyMessage) ProtoMessage() {} + +var extRange_MyMessage = []proto.ExtensionRange{ + {100, 536870911}, +} + +func (*MyMessage) ExtensionRangeArray() []proto.ExtensionRange { + return extRange_MyMessage +} +func (m *MyMessage) ExtensionMap() map[int32]proto.Extension { + if m.XXX_extensions == nil { + m.XXX_extensions = make(map[int32]proto.Extension) + } + return m.XXX_extensions +} + +func (m *MyMessage) GetCount() int32 { + if m != nil && m.Count != nil { + return *m.Count + } + return 0 +} + +func (m *MyMessage) GetName() string { + if m != nil && m.Name != nil { + return *m.Name + } + return "" +} + +func (m *MyMessage) GetQuote() string { + if m != nil && m.Quote != nil { + return *m.Quote + } + return "" +} + +func (m *MyMessage) GetPet() []string { + if m != nil { + return m.Pet + } + return nil +} + +func (m *MyMessage) GetInner() *InnerMessage { + if m != nil { + return m.Inner + } + return nil +} + +func (m *MyMessage) GetOthers() []*OtherMessage { + if m != nil { + return m.Others + } + return nil +} + +func (m *MyMessage) GetRepInner() []*InnerMessage { + if m != nil { + return m.RepInner + } + return nil +} + +func (m *MyMessage) GetBikeshed() MyMessage_Color { + if m != nil && m.Bikeshed != nil { + return *m.Bikeshed + } + return MyMessage_RED +} + +func (m *MyMessage) GetSomegroup() *MyMessage_SomeGroup { + if m != nil { + return m.Somegroup + } + return nil +} + +func (m *MyMessage) GetRepBytes() [][]byte { + if m != nil { + return m.RepBytes + } + return nil +} + +func (m *MyMessage) GetBigfloat() float64 { + if m != nil && m.Bigfloat != nil { + return *m.Bigfloat + } + return 0 +} + +type MyMessage_SomeGroup struct { + GroupField *int32 `protobuf:"varint,9,opt,name=group_field" json:"group_field,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MyMessage_SomeGroup) Reset() { *m = MyMessage_SomeGroup{} } +func (m *MyMessage_SomeGroup) String() string { return proto.CompactTextString(m) } +func (*MyMessage_SomeGroup) ProtoMessage() {} + +func (m *MyMessage_SomeGroup) GetGroupField() int32 { + if m != nil && m.GroupField != nil { + return *m.GroupField + } + return 0 +} + +type Ext struct { + Data *string `protobuf:"bytes,1,opt,name=data" json:"data,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Ext) Reset() { *m = Ext{} } +func (m *Ext) String() string { return proto.CompactTextString(m) } +func (*Ext) ProtoMessage() {} + +func (m *Ext) GetData() string { + if m != nil && m.Data != nil { + return *m.Data + } + return "" +} + +var E_Ext_More = &proto.ExtensionDesc{ + ExtendedType: (*MyMessage)(nil), + ExtensionType: (*Ext)(nil), + Field: 103, + Name: "testdata.Ext.more", + Tag: "bytes,103,opt,name=more", +} + +var E_Ext_Text = &proto.ExtensionDesc{ + ExtendedType: (*MyMessage)(nil), + ExtensionType: (*string)(nil), + Field: 104, + Name: "testdata.Ext.text", + Tag: "bytes,104,opt,name=text", +} + +var E_Ext_Number = &proto.ExtensionDesc{ + ExtendedType: (*MyMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 105, + Name: "testdata.Ext.number", + Tag: "varint,105,opt,name=number", +} + +type DefaultsMessage struct { + XXX_extensions map[int32]proto.Extension `json:"-"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *DefaultsMessage) Reset() { *m = DefaultsMessage{} } +func (m *DefaultsMessage) String() string { return proto.CompactTextString(m) } +func (*DefaultsMessage) ProtoMessage() {} + +var extRange_DefaultsMessage = []proto.ExtensionRange{ + {100, 536870911}, +} + +func (*DefaultsMessage) ExtensionRangeArray() []proto.ExtensionRange { + return extRange_DefaultsMessage +} +func (m *DefaultsMessage) ExtensionMap() map[int32]proto.Extension { + if m.XXX_extensions == nil { + m.XXX_extensions = make(map[int32]proto.Extension) + } + return m.XXX_extensions +} + +type MyMessageSet struct { + XXX_extensions map[int32]proto.Extension `json:"-"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MyMessageSet) Reset() { *m = MyMessageSet{} } +func (m *MyMessageSet) String() string { return proto.CompactTextString(m) } +func (*MyMessageSet) ProtoMessage() {} + +func (m *MyMessageSet) Marshal() ([]byte, error) { + return proto.MarshalMessageSet(m.ExtensionMap()) +} +func (m *MyMessageSet) Unmarshal(buf []byte) error { + return proto.UnmarshalMessageSet(buf, m.ExtensionMap()) +} +func (m *MyMessageSet) MarshalJSON() ([]byte, error) { + return proto.MarshalMessageSetJSON(m.XXX_extensions) +} +func (m *MyMessageSet) UnmarshalJSON(buf []byte) error { + return proto.UnmarshalMessageSetJSON(buf, m.XXX_extensions) +} + +// ensure MyMessageSet satisfies proto.Marshaler and proto.Unmarshaler +var _ proto.Marshaler = (*MyMessageSet)(nil) +var _ proto.Unmarshaler = (*MyMessageSet)(nil) + +var extRange_MyMessageSet = []proto.ExtensionRange{ + {100, 2147483646}, +} + +func (*MyMessageSet) ExtensionRangeArray() []proto.ExtensionRange { + return extRange_MyMessageSet +} +func (m *MyMessageSet) ExtensionMap() map[int32]proto.Extension { + if m.XXX_extensions == nil { + m.XXX_extensions = make(map[int32]proto.Extension) + } + return m.XXX_extensions +} + +type Empty struct { + XXX_unrecognized []byte `json:"-"` +} + +func (m *Empty) Reset() { *m = Empty{} } +func (m *Empty) String() string { return proto.CompactTextString(m) } +func (*Empty) ProtoMessage() {} + +type MessageList struct { + Message []*MessageList_Message `protobuf:"group,1,rep" json:"message,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MessageList) Reset() { *m = MessageList{} } +func (m *MessageList) String() string { return proto.CompactTextString(m) } +func (*MessageList) ProtoMessage() {} + +func (m *MessageList) GetMessage() []*MessageList_Message { + if m != nil { + return m.Message + } + return nil +} + +type MessageList_Message struct { + Name *string `protobuf:"bytes,2,req,name=name" json:"name,omitempty"` + Count *int32 `protobuf:"varint,3,req,name=count" json:"count,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MessageList_Message) Reset() { *m = MessageList_Message{} } +func (m *MessageList_Message) String() string { return proto.CompactTextString(m) } +func (*MessageList_Message) ProtoMessage() {} + +func (m *MessageList_Message) GetName() string { + if m != nil && m.Name != nil { + return *m.Name + } + return "" +} + +func (m *MessageList_Message) GetCount() int32 { + if m != nil && m.Count != nil { + return *m.Count + } + return 0 +} + +type Strings struct { + StringField *string `protobuf:"bytes,1,opt,name=string_field" json:"string_field,omitempty"` + BytesField []byte `protobuf:"bytes,2,opt,name=bytes_field" json:"bytes_field,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Strings) Reset() { *m = Strings{} } +func (m *Strings) String() string { return proto.CompactTextString(m) } +func (*Strings) ProtoMessage() {} + +func (m *Strings) GetStringField() string { + if m != nil && m.StringField != nil { + return *m.StringField + } + return "" +} + +func (m *Strings) GetBytesField() []byte { + if m != nil { + return m.BytesField + } + return nil +} + +type Defaults struct { + // Default-valued fields of all basic types. + // Same as GoTest, but copied here to make testing easier. + F_Bool *bool `protobuf:"varint,1,opt,def=1" json:"F_Bool,omitempty"` + F_Int32 *int32 `protobuf:"varint,2,opt,def=32" json:"F_Int32,omitempty"` + F_Int64 *int64 `protobuf:"varint,3,opt,def=64" json:"F_Int64,omitempty"` + F_Fixed32 *uint32 `protobuf:"fixed32,4,opt,def=320" json:"F_Fixed32,omitempty"` + F_Fixed64 *uint64 `protobuf:"fixed64,5,opt,def=640" json:"F_Fixed64,omitempty"` + F_Uint32 *uint32 `protobuf:"varint,6,opt,def=3200" json:"F_Uint32,omitempty"` + F_Uint64 *uint64 `protobuf:"varint,7,opt,def=6400" json:"F_Uint64,omitempty"` + F_Float *float32 `protobuf:"fixed32,8,opt,def=314159" json:"F_Float,omitempty"` + F_Double *float64 `protobuf:"fixed64,9,opt,def=271828" json:"F_Double,omitempty"` + F_String *string `protobuf:"bytes,10,opt,def=hello, \"world!\"\n" json:"F_String,omitempty"` + F_Bytes []byte `protobuf:"bytes,11,opt,def=Bignose" json:"F_Bytes,omitempty"` + F_Sint32 *int32 `protobuf:"zigzag32,12,opt,def=-32" json:"F_Sint32,omitempty"` + F_Sint64 *int64 `protobuf:"zigzag64,13,opt,def=-64" json:"F_Sint64,omitempty"` + F_Enum *Defaults_Color `protobuf:"varint,14,opt,enum=testdata.Defaults_Color,def=1" json:"F_Enum,omitempty"` + // More fields with crazy defaults. + F_Pinf *float32 `protobuf:"fixed32,15,opt,def=inf" json:"F_Pinf,omitempty"` + F_Ninf *float32 `protobuf:"fixed32,16,opt,def=-inf" json:"F_Ninf,omitempty"` + F_Nan *float32 `protobuf:"fixed32,17,opt,def=nan" json:"F_Nan,omitempty"` + // Sub-message. + Sub *SubDefaults `protobuf:"bytes,18,opt,name=sub" json:"sub,omitempty"` + // Redundant but explicit defaults. + StrZero *string `protobuf:"bytes,19,opt,name=str_zero,def=" json:"str_zero,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *Defaults) Reset() { *m = Defaults{} } +func (m *Defaults) String() string { return proto.CompactTextString(m) } +func (*Defaults) ProtoMessage() {} + +const Default_Defaults_F_Bool bool = true +const Default_Defaults_F_Int32 int32 = 32 +const Default_Defaults_F_Int64 int64 = 64 +const Default_Defaults_F_Fixed32 uint32 = 320 +const Default_Defaults_F_Fixed64 uint64 = 640 +const Default_Defaults_F_Uint32 uint32 = 3200 +const Default_Defaults_F_Uint64 uint64 = 6400 +const Default_Defaults_F_Float float32 = 314159 +const Default_Defaults_F_Double float64 = 271828 +const Default_Defaults_F_String string = "hello, \"world!\"\n" + +var Default_Defaults_F_Bytes []byte = []byte("Bignose") + +const Default_Defaults_F_Sint32 int32 = -32 +const Default_Defaults_F_Sint64 int64 = -64 +const Default_Defaults_F_Enum Defaults_Color = Defaults_GREEN + +var Default_Defaults_F_Pinf float32 = float32(math.Inf(1)) +var Default_Defaults_F_Ninf float32 = float32(math.Inf(-1)) +var Default_Defaults_F_Nan float32 = float32(math.NaN()) + +func (m *Defaults) GetF_Bool() bool { + if m != nil && m.F_Bool != nil { + return *m.F_Bool + } + return Default_Defaults_F_Bool +} + +func (m *Defaults) GetF_Int32() int32 { + if m != nil && m.F_Int32 != nil { + return *m.F_Int32 + } + return Default_Defaults_F_Int32 +} + +func (m *Defaults) GetF_Int64() int64 { + if m != nil && m.F_Int64 != nil { + return *m.F_Int64 + } + return Default_Defaults_F_Int64 +} + +func (m *Defaults) GetF_Fixed32() uint32 { + if m != nil && m.F_Fixed32 != nil { + return *m.F_Fixed32 + } + return Default_Defaults_F_Fixed32 +} + +func (m *Defaults) GetF_Fixed64() uint64 { + if m != nil && m.F_Fixed64 != nil { + return *m.F_Fixed64 + } + return Default_Defaults_F_Fixed64 +} + +func (m *Defaults) GetF_Uint32() uint32 { + if m != nil && m.F_Uint32 != nil { + return *m.F_Uint32 + } + return Default_Defaults_F_Uint32 +} + +func (m *Defaults) GetF_Uint64() uint64 { + if m != nil && m.F_Uint64 != nil { + return *m.F_Uint64 + } + return Default_Defaults_F_Uint64 +} + +func (m *Defaults) GetF_Float() float32 { + if m != nil && m.F_Float != nil { + return *m.F_Float + } + return Default_Defaults_F_Float +} + +func (m *Defaults) GetF_Double() float64 { + if m != nil && m.F_Double != nil { + return *m.F_Double + } + return Default_Defaults_F_Double +} + +func (m *Defaults) GetF_String() string { + if m != nil && m.F_String != nil { + return *m.F_String + } + return Default_Defaults_F_String +} + +func (m *Defaults) GetF_Bytes() []byte { + if m != nil && m.F_Bytes != nil { + return m.F_Bytes + } + return append([]byte(nil), Default_Defaults_F_Bytes...) +} + +func (m *Defaults) GetF_Sint32() int32 { + if m != nil && m.F_Sint32 != nil { + return *m.F_Sint32 + } + return Default_Defaults_F_Sint32 +} + +func (m *Defaults) GetF_Sint64() int64 { + if m != nil && m.F_Sint64 != nil { + return *m.F_Sint64 + } + return Default_Defaults_F_Sint64 +} + +func (m *Defaults) GetF_Enum() Defaults_Color { + if m != nil && m.F_Enum != nil { + return *m.F_Enum + } + return Default_Defaults_F_Enum +} + +func (m *Defaults) GetF_Pinf() float32 { + if m != nil && m.F_Pinf != nil { + return *m.F_Pinf + } + return Default_Defaults_F_Pinf +} + +func (m *Defaults) GetF_Ninf() float32 { + if m != nil && m.F_Ninf != nil { + return *m.F_Ninf + } + return Default_Defaults_F_Ninf +} + +func (m *Defaults) GetF_Nan() float32 { + if m != nil && m.F_Nan != nil { + return *m.F_Nan + } + return Default_Defaults_F_Nan +} + +func (m *Defaults) GetSub() *SubDefaults { + if m != nil { + return m.Sub + } + return nil +} + +func (m *Defaults) GetStrZero() string { + if m != nil && m.StrZero != nil { + return *m.StrZero + } + return "" +} + +type SubDefaults struct { + N *int64 `protobuf:"varint,1,opt,name=n,def=7" json:"n,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *SubDefaults) Reset() { *m = SubDefaults{} } +func (m *SubDefaults) String() string { return proto.CompactTextString(m) } +func (*SubDefaults) ProtoMessage() {} + +const Default_SubDefaults_N int64 = 7 + +func (m *SubDefaults) GetN() int64 { + if m != nil && m.N != nil { + return *m.N + } + return Default_SubDefaults_N +} + +type RepeatedEnum struct { + Color []RepeatedEnum_Color `protobuf:"varint,1,rep,name=color,enum=testdata.RepeatedEnum_Color" json:"color,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *RepeatedEnum) Reset() { *m = RepeatedEnum{} } +func (m *RepeatedEnum) String() string { return proto.CompactTextString(m) } +func (*RepeatedEnum) ProtoMessage() {} + +func (m *RepeatedEnum) GetColor() []RepeatedEnum_Color { + if m != nil { + return m.Color + } + return nil +} + +type MoreRepeated struct { + Bools []bool `protobuf:"varint,1,rep,name=bools" json:"bools,omitempty"` + BoolsPacked []bool `protobuf:"varint,2,rep,packed,name=bools_packed" json:"bools_packed,omitempty"` + Ints []int32 `protobuf:"varint,3,rep,name=ints" json:"ints,omitempty"` + IntsPacked []int32 `protobuf:"varint,4,rep,packed,name=ints_packed" json:"ints_packed,omitempty"` + Int64SPacked []int64 `protobuf:"varint,7,rep,packed,name=int64s_packed" json:"int64s_packed,omitempty"` + Strings []string `protobuf:"bytes,5,rep,name=strings" json:"strings,omitempty"` + Fixeds []uint32 `protobuf:"fixed32,6,rep,name=fixeds" json:"fixeds,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MoreRepeated) Reset() { *m = MoreRepeated{} } +func (m *MoreRepeated) String() string { return proto.CompactTextString(m) } +func (*MoreRepeated) ProtoMessage() {} + +func (m *MoreRepeated) GetBools() []bool { + if m != nil { + return m.Bools + } + return nil +} + +func (m *MoreRepeated) GetBoolsPacked() []bool { + if m != nil { + return m.BoolsPacked + } + return nil +} + +func (m *MoreRepeated) GetInts() []int32 { + if m != nil { + return m.Ints + } + return nil +} + +func (m *MoreRepeated) GetIntsPacked() []int32 { + if m != nil { + return m.IntsPacked + } + return nil +} + +func (m *MoreRepeated) GetInt64SPacked() []int64 { + if m != nil { + return m.Int64SPacked + } + return nil +} + +func (m *MoreRepeated) GetStrings() []string { + if m != nil { + return m.Strings + } + return nil +} + +func (m *MoreRepeated) GetFixeds() []uint32 { + if m != nil { + return m.Fixeds + } + return nil +} + +type GroupOld struct { + G *GroupOld_G `protobuf:"group,101,opt" json:"g,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GroupOld) Reset() { *m = GroupOld{} } +func (m *GroupOld) String() string { return proto.CompactTextString(m) } +func (*GroupOld) ProtoMessage() {} + +func (m *GroupOld) GetG() *GroupOld_G { + if m != nil { + return m.G + } + return nil +} + +type GroupOld_G struct { + X *int32 `protobuf:"varint,2,opt,name=x" json:"x,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GroupOld_G) Reset() { *m = GroupOld_G{} } +func (m *GroupOld_G) String() string { return proto.CompactTextString(m) } +func (*GroupOld_G) ProtoMessage() {} + +func (m *GroupOld_G) GetX() int32 { + if m != nil && m.X != nil { + return *m.X + } + return 0 +} + +type GroupNew struct { + G *GroupNew_G `protobuf:"group,101,opt" json:"g,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GroupNew) Reset() { *m = GroupNew{} } +func (m *GroupNew) String() string { return proto.CompactTextString(m) } +func (*GroupNew) ProtoMessage() {} + +func (m *GroupNew) GetG() *GroupNew_G { + if m != nil { + return m.G + } + return nil +} + +type GroupNew_G struct { + X *int32 `protobuf:"varint,2,opt,name=x" json:"x,omitempty"` + Y *int32 `protobuf:"varint,3,opt,name=y" json:"y,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *GroupNew_G) Reset() { *m = GroupNew_G{} } +func (m *GroupNew_G) String() string { return proto.CompactTextString(m) } +func (*GroupNew_G) ProtoMessage() {} + +func (m *GroupNew_G) GetX() int32 { + if m != nil && m.X != nil { + return *m.X + } + return 0 +} + +func (m *GroupNew_G) GetY() int32 { + if m != nil && m.Y != nil { + return *m.Y + } + return 0 +} + +type FloatingPoint struct { + F *float64 `protobuf:"fixed64,1,req,name=f" json:"f,omitempty"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *FloatingPoint) Reset() { *m = FloatingPoint{} } +func (m *FloatingPoint) String() string { return proto.CompactTextString(m) } +func (*FloatingPoint) ProtoMessage() {} + +func (m *FloatingPoint) GetF() float64 { + if m != nil && m.F != nil { + return *m.F + } + return 0 +} + +type MessageWithMap struct { + NameMapping map[int32]string `protobuf:"bytes,1,rep,name=name_mapping" json:"name_mapping,omitempty" protobuf_key:"varint,1,opt,name=key" protobuf_val:"bytes,2,opt,name=value"` + MsgMapping map[int64]*FloatingPoint `protobuf:"bytes,2,rep,name=msg_mapping" json:"msg_mapping,omitempty" protobuf_key:"zigzag64,1,opt,name=key" protobuf_val:"bytes,2,opt,name=value"` + ByteMapping map[bool][]byte `protobuf:"bytes,3,rep,name=byte_mapping" json:"byte_mapping,omitempty" protobuf_key:"varint,1,opt,name=key" protobuf_val:"bytes,2,opt,name=value"` + StrToStr map[string]string `protobuf:"bytes,4,rep,name=str_to_str" json:"str_to_str,omitempty" protobuf_key:"bytes,1,opt,name=key" protobuf_val:"bytes,2,opt,name=value"` + XXX_unrecognized []byte `json:"-"` +} + +func (m *MessageWithMap) Reset() { *m = MessageWithMap{} } +func (m *MessageWithMap) String() string { return proto.CompactTextString(m) } +func (*MessageWithMap) ProtoMessage() {} + +func (m *MessageWithMap) GetNameMapping() map[int32]string { + if m != nil { + return m.NameMapping + } + return nil +} + +func (m *MessageWithMap) GetMsgMapping() map[int64]*FloatingPoint { + if m != nil { + return m.MsgMapping + } + return nil +} + +func (m *MessageWithMap) GetByteMapping() map[bool][]byte { + if m != nil { + return m.ByteMapping + } + return nil +} + +func (m *MessageWithMap) GetStrToStr() map[string]string { + if m != nil { + return m.StrToStr + } + return nil +} + +var E_Greeting = &proto.ExtensionDesc{ + ExtendedType: (*MyMessage)(nil), + ExtensionType: ([]string)(nil), + Field: 106, + Name: "testdata.greeting", + Tag: "bytes,106,rep,name=greeting", +} + +var E_NoDefaultDouble = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*float64)(nil), + Field: 101, + Name: "testdata.no_default_double", + Tag: "fixed64,101,opt,name=no_default_double", +} + +var E_NoDefaultFloat = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*float32)(nil), + Field: 102, + Name: "testdata.no_default_float", + Tag: "fixed32,102,opt,name=no_default_float", +} + +var E_NoDefaultInt32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 103, + Name: "testdata.no_default_int32", + Tag: "varint,103,opt,name=no_default_int32", +} + +var E_NoDefaultInt64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int64)(nil), + Field: 104, + Name: "testdata.no_default_int64", + Tag: "varint,104,opt,name=no_default_int64", +} + +var E_NoDefaultUint32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint32)(nil), + Field: 105, + Name: "testdata.no_default_uint32", + Tag: "varint,105,opt,name=no_default_uint32", +} + +var E_NoDefaultUint64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint64)(nil), + Field: 106, + Name: "testdata.no_default_uint64", + Tag: "varint,106,opt,name=no_default_uint64", +} + +var E_NoDefaultSint32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 107, + Name: "testdata.no_default_sint32", + Tag: "zigzag32,107,opt,name=no_default_sint32", +} + +var E_NoDefaultSint64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int64)(nil), + Field: 108, + Name: "testdata.no_default_sint64", + Tag: "zigzag64,108,opt,name=no_default_sint64", +} + +var E_NoDefaultFixed32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint32)(nil), + Field: 109, + Name: "testdata.no_default_fixed32", + Tag: "fixed32,109,opt,name=no_default_fixed32", +} + +var E_NoDefaultFixed64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint64)(nil), + Field: 110, + Name: "testdata.no_default_fixed64", + Tag: "fixed64,110,opt,name=no_default_fixed64", +} + +var E_NoDefaultSfixed32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 111, + Name: "testdata.no_default_sfixed32", + Tag: "fixed32,111,opt,name=no_default_sfixed32", +} + +var E_NoDefaultSfixed64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int64)(nil), + Field: 112, + Name: "testdata.no_default_sfixed64", + Tag: "fixed64,112,opt,name=no_default_sfixed64", +} + +var E_NoDefaultBool = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*bool)(nil), + Field: 113, + Name: "testdata.no_default_bool", + Tag: "varint,113,opt,name=no_default_bool", +} + +var E_NoDefaultString = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*string)(nil), + Field: 114, + Name: "testdata.no_default_string", + Tag: "bytes,114,opt,name=no_default_string", +} + +var E_NoDefaultBytes = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: ([]byte)(nil), + Field: 115, + Name: "testdata.no_default_bytes", + Tag: "bytes,115,opt,name=no_default_bytes", +} + +var E_NoDefaultEnum = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*DefaultsMessage_DefaultsEnum)(nil), + Field: 116, + Name: "testdata.no_default_enum", + Tag: "varint,116,opt,name=no_default_enum,enum=testdata.DefaultsMessage_DefaultsEnum", +} + +var E_DefaultDouble = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*float64)(nil), + Field: 201, + Name: "testdata.default_double", + Tag: "fixed64,201,opt,name=default_double,def=3.1415", +} + +var E_DefaultFloat = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*float32)(nil), + Field: 202, + Name: "testdata.default_float", + Tag: "fixed32,202,opt,name=default_float,def=3.14", +} + +var E_DefaultInt32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 203, + Name: "testdata.default_int32", + Tag: "varint,203,opt,name=default_int32,def=42", +} + +var E_DefaultInt64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int64)(nil), + Field: 204, + Name: "testdata.default_int64", + Tag: "varint,204,opt,name=default_int64,def=43", +} + +var E_DefaultUint32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint32)(nil), + Field: 205, + Name: "testdata.default_uint32", + Tag: "varint,205,opt,name=default_uint32,def=44", +} + +var E_DefaultUint64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint64)(nil), + Field: 206, + Name: "testdata.default_uint64", + Tag: "varint,206,opt,name=default_uint64,def=45", +} + +var E_DefaultSint32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 207, + Name: "testdata.default_sint32", + Tag: "zigzag32,207,opt,name=default_sint32,def=46", +} + +var E_DefaultSint64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int64)(nil), + Field: 208, + Name: "testdata.default_sint64", + Tag: "zigzag64,208,opt,name=default_sint64,def=47", +} + +var E_DefaultFixed32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint32)(nil), + Field: 209, + Name: "testdata.default_fixed32", + Tag: "fixed32,209,opt,name=default_fixed32,def=48", +} + +var E_DefaultFixed64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*uint64)(nil), + Field: 210, + Name: "testdata.default_fixed64", + Tag: "fixed64,210,opt,name=default_fixed64,def=49", +} + +var E_DefaultSfixed32 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int32)(nil), + Field: 211, + Name: "testdata.default_sfixed32", + Tag: "fixed32,211,opt,name=default_sfixed32,def=50", +} + +var E_DefaultSfixed64 = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*int64)(nil), + Field: 212, + Name: "testdata.default_sfixed64", + Tag: "fixed64,212,opt,name=default_sfixed64,def=51", +} + +var E_DefaultBool = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*bool)(nil), + Field: 213, + Name: "testdata.default_bool", + Tag: "varint,213,opt,name=default_bool,def=1", +} + +var E_DefaultString = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*string)(nil), + Field: 214, + Name: "testdata.default_string", + Tag: "bytes,214,opt,name=default_string,def=Hello, string", +} + +var E_DefaultBytes = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: ([]byte)(nil), + Field: 215, + Name: "testdata.default_bytes", + Tag: "bytes,215,opt,name=default_bytes,def=Hello, bytes", +} + +var E_DefaultEnum = &proto.ExtensionDesc{ + ExtendedType: (*DefaultsMessage)(nil), + ExtensionType: (*DefaultsMessage_DefaultsEnum)(nil), + Field: 216, + Name: "testdata.default_enum", + Tag: "varint,216,opt,name=default_enum,enum=testdata.DefaultsMessage_DefaultsEnum,def=1", +} + +var E_X201 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 201, + Name: "testdata.x201", + Tag: "bytes,201,opt,name=x201", +} + +var E_X202 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 202, + Name: "testdata.x202", + Tag: "bytes,202,opt,name=x202", +} + +var E_X203 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 203, + Name: "testdata.x203", + Tag: "bytes,203,opt,name=x203", +} + +var E_X204 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 204, + Name: "testdata.x204", + Tag: "bytes,204,opt,name=x204", +} + +var E_X205 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 205, + Name: "testdata.x205", + Tag: "bytes,205,opt,name=x205", +} + +var E_X206 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 206, + Name: "testdata.x206", + Tag: "bytes,206,opt,name=x206", +} + +var E_X207 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 207, + Name: "testdata.x207", + Tag: "bytes,207,opt,name=x207", +} + +var E_X208 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 208, + Name: "testdata.x208", + Tag: "bytes,208,opt,name=x208", +} + +var E_X209 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 209, + Name: "testdata.x209", + Tag: "bytes,209,opt,name=x209", +} + +var E_X210 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 210, + Name: "testdata.x210", + Tag: "bytes,210,opt,name=x210", +} + +var E_X211 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 211, + Name: "testdata.x211", + Tag: "bytes,211,opt,name=x211", +} + +var E_X212 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 212, + Name: "testdata.x212", + Tag: "bytes,212,opt,name=x212", +} + +var E_X213 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 213, + Name: "testdata.x213", + Tag: "bytes,213,opt,name=x213", +} + +var E_X214 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 214, + Name: "testdata.x214", + Tag: "bytes,214,opt,name=x214", +} + +var E_X215 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 215, + Name: "testdata.x215", + Tag: "bytes,215,opt,name=x215", +} + +var E_X216 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 216, + Name: "testdata.x216", + Tag: "bytes,216,opt,name=x216", +} + +var E_X217 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 217, + Name: "testdata.x217", + Tag: "bytes,217,opt,name=x217", +} + +var E_X218 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 218, + Name: "testdata.x218", + Tag: "bytes,218,opt,name=x218", +} + +var E_X219 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 219, + Name: "testdata.x219", + Tag: "bytes,219,opt,name=x219", +} + +var E_X220 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 220, + Name: "testdata.x220", + Tag: "bytes,220,opt,name=x220", +} + +var E_X221 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 221, + Name: "testdata.x221", + Tag: "bytes,221,opt,name=x221", +} + +var E_X222 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 222, + Name: "testdata.x222", + Tag: "bytes,222,opt,name=x222", +} + +var E_X223 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 223, + Name: "testdata.x223", + Tag: "bytes,223,opt,name=x223", +} + +var E_X224 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 224, + Name: "testdata.x224", + Tag: "bytes,224,opt,name=x224", +} + +var E_X225 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 225, + Name: "testdata.x225", + Tag: "bytes,225,opt,name=x225", +} + +var E_X226 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 226, + Name: "testdata.x226", + Tag: "bytes,226,opt,name=x226", +} + +var E_X227 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 227, + Name: "testdata.x227", + Tag: "bytes,227,opt,name=x227", +} + +var E_X228 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 228, + Name: "testdata.x228", + Tag: "bytes,228,opt,name=x228", +} + +var E_X229 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 229, + Name: "testdata.x229", + Tag: "bytes,229,opt,name=x229", +} + +var E_X230 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 230, + Name: "testdata.x230", + Tag: "bytes,230,opt,name=x230", +} + +var E_X231 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 231, + Name: "testdata.x231", + Tag: "bytes,231,opt,name=x231", +} + +var E_X232 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 232, + Name: "testdata.x232", + Tag: "bytes,232,opt,name=x232", +} + +var E_X233 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 233, + Name: "testdata.x233", + Tag: "bytes,233,opt,name=x233", +} + +var E_X234 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 234, + Name: "testdata.x234", + Tag: "bytes,234,opt,name=x234", +} + +var E_X235 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 235, + Name: "testdata.x235", + Tag: "bytes,235,opt,name=x235", +} + +var E_X236 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 236, + Name: "testdata.x236", + Tag: "bytes,236,opt,name=x236", +} + +var E_X237 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 237, + Name: "testdata.x237", + Tag: "bytes,237,opt,name=x237", +} + +var E_X238 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 238, + Name: "testdata.x238", + Tag: "bytes,238,opt,name=x238", +} + +var E_X239 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 239, + Name: "testdata.x239", + Tag: "bytes,239,opt,name=x239", +} + +var E_X240 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 240, + Name: "testdata.x240", + Tag: "bytes,240,opt,name=x240", +} + +var E_X241 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 241, + Name: "testdata.x241", + Tag: "bytes,241,opt,name=x241", +} + +var E_X242 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 242, + Name: "testdata.x242", + Tag: "bytes,242,opt,name=x242", +} + +var E_X243 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 243, + Name: "testdata.x243", + Tag: "bytes,243,opt,name=x243", +} + +var E_X244 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 244, + Name: "testdata.x244", + Tag: "bytes,244,opt,name=x244", +} + +var E_X245 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 245, + Name: "testdata.x245", + Tag: "bytes,245,opt,name=x245", +} + +var E_X246 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 246, + Name: "testdata.x246", + Tag: "bytes,246,opt,name=x246", +} + +var E_X247 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 247, + Name: "testdata.x247", + Tag: "bytes,247,opt,name=x247", +} + +var E_X248 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 248, + Name: "testdata.x248", + Tag: "bytes,248,opt,name=x248", +} + +var E_X249 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 249, + Name: "testdata.x249", + Tag: "bytes,249,opt,name=x249", +} + +var E_X250 = &proto.ExtensionDesc{ + ExtendedType: (*MyMessageSet)(nil), + ExtensionType: (*Empty)(nil), + Field: 250, + Name: "testdata.x250", + Tag: "bytes,250,opt,name=x250", +} + +func init() { + proto.RegisterEnum("testdata.FOO", FOO_name, FOO_value) + proto.RegisterEnum("testdata.GoTest_KIND", GoTest_KIND_name, GoTest_KIND_value) + proto.RegisterEnum("testdata.MyMessage_Color", MyMessage_Color_name, MyMessage_Color_value) + proto.RegisterEnum("testdata.DefaultsMessage_DefaultsEnum", DefaultsMessage_DefaultsEnum_name, DefaultsMessage_DefaultsEnum_value) + proto.RegisterEnum("testdata.Defaults_Color", Defaults_Color_name, Defaults_Color_value) + proto.RegisterEnum("testdata.RepeatedEnum_Color", RepeatedEnum_Color_name, RepeatedEnum_Color_value) + proto.RegisterExtension(E_Ext_More) + proto.RegisterExtension(E_Ext_Text) + proto.RegisterExtension(E_Ext_Number) + proto.RegisterExtension(E_Greeting) + proto.RegisterExtension(E_NoDefaultDouble) + proto.RegisterExtension(E_NoDefaultFloat) + proto.RegisterExtension(E_NoDefaultInt32) + proto.RegisterExtension(E_NoDefaultInt64) + proto.RegisterExtension(E_NoDefaultUint32) + proto.RegisterExtension(E_NoDefaultUint64) + proto.RegisterExtension(E_NoDefaultSint32) + proto.RegisterExtension(E_NoDefaultSint64) + proto.RegisterExtension(E_NoDefaultFixed32) + proto.RegisterExtension(E_NoDefaultFixed64) + proto.RegisterExtension(E_NoDefaultSfixed32) + proto.RegisterExtension(E_NoDefaultSfixed64) + proto.RegisterExtension(E_NoDefaultBool) + proto.RegisterExtension(E_NoDefaultString) + proto.RegisterExtension(E_NoDefaultBytes) + proto.RegisterExtension(E_NoDefaultEnum) + proto.RegisterExtension(E_DefaultDouble) + proto.RegisterExtension(E_DefaultFloat) + proto.RegisterExtension(E_DefaultInt32) + proto.RegisterExtension(E_DefaultInt64) + proto.RegisterExtension(E_DefaultUint32) + proto.RegisterExtension(E_DefaultUint64) + proto.RegisterExtension(E_DefaultSint32) + proto.RegisterExtension(E_DefaultSint64) + proto.RegisterExtension(E_DefaultFixed32) + proto.RegisterExtension(E_DefaultFixed64) + proto.RegisterExtension(E_DefaultSfixed32) + proto.RegisterExtension(E_DefaultSfixed64) + proto.RegisterExtension(E_DefaultBool) + proto.RegisterExtension(E_DefaultString) + proto.RegisterExtension(E_DefaultBytes) + proto.RegisterExtension(E_DefaultEnum) + proto.RegisterExtension(E_X201) + proto.RegisterExtension(E_X202) + proto.RegisterExtension(E_X203) + proto.RegisterExtension(E_X204) + proto.RegisterExtension(E_X205) + proto.RegisterExtension(E_X206) + proto.RegisterExtension(E_X207) + proto.RegisterExtension(E_X208) + proto.RegisterExtension(E_X209) + proto.RegisterExtension(E_X210) + proto.RegisterExtension(E_X211) + proto.RegisterExtension(E_X212) + proto.RegisterExtension(E_X213) + proto.RegisterExtension(E_X214) + proto.RegisterExtension(E_X215) + proto.RegisterExtension(E_X216) + proto.RegisterExtension(E_X217) + proto.RegisterExtension(E_X218) + proto.RegisterExtension(E_X219) + proto.RegisterExtension(E_X220) + proto.RegisterExtension(E_X221) + proto.RegisterExtension(E_X222) + proto.RegisterExtension(E_X223) + proto.RegisterExtension(E_X224) + proto.RegisterExtension(E_X225) + proto.RegisterExtension(E_X226) + proto.RegisterExtension(E_X227) + proto.RegisterExtension(E_X228) + proto.RegisterExtension(E_X229) + proto.RegisterExtension(E_X230) + proto.RegisterExtension(E_X231) + proto.RegisterExtension(E_X232) + proto.RegisterExtension(E_X233) + proto.RegisterExtension(E_X234) + proto.RegisterExtension(E_X235) + proto.RegisterExtension(E_X236) + proto.RegisterExtension(E_X237) + proto.RegisterExtension(E_X238) + proto.RegisterExtension(E_X239) + proto.RegisterExtension(E_X240) + proto.RegisterExtension(E_X241) + proto.RegisterExtension(E_X242) + proto.RegisterExtension(E_X243) + proto.RegisterExtension(E_X244) + proto.RegisterExtension(E_X245) + proto.RegisterExtension(E_X246) + proto.RegisterExtension(E_X247) + proto.RegisterExtension(E_X248) + proto.RegisterExtension(E_X249) + proto.RegisterExtension(E_X250) +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/testdata/test.proto b/Godeps/_workspace/src/github.com/golang/protobuf/proto/testdata/test.proto new file mode 100644 index 0000000000000..440dba38dd3bd --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/testdata/test.proto @@ -0,0 +1,480 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// A feature-rich test file for the protocol compiler and libraries. + +syntax = "proto2"; + +package testdata; + +enum FOO { FOO1 = 1; }; + +message GoEnum { + required FOO foo = 1; +} + +message GoTestField { + required string Label = 1; + required string Type = 2; +} + +message GoTest { + // An enum, for completeness. + enum KIND { + VOID = 0; + + // Basic types + BOOL = 1; + BYTES = 2; + FINGERPRINT = 3; + FLOAT = 4; + INT = 5; + STRING = 6; + TIME = 7; + + // Groupings + TUPLE = 8; + ARRAY = 9; + MAP = 10; + + // Table types + TABLE = 11; + + // Functions + FUNCTION = 12; // last tag + }; + + // Some typical parameters + required KIND Kind = 1; + optional string Table = 2; + optional int32 Param = 3; + + // Required, repeated and optional foreign fields. + required GoTestField RequiredField = 4; + repeated GoTestField RepeatedField = 5; + optional GoTestField OptionalField = 6; + + // Required fields of all basic types + required bool F_Bool_required = 10; + required int32 F_Int32_required = 11; + required int64 F_Int64_required = 12; + required fixed32 F_Fixed32_required = 13; + required fixed64 F_Fixed64_required = 14; + required uint32 F_Uint32_required = 15; + required uint64 F_Uint64_required = 16; + required float F_Float_required = 17; + required double F_Double_required = 18; + required string F_String_required = 19; + required bytes F_Bytes_required = 101; + required sint32 F_Sint32_required = 102; + required sint64 F_Sint64_required = 103; + + // Repeated fields of all basic types + repeated bool F_Bool_repeated = 20; + repeated int32 F_Int32_repeated = 21; + repeated int64 F_Int64_repeated = 22; + repeated fixed32 F_Fixed32_repeated = 23; + repeated fixed64 F_Fixed64_repeated = 24; + repeated uint32 F_Uint32_repeated = 25; + repeated uint64 F_Uint64_repeated = 26; + repeated float F_Float_repeated = 27; + repeated double F_Double_repeated = 28; + repeated string F_String_repeated = 29; + repeated bytes F_Bytes_repeated = 201; + repeated sint32 F_Sint32_repeated = 202; + repeated sint64 F_Sint64_repeated = 203; + + // Optional fields of all basic types + optional bool F_Bool_optional = 30; + optional int32 F_Int32_optional = 31; + optional int64 F_Int64_optional = 32; + optional fixed32 F_Fixed32_optional = 33; + optional fixed64 F_Fixed64_optional = 34; + optional uint32 F_Uint32_optional = 35; + optional uint64 F_Uint64_optional = 36; + optional float F_Float_optional = 37; + optional double F_Double_optional = 38; + optional string F_String_optional = 39; + optional bytes F_Bytes_optional = 301; + optional sint32 F_Sint32_optional = 302; + optional sint64 F_Sint64_optional = 303; + + // Default-valued fields of all basic types + optional bool F_Bool_defaulted = 40 [default=true]; + optional int32 F_Int32_defaulted = 41 [default=32]; + optional int64 F_Int64_defaulted = 42 [default=64]; + optional fixed32 F_Fixed32_defaulted = 43 [default=320]; + optional fixed64 F_Fixed64_defaulted = 44 [default=640]; + optional uint32 F_Uint32_defaulted = 45 [default=3200]; + optional uint64 F_Uint64_defaulted = 46 [default=6400]; + optional float F_Float_defaulted = 47 [default=314159.]; + optional double F_Double_defaulted = 48 [default=271828.]; + optional string F_String_defaulted = 49 [default="hello, \"world!\"\n"]; + optional bytes F_Bytes_defaulted = 401 [default="Bignose"]; + optional sint32 F_Sint32_defaulted = 402 [default = -32]; + optional sint64 F_Sint64_defaulted = 403 [default = -64]; + + // Packed repeated fields (no string or bytes). + repeated bool F_Bool_repeated_packed = 50 [packed=true]; + repeated int32 F_Int32_repeated_packed = 51 [packed=true]; + repeated int64 F_Int64_repeated_packed = 52 [packed=true]; + repeated fixed32 F_Fixed32_repeated_packed = 53 [packed=true]; + repeated fixed64 F_Fixed64_repeated_packed = 54 [packed=true]; + repeated uint32 F_Uint32_repeated_packed = 55 [packed=true]; + repeated uint64 F_Uint64_repeated_packed = 56 [packed=true]; + repeated float F_Float_repeated_packed = 57 [packed=true]; + repeated double F_Double_repeated_packed = 58 [packed=true]; + repeated sint32 F_Sint32_repeated_packed = 502 [packed=true]; + repeated sint64 F_Sint64_repeated_packed = 503 [packed=true]; + + // Required, repeated, and optional groups. + required group RequiredGroup = 70 { + required string RequiredField = 71; + }; + + repeated group RepeatedGroup = 80 { + required string RequiredField = 81; + }; + + optional group OptionalGroup = 90 { + required string RequiredField = 91; + }; +} + +// For testing skipping of unrecognized fields. +// Numbers are all big, larger than tag numbers in GoTestField, +// the message used in the corresponding test. +message GoSkipTest { + required int32 skip_int32 = 11; + required fixed32 skip_fixed32 = 12; + required fixed64 skip_fixed64 = 13; + required string skip_string = 14; + required group SkipGroup = 15 { + required int32 group_int32 = 16; + required string group_string = 17; + } +} + +// For testing packed/non-packed decoder switching. +// A serialized instance of one should be deserializable as the other. +message NonPackedTest { + repeated int32 a = 1; +} + +message PackedTest { + repeated int32 b = 1 [packed=true]; +} + +message MaxTag { + // Maximum possible tag number. + optional string last_field = 536870911; +} + +message OldMessage { + message Nested { + optional string name = 1; + } + optional Nested nested = 1; + + optional int32 num = 2; +} + +// NewMessage is wire compatible with OldMessage; +// imagine it as a future version. +message NewMessage { + message Nested { + optional string name = 1; + optional string food_group = 2; + } + optional Nested nested = 1; + + // This is an int32 in OldMessage. + optional int64 num = 2; +} + +// Smaller tests for ASCII formatting. + +message InnerMessage { + required string host = 1; + optional int32 port = 2 [default=4000]; + optional bool connected = 3; +} + +message OtherMessage { + optional int64 key = 1; + optional bytes value = 2; + optional float weight = 3; + optional InnerMessage inner = 4; +} + +message MyMessage { + required int32 count = 1; + optional string name = 2; + optional string quote = 3; + repeated string pet = 4; + optional InnerMessage inner = 5; + repeated OtherMessage others = 6; + repeated InnerMessage rep_inner = 12; + + enum Color { + RED = 0; + GREEN = 1; + BLUE = 2; + }; + optional Color bikeshed = 7; + + optional group SomeGroup = 8 { + optional int32 group_field = 9; + } + + // This field becomes [][]byte in the generated code. + repeated bytes rep_bytes = 10; + + optional double bigfloat = 11; + + extensions 100 to max; +} + +message Ext { + extend MyMessage { + optional Ext more = 103; + optional string text = 104; + optional int32 number = 105; + } + + optional string data = 1; +} + +extend MyMessage { + repeated string greeting = 106; +} + +message DefaultsMessage { + enum DefaultsEnum { + ZERO = 0; + ONE = 1; + TWO = 2; + }; + extensions 100 to max; +} + +extend DefaultsMessage { + optional double no_default_double = 101; + optional float no_default_float = 102; + optional int32 no_default_int32 = 103; + optional int64 no_default_int64 = 104; + optional uint32 no_default_uint32 = 105; + optional uint64 no_default_uint64 = 106; + optional sint32 no_default_sint32 = 107; + optional sint64 no_default_sint64 = 108; + optional fixed32 no_default_fixed32 = 109; + optional fixed64 no_default_fixed64 = 110; + optional sfixed32 no_default_sfixed32 = 111; + optional sfixed64 no_default_sfixed64 = 112; + optional bool no_default_bool = 113; + optional string no_default_string = 114; + optional bytes no_default_bytes = 115; + optional DefaultsMessage.DefaultsEnum no_default_enum = 116; + + optional double default_double = 201 [default = 3.1415]; + optional float default_float = 202 [default = 3.14]; + optional int32 default_int32 = 203 [default = 42]; + optional int64 default_int64 = 204 [default = 43]; + optional uint32 default_uint32 = 205 [default = 44]; + optional uint64 default_uint64 = 206 [default = 45]; + optional sint32 default_sint32 = 207 [default = 46]; + optional sint64 default_sint64 = 208 [default = 47]; + optional fixed32 default_fixed32 = 209 [default = 48]; + optional fixed64 default_fixed64 = 210 [default = 49]; + optional sfixed32 default_sfixed32 = 211 [default = 50]; + optional sfixed64 default_sfixed64 = 212 [default = 51]; + optional bool default_bool = 213 [default = true]; + optional string default_string = 214 [default = "Hello, string"]; + optional bytes default_bytes = 215 [default = "Hello, bytes"]; + optional DefaultsMessage.DefaultsEnum default_enum = 216 [default = ONE]; +} + +message MyMessageSet { + option message_set_wire_format = true; + extensions 100 to max; +} + +message Empty { +} + +extend MyMessageSet { + optional Empty x201 = 201; + optional Empty x202 = 202; + optional Empty x203 = 203; + optional Empty x204 = 204; + optional Empty x205 = 205; + optional Empty x206 = 206; + optional Empty x207 = 207; + optional Empty x208 = 208; + optional Empty x209 = 209; + optional Empty x210 = 210; + optional Empty x211 = 211; + optional Empty x212 = 212; + optional Empty x213 = 213; + optional Empty x214 = 214; + optional Empty x215 = 215; + optional Empty x216 = 216; + optional Empty x217 = 217; + optional Empty x218 = 218; + optional Empty x219 = 219; + optional Empty x220 = 220; + optional Empty x221 = 221; + optional Empty x222 = 222; + optional Empty x223 = 223; + optional Empty x224 = 224; + optional Empty x225 = 225; + optional Empty x226 = 226; + optional Empty x227 = 227; + optional Empty x228 = 228; + optional Empty x229 = 229; + optional Empty x230 = 230; + optional Empty x231 = 231; + optional Empty x232 = 232; + optional Empty x233 = 233; + optional Empty x234 = 234; + optional Empty x235 = 235; + optional Empty x236 = 236; + optional Empty x237 = 237; + optional Empty x238 = 238; + optional Empty x239 = 239; + optional Empty x240 = 240; + optional Empty x241 = 241; + optional Empty x242 = 242; + optional Empty x243 = 243; + optional Empty x244 = 244; + optional Empty x245 = 245; + optional Empty x246 = 246; + optional Empty x247 = 247; + optional Empty x248 = 248; + optional Empty x249 = 249; + optional Empty x250 = 250; +} + +message MessageList { + repeated group Message = 1 { + required string name = 2; + required int32 count = 3; + } +} + +message Strings { + optional string string_field = 1; + optional bytes bytes_field = 2; +} + +message Defaults { + enum Color { + RED = 0; + GREEN = 1; + BLUE = 2; + } + + // Default-valued fields of all basic types. + // Same as GoTest, but copied here to make testing easier. + optional bool F_Bool = 1 [default=true]; + optional int32 F_Int32 = 2 [default=32]; + optional int64 F_Int64 = 3 [default=64]; + optional fixed32 F_Fixed32 = 4 [default=320]; + optional fixed64 F_Fixed64 = 5 [default=640]; + optional uint32 F_Uint32 = 6 [default=3200]; + optional uint64 F_Uint64 = 7 [default=6400]; + optional float F_Float = 8 [default=314159.]; + optional double F_Double = 9 [default=271828.]; + optional string F_String = 10 [default="hello, \"world!\"\n"]; + optional bytes F_Bytes = 11 [default="Bignose"]; + optional sint32 F_Sint32 = 12 [default=-32]; + optional sint64 F_Sint64 = 13 [default=-64]; + optional Color F_Enum = 14 [default=GREEN]; + + // More fields with crazy defaults. + optional float F_Pinf = 15 [default=inf]; + optional float F_Ninf = 16 [default=-inf]; + optional float F_Nan = 17 [default=nan]; + + // Sub-message. + optional SubDefaults sub = 18; + + // Redundant but explicit defaults. + optional string str_zero = 19 [default=""]; +} + +message SubDefaults { + optional int64 n = 1 [default=7]; +} + +message RepeatedEnum { + enum Color { + RED = 1; + } + repeated Color color = 1; +} + +message MoreRepeated { + repeated bool bools = 1; + repeated bool bools_packed = 2 [packed=true]; + repeated int32 ints = 3; + repeated int32 ints_packed = 4 [packed=true]; + repeated int64 int64s_packed = 7 [packed=true]; + repeated string strings = 5; + repeated fixed32 fixeds = 6; +} + +// GroupOld and GroupNew have the same wire format. +// GroupNew has a new field inside a group. + +message GroupOld { + optional group G = 101 { + optional int32 x = 2; + } +} + +message GroupNew { + optional group G = 101 { + optional int32 x = 2; + optional int32 y = 3; + } +} + +message FloatingPoint { + required double f = 1; +} + +message MessageWithMap { + map name_mapping = 1; + map msg_mapping = 2; + map byte_mapping = 3; + map str_to_str = 4; +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/text.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/text.go new file mode 100644 index 0000000000000..f3db2cf5e2375 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/text.go @@ -0,0 +1,769 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +// Functions for writing the text protocol buffer format. + +import ( + "bufio" + "bytes" + "encoding" + "fmt" + "io" + "log" + "math" + "reflect" + "sort" + "strings" +) + +var ( + newline = []byte("\n") + spaces = []byte(" ") + gtNewline = []byte(">\n") + endBraceNewline = []byte("}\n") + backslashN = []byte{'\\', 'n'} + backslashR = []byte{'\\', 'r'} + backslashT = []byte{'\\', 't'} + backslashDQ = []byte{'\\', '"'} + backslashBS = []byte{'\\', '\\'} + posInf = []byte("inf") + negInf = []byte("-inf") + nan = []byte("nan") +) + +type writer interface { + io.Writer + WriteByte(byte) error +} + +// textWriter is an io.Writer that tracks its indentation level. +type textWriter struct { + ind int + complete bool // if the current position is a complete line + compact bool // whether to write out as a one-liner + w writer +} + +func (w *textWriter) WriteString(s string) (n int, err error) { + if !strings.Contains(s, "\n") { + if !w.compact && w.complete { + w.writeIndent() + } + w.complete = false + return io.WriteString(w.w, s) + } + // WriteString is typically called without newlines, so this + // codepath and its copy are rare. We copy to avoid + // duplicating all of Write's logic here. + return w.Write([]byte(s)) +} + +func (w *textWriter) Write(p []byte) (n int, err error) { + newlines := bytes.Count(p, newline) + if newlines == 0 { + if !w.compact && w.complete { + w.writeIndent() + } + n, err = w.w.Write(p) + w.complete = false + return n, err + } + + frags := bytes.SplitN(p, newline, newlines+1) + if w.compact { + for i, frag := range frags { + if i > 0 { + if err := w.w.WriteByte(' '); err != nil { + return n, err + } + n++ + } + nn, err := w.w.Write(frag) + n += nn + if err != nil { + return n, err + } + } + return n, nil + } + + for i, frag := range frags { + if w.complete { + w.writeIndent() + } + nn, err := w.w.Write(frag) + n += nn + if err != nil { + return n, err + } + if i+1 < len(frags) { + if err := w.w.WriteByte('\n'); err != nil { + return n, err + } + n++ + } + } + w.complete = len(frags[len(frags)-1]) == 0 + return n, nil +} + +func (w *textWriter) WriteByte(c byte) error { + if w.compact && c == '\n' { + c = ' ' + } + if !w.compact && w.complete { + w.writeIndent() + } + err := w.w.WriteByte(c) + w.complete = c == '\n' + return err +} + +func (w *textWriter) indent() { w.ind++ } + +func (w *textWriter) unindent() { + if w.ind == 0 { + log.Printf("proto: textWriter unindented too far") + return + } + w.ind-- +} + +func writeName(w *textWriter, props *Properties) error { + if _, err := w.WriteString(props.OrigName); err != nil { + return err + } + if props.Wire != "group" { + return w.WriteByte(':') + } + return nil +} + +var ( + messageSetType = reflect.TypeOf((*MessageSet)(nil)).Elem() +) + +// raw is the interface satisfied by RawMessage. +type raw interface { + Bytes() []byte +} + +func writeStruct(w *textWriter, sv reflect.Value) error { + if sv.Type() == messageSetType { + return writeMessageSet(w, sv.Addr().Interface().(*MessageSet)) + } + + st := sv.Type() + sprops := GetProperties(st) + for i := 0; i < sv.NumField(); i++ { + fv := sv.Field(i) + props := sprops.Prop[i] + name := st.Field(i).Name + + if strings.HasPrefix(name, "XXX_") { + // There are two XXX_ fields: + // XXX_unrecognized []byte + // XXX_extensions map[int32]proto.Extension + // The first is handled here; + // the second is handled at the bottom of this function. + if name == "XXX_unrecognized" && !fv.IsNil() { + if err := writeUnknownStruct(w, fv.Interface().([]byte)); err != nil { + return err + } + } + continue + } + if fv.Kind() == reflect.Ptr && fv.IsNil() { + // Field not filled in. This could be an optional field or + // a required field that wasn't filled in. Either way, there + // isn't anything we can show for it. + continue + } + if fv.Kind() == reflect.Slice && fv.IsNil() { + // Repeated field that is empty, or a bytes field that is unused. + continue + } + + if props.Repeated && fv.Kind() == reflect.Slice { + // Repeated field. + for j := 0; j < fv.Len(); j++ { + if err := writeName(w, props); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte(' '); err != nil { + return err + } + } + v := fv.Index(j) + if v.Kind() == reflect.Ptr && v.IsNil() { + // A nil message in a repeated field is not valid, + // but we can handle that more gracefully than panicking. + if _, err := w.Write([]byte("\n")); err != nil { + return err + } + continue + } + if err := writeAny(w, v, props); err != nil { + return err + } + if err := w.WriteByte('\n'); err != nil { + return err + } + } + continue + } + if fv.Kind() == reflect.Map { + // Map fields are rendered as a repeated struct with key/value fields. + keys := fv.MapKeys() + sort.Sort(mapKeys(keys)) + for _, key := range keys { + val := fv.MapIndex(key) + if err := writeName(w, props); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte(' '); err != nil { + return err + } + } + // open struct + if err := w.WriteByte('<'); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte('\n'); err != nil { + return err + } + } + w.indent() + // key + if _, err := w.WriteString("key:"); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte(' '); err != nil { + return err + } + } + if err := writeAny(w, key, props.mkeyprop); err != nil { + return err + } + if err := w.WriteByte('\n'); err != nil { + return err + } + // nil values aren't legal, but we can avoid panicking because of them. + if val.Kind() != reflect.Ptr || !val.IsNil() { + // value + if _, err := w.WriteString("value:"); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte(' '); err != nil { + return err + } + } + if err := writeAny(w, val, props.mvalprop); err != nil { + return err + } + if err := w.WriteByte('\n'); err != nil { + return err + } + } + // close struct + w.unindent() + if err := w.WriteByte('>'); err != nil { + return err + } + if err := w.WriteByte('\n'); err != nil { + return err + } + } + continue + } + if props.proto3 && fv.Kind() == reflect.Slice && fv.Len() == 0 { + // empty bytes field + continue + } + if fv.Kind() != reflect.Ptr && fv.Kind() != reflect.Slice { + // proto3 non-repeated scalar field; skip if zero value + if isProto3Zero(fv) { + continue + } + } + + if err := writeName(w, props); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte(' '); err != nil { + return err + } + } + if b, ok := fv.Interface().(raw); ok { + if err := writeRaw(w, b.Bytes()); err != nil { + return err + } + continue + } + + // Enums have a String method, so writeAny will work fine. + if err := writeAny(w, fv, props); err != nil { + return err + } + + if err := w.WriteByte('\n'); err != nil { + return err + } + } + + // Extensions (the XXX_extensions field). + pv := sv.Addr() + if pv.Type().Implements(extendableProtoType) { + if err := writeExtensions(w, pv); err != nil { + return err + } + } + + return nil +} + +// writeRaw writes an uninterpreted raw message. +func writeRaw(w *textWriter, b []byte) error { + if err := w.WriteByte('<'); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte('\n'); err != nil { + return err + } + } + w.indent() + if err := writeUnknownStruct(w, b); err != nil { + return err + } + w.unindent() + if err := w.WriteByte('>'); err != nil { + return err + } + return nil +} + +// writeAny writes an arbitrary field. +func writeAny(w *textWriter, v reflect.Value, props *Properties) error { + v = reflect.Indirect(v) + + // Floats have special cases. + if v.Kind() == reflect.Float32 || v.Kind() == reflect.Float64 { + x := v.Float() + var b []byte + switch { + case math.IsInf(x, 1): + b = posInf + case math.IsInf(x, -1): + b = negInf + case math.IsNaN(x): + b = nan + } + if b != nil { + _, err := w.Write(b) + return err + } + // Other values are handled below. + } + + // We don't attempt to serialise every possible value type; only those + // that can occur in protocol buffers. + switch v.Kind() { + case reflect.Slice: + // Should only be a []byte; repeated fields are handled in writeStruct. + if err := writeString(w, string(v.Interface().([]byte))); err != nil { + return err + } + case reflect.String: + if err := writeString(w, v.String()); err != nil { + return err + } + case reflect.Struct: + // Required/optional group/message. + var bra, ket byte = '<', '>' + if props != nil && props.Wire == "group" { + bra, ket = '{', '}' + } + if err := w.WriteByte(bra); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte('\n'); err != nil { + return err + } + } + w.indent() + if tm, ok := v.Interface().(encoding.TextMarshaler); ok { + text, err := tm.MarshalText() + if err != nil { + return err + } + if _, err = w.Write(text); err != nil { + return err + } + } else if err := writeStruct(w, v); err != nil { + return err + } + w.unindent() + if err := w.WriteByte(ket); err != nil { + return err + } + default: + _, err := fmt.Fprint(w, v.Interface()) + return err + } + return nil +} + +// equivalent to C's isprint. +func isprint(c byte) bool { + return c >= 0x20 && c < 0x7f +} + +// writeString writes a string in the protocol buffer text format. +// It is similar to strconv.Quote except we don't use Go escape sequences, +// we treat the string as a byte sequence, and we use octal escapes. +// These differences are to maintain interoperability with the other +// languages' implementations of the text format. +func writeString(w *textWriter, s string) error { + // use WriteByte here to get any needed indent + if err := w.WriteByte('"'); err != nil { + return err + } + // Loop over the bytes, not the runes. + for i := 0; i < len(s); i++ { + var err error + // Divergence from C++: we don't escape apostrophes. + // There's no need to escape them, and the C++ parser + // copes with a naked apostrophe. + switch c := s[i]; c { + case '\n': + _, err = w.w.Write(backslashN) + case '\r': + _, err = w.w.Write(backslashR) + case '\t': + _, err = w.w.Write(backslashT) + case '"': + _, err = w.w.Write(backslashDQ) + case '\\': + _, err = w.w.Write(backslashBS) + default: + if isprint(c) { + err = w.w.WriteByte(c) + } else { + _, err = fmt.Fprintf(w.w, "\\%03o", c) + } + } + if err != nil { + return err + } + } + return w.WriteByte('"') +} + +func writeMessageSet(w *textWriter, ms *MessageSet) error { + for _, item := range ms.Item { + id := *item.TypeId + if msd, ok := messageSetMap[id]; ok { + // Known message set type. + if _, err := fmt.Fprintf(w, "[%s]: <\n", msd.name); err != nil { + return err + } + w.indent() + + pb := reflect.New(msd.t.Elem()) + if err := Unmarshal(item.Message, pb.Interface().(Message)); err != nil { + if _, err := fmt.Fprintf(w, "/* bad message: %v */\n", err); err != nil { + return err + } + } else { + if err := writeStruct(w, pb.Elem()); err != nil { + return err + } + } + } else { + // Unknown type. + if _, err := fmt.Fprintf(w, "[%d]: <\n", id); err != nil { + return err + } + w.indent() + if err := writeUnknownStruct(w, item.Message); err != nil { + return err + } + } + w.unindent() + if _, err := w.Write(gtNewline); err != nil { + return err + } + } + return nil +} + +func writeUnknownStruct(w *textWriter, data []byte) (err error) { + if !w.compact { + if _, err := fmt.Fprintf(w, "/* %d unknown bytes */\n", len(data)); err != nil { + return err + } + } + b := NewBuffer(data) + for b.index < len(b.buf) { + x, err := b.DecodeVarint() + if err != nil { + _, err := fmt.Fprintf(w, "/* %v */\n", err) + return err + } + wire, tag := x&7, x>>3 + if wire == WireEndGroup { + w.unindent() + if _, err := w.Write(endBraceNewline); err != nil { + return err + } + continue + } + if _, err := fmt.Fprint(w, tag); err != nil { + return err + } + if wire != WireStartGroup { + if err := w.WriteByte(':'); err != nil { + return err + } + } + if !w.compact || wire == WireStartGroup { + if err := w.WriteByte(' '); err != nil { + return err + } + } + switch wire { + case WireBytes: + buf, e := b.DecodeRawBytes(false) + if e == nil { + _, err = fmt.Fprintf(w, "%q", buf) + } else { + _, err = fmt.Fprintf(w, "/* %v */", e) + } + case WireFixed32: + x, err = b.DecodeFixed32() + err = writeUnknownInt(w, x, err) + case WireFixed64: + x, err = b.DecodeFixed64() + err = writeUnknownInt(w, x, err) + case WireStartGroup: + err = w.WriteByte('{') + w.indent() + case WireVarint: + x, err = b.DecodeVarint() + err = writeUnknownInt(w, x, err) + default: + _, err = fmt.Fprintf(w, "/* unknown wire type %d */", wire) + } + if err != nil { + return err + } + if err = w.WriteByte('\n'); err != nil { + return err + } + } + return nil +} + +func writeUnknownInt(w *textWriter, x uint64, err error) error { + if err == nil { + _, err = fmt.Fprint(w, x) + } else { + _, err = fmt.Fprintf(w, "/* %v */", err) + } + return err +} + +type int32Slice []int32 + +func (s int32Slice) Len() int { return len(s) } +func (s int32Slice) Less(i, j int) bool { return s[i] < s[j] } +func (s int32Slice) Swap(i, j int) { s[i], s[j] = s[j], s[i] } + +// writeExtensions writes all the extensions in pv. +// pv is assumed to be a pointer to a protocol message struct that is extendable. +func writeExtensions(w *textWriter, pv reflect.Value) error { + emap := extensionMaps[pv.Type().Elem()] + ep := pv.Interface().(extendableProto) + + // Order the extensions by ID. + // This isn't strictly necessary, but it will give us + // canonical output, which will also make testing easier. + m := ep.ExtensionMap() + ids := make([]int32, 0, len(m)) + for id := range m { + ids = append(ids, id) + } + sort.Sort(int32Slice(ids)) + + for _, extNum := range ids { + ext := m[extNum] + var desc *ExtensionDesc + if emap != nil { + desc = emap[extNum] + } + if desc == nil { + // Unknown extension. + if err := writeUnknownStruct(w, ext.enc); err != nil { + return err + } + continue + } + + pb, err := GetExtension(ep, desc) + if err != nil { + return fmt.Errorf("failed getting extension: %v", err) + } + + // Repeated extensions will appear as a slice. + if !desc.repeated() { + if err := writeExtension(w, desc.Name, pb); err != nil { + return err + } + } else { + v := reflect.ValueOf(pb) + for i := 0; i < v.Len(); i++ { + if err := writeExtension(w, desc.Name, v.Index(i).Interface()); err != nil { + return err + } + } + } + } + return nil +} + +func writeExtension(w *textWriter, name string, pb interface{}) error { + if _, err := fmt.Fprintf(w, "[%s]:", name); err != nil { + return err + } + if !w.compact { + if err := w.WriteByte(' '); err != nil { + return err + } + } + if err := writeAny(w, reflect.ValueOf(pb), nil); err != nil { + return err + } + if err := w.WriteByte('\n'); err != nil { + return err + } + return nil +} + +func (w *textWriter) writeIndent() { + if !w.complete { + return + } + remain := w.ind * 2 + for remain > 0 { + n := remain + if n > len(spaces) { + n = len(spaces) + } + w.w.Write(spaces[:n]) + remain -= n + } + w.complete = false +} + +func marshalText(w io.Writer, pb Message, compact bool) error { + val := reflect.ValueOf(pb) + if pb == nil || val.IsNil() { + w.Write([]byte("")) + return nil + } + var bw *bufio.Writer + ww, ok := w.(writer) + if !ok { + bw = bufio.NewWriter(w) + ww = bw + } + aw := &textWriter{ + w: ww, + complete: true, + compact: compact, + } + + if tm, ok := pb.(encoding.TextMarshaler); ok { + text, err := tm.MarshalText() + if err != nil { + return err + } + if _, err = aw.Write(text); err != nil { + return err + } + if bw != nil { + return bw.Flush() + } + return nil + } + // Dereference the received pointer so we don't have outer < and >. + v := reflect.Indirect(val) + if err := writeStruct(aw, v); err != nil { + return err + } + if bw != nil { + return bw.Flush() + } + return nil +} + +// MarshalText writes a given protocol buffer in text format. +// The only errors returned are from w. +func MarshalText(w io.Writer, pb Message) error { + return marshalText(w, pb, false) +} + +// MarshalTextString is the same as MarshalText, but returns the string directly. +func MarshalTextString(pb Message) string { + var buf bytes.Buffer + marshalText(&buf, pb, false) + return buf.String() +} + +// CompactText writes a given protocol buffer in compact text format (one line). +func CompactText(w io.Writer, pb Message) error { return marshalText(w, pb, true) } + +// CompactTextString is the same as CompactText, but returns the string directly. +func CompactTextString(pb Message) string { + var buf bytes.Buffer + marshalText(&buf, pb, true) + return buf.String() +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/text_parser.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/text_parser.go new file mode 100644 index 0000000000000..7d0c757198617 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/text_parser.go @@ -0,0 +1,772 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto + +// Functions for parsing the Text protocol buffer format. +// TODO: message sets. + +import ( + "encoding" + "errors" + "fmt" + "reflect" + "strconv" + "strings" + "unicode/utf8" +) + +type ParseError struct { + Message string + Line int // 1-based line number + Offset int // 0-based byte offset from start of input +} + +func (p *ParseError) Error() string { + if p.Line == 1 { + // show offset only for first line + return fmt.Sprintf("line 1.%d: %v", p.Offset, p.Message) + } + return fmt.Sprintf("line %d: %v", p.Line, p.Message) +} + +type token struct { + value string + err *ParseError + line int // line number + offset int // byte number from start of input, not start of line + unquoted string // the unquoted version of value, if it was a quoted string +} + +func (t *token) String() string { + if t.err == nil { + return fmt.Sprintf("%q (line=%d, offset=%d)", t.value, t.line, t.offset) + } + return fmt.Sprintf("parse error: %v", t.err) +} + +type textParser struct { + s string // remaining input + done bool // whether the parsing is finished (success or error) + backed bool // whether back() was called + offset, line int + cur token +} + +func newTextParser(s string) *textParser { + p := new(textParser) + p.s = s + p.line = 1 + p.cur.line = 1 + return p +} + +func (p *textParser) errorf(format string, a ...interface{}) *ParseError { + pe := &ParseError{fmt.Sprintf(format, a...), p.cur.line, p.cur.offset} + p.cur.err = pe + p.done = true + return pe +} + +// Numbers and identifiers are matched by [-+._A-Za-z0-9] +func isIdentOrNumberChar(c byte) bool { + switch { + case 'A' <= c && c <= 'Z', 'a' <= c && c <= 'z': + return true + case '0' <= c && c <= '9': + return true + } + switch c { + case '-', '+', '.', '_': + return true + } + return false +} + +func isWhitespace(c byte) bool { + switch c { + case ' ', '\t', '\n', '\r': + return true + } + return false +} + +func (p *textParser) skipWhitespace() { + i := 0 + for i < len(p.s) && (isWhitespace(p.s[i]) || p.s[i] == '#') { + if p.s[i] == '#' { + // comment; skip to end of line or input + for i < len(p.s) && p.s[i] != '\n' { + i++ + } + if i == len(p.s) { + break + } + } + if p.s[i] == '\n' { + p.line++ + } + i++ + } + p.offset += i + p.s = p.s[i:len(p.s)] + if len(p.s) == 0 { + p.done = true + } +} + +func (p *textParser) advance() { + // Skip whitespace + p.skipWhitespace() + if p.done { + return + } + + // Start of non-whitespace + p.cur.err = nil + p.cur.offset, p.cur.line = p.offset, p.line + p.cur.unquoted = "" + switch p.s[0] { + case '<', '>', '{', '}', ':', '[', ']', ';', ',': + // Single symbol + p.cur.value, p.s = p.s[0:1], p.s[1:len(p.s)] + case '"', '\'': + // Quoted string + i := 1 + for i < len(p.s) && p.s[i] != p.s[0] && p.s[i] != '\n' { + if p.s[i] == '\\' && i+1 < len(p.s) { + // skip escaped char + i++ + } + i++ + } + if i >= len(p.s) || p.s[i] != p.s[0] { + p.errorf("unmatched quote") + return + } + unq, err := unquoteC(p.s[1:i], rune(p.s[0])) + if err != nil { + p.errorf("invalid quoted string %s: %v", p.s[0:i+1], err) + return + } + p.cur.value, p.s = p.s[0:i+1], p.s[i+1:len(p.s)] + p.cur.unquoted = unq + default: + i := 0 + for i < len(p.s) && isIdentOrNumberChar(p.s[i]) { + i++ + } + if i == 0 { + p.errorf("unexpected byte %#x", p.s[0]) + return + } + p.cur.value, p.s = p.s[0:i], p.s[i:len(p.s)] + } + p.offset += len(p.cur.value) +} + +var ( + errBadUTF8 = errors.New("proto: bad UTF-8") + errBadHex = errors.New("proto: bad hexadecimal") +) + +func unquoteC(s string, quote rune) (string, error) { + // This is based on C++'s tokenizer.cc. + // Despite its name, this is *not* parsing C syntax. + // For instance, "\0" is an invalid quoted string. + + // Avoid allocation in trivial cases. + simple := true + for _, r := range s { + if r == '\\' || r == quote { + simple = false + break + } + } + if simple { + return s, nil + } + + buf := make([]byte, 0, 3*len(s)/2) + for len(s) > 0 { + r, n := utf8.DecodeRuneInString(s) + if r == utf8.RuneError && n == 1 { + return "", errBadUTF8 + } + s = s[n:] + if r != '\\' { + if r < utf8.RuneSelf { + buf = append(buf, byte(r)) + } else { + buf = append(buf, string(r)...) + } + continue + } + + ch, tail, err := unescape(s) + if err != nil { + return "", err + } + buf = append(buf, ch...) + s = tail + } + return string(buf), nil +} + +func unescape(s string) (ch string, tail string, err error) { + r, n := utf8.DecodeRuneInString(s) + if r == utf8.RuneError && n == 1 { + return "", "", errBadUTF8 + } + s = s[n:] + switch r { + case 'a': + return "\a", s, nil + case 'b': + return "\b", s, nil + case 'f': + return "\f", s, nil + case 'n': + return "\n", s, nil + case 'r': + return "\r", s, nil + case 't': + return "\t", s, nil + case 'v': + return "\v", s, nil + case '?': + return "?", s, nil // trigraph workaround + case '\'', '"', '\\': + return string(r), s, nil + case '0', '1', '2', '3', '4', '5', '6', '7', 'x', 'X': + if len(s) < 2 { + return "", "", fmt.Errorf(`\%c requires 2 following digits`, r) + } + base := 8 + ss := s[:2] + s = s[2:] + if r == 'x' || r == 'X' { + base = 16 + } else { + ss = string(r) + ss + } + i, err := strconv.ParseUint(ss, base, 8) + if err != nil { + return "", "", err + } + return string([]byte{byte(i)}), s, nil + case 'u', 'U': + n := 4 + if r == 'U' { + n = 8 + } + if len(s) < n { + return "", "", fmt.Errorf(`\%c requires %d digits`, r, n) + } + + bs := make([]byte, n/2) + for i := 0; i < n; i += 2 { + a, ok1 := unhex(s[i]) + b, ok2 := unhex(s[i+1]) + if !ok1 || !ok2 { + return "", "", errBadHex + } + bs[i/2] = a<<4 | b + } + s = s[n:] + return string(bs), s, nil + } + return "", "", fmt.Errorf(`unknown escape \%c`, r) +} + +// Adapted from src/pkg/strconv/quote.go. +func unhex(b byte) (v byte, ok bool) { + switch { + case '0' <= b && b <= '9': + return b - '0', true + case 'a' <= b && b <= 'f': + return b - 'a' + 10, true + case 'A' <= b && b <= 'F': + return b - 'A' + 10, true + } + return 0, false +} + +// Back off the parser by one token. Can only be done between calls to next(). +// It makes the next advance() a no-op. +func (p *textParser) back() { p.backed = true } + +// Advances the parser and returns the new current token. +func (p *textParser) next() *token { + if p.backed || p.done { + p.backed = false + return &p.cur + } + p.advance() + if p.done { + p.cur.value = "" + } else if len(p.cur.value) > 0 && p.cur.value[0] == '"' { + // Look for multiple quoted strings separated by whitespace, + // and concatenate them. + cat := p.cur + for { + p.skipWhitespace() + if p.done || p.s[0] != '"' { + break + } + p.advance() + if p.cur.err != nil { + return &p.cur + } + cat.value += " " + p.cur.value + cat.unquoted += p.cur.unquoted + } + p.done = false // parser may have seen EOF, but we want to return cat + p.cur = cat + } + return &p.cur +} + +func (p *textParser) consumeToken(s string) error { + tok := p.next() + if tok.err != nil { + return tok.err + } + if tok.value != s { + p.back() + return p.errorf("expected %q, found %q", s, tok.value) + } + return nil +} + +// Return a RequiredNotSetError indicating which required field was not set. +func (p *textParser) missingRequiredFieldError(sv reflect.Value) *RequiredNotSetError { + st := sv.Type() + sprops := GetProperties(st) + for i := 0; i < st.NumField(); i++ { + if !isNil(sv.Field(i)) { + continue + } + + props := sprops.Prop[i] + if props.Required { + return &RequiredNotSetError{fmt.Sprintf("%v.%v", st, props.OrigName)} + } + } + return &RequiredNotSetError{fmt.Sprintf("%v.", st)} // should not happen +} + +// Returns the index in the struct for the named field, as well as the parsed tag properties. +func structFieldByName(st reflect.Type, name string) (int, *Properties, bool) { + sprops := GetProperties(st) + i, ok := sprops.decoderOrigNames[name] + if ok { + return i, sprops.Prop[i], true + } + return -1, nil, false +} + +// Consume a ':' from the input stream (if the next token is a colon), +// returning an error if a colon is needed but not present. +func (p *textParser) checkForColon(props *Properties, typ reflect.Type) *ParseError { + tok := p.next() + if tok.err != nil { + return tok.err + } + if tok.value != ":" { + // Colon is optional when the field is a group or message. + needColon := true + switch props.Wire { + case "group": + needColon = false + case "bytes": + // A "bytes" field is either a message, a string, or a repeated field; + // those three become *T, *string and []T respectively, so we can check for + // this field being a pointer to a non-string. + if typ.Kind() == reflect.Ptr { + // *T or *string + if typ.Elem().Kind() == reflect.String { + break + } + } else if typ.Kind() == reflect.Slice { + // []T or []*T + if typ.Elem().Kind() != reflect.Ptr { + break + } + } else if typ.Kind() == reflect.String { + // The proto3 exception is for a string field, + // which requires a colon. + break + } + needColon = false + } + if needColon { + return p.errorf("expected ':', found %q", tok.value) + } + p.back() + } + return nil +} + +func (p *textParser) readStruct(sv reflect.Value, terminator string) error { + st := sv.Type() + reqCount := GetProperties(st).reqCount + var reqFieldErr error + fieldSet := make(map[string]bool) + // A struct is a sequence of "name: value", terminated by one of + // '>' or '}', or the end of the input. A name may also be + // "[extension]". + for { + tok := p.next() + if tok.err != nil { + return tok.err + } + if tok.value == terminator { + break + } + if tok.value == "[" { + // Looks like an extension. + // + // TODO: Check whether we need to handle + // namespace rooted names (e.g. ".something.Foo"). + tok = p.next() + if tok.err != nil { + return tok.err + } + var desc *ExtensionDesc + // This could be faster, but it's functional. + // TODO: Do something smarter than a linear scan. + for _, d := range RegisteredExtensions(reflect.New(st).Interface().(Message)) { + if d.Name == tok.value { + desc = d + break + } + } + if desc == nil { + return p.errorf("unrecognized extension %q", tok.value) + } + // Check the extension terminator. + tok = p.next() + if tok.err != nil { + return tok.err + } + if tok.value != "]" { + return p.errorf("unrecognized extension terminator %q", tok.value) + } + + props := &Properties{} + props.Parse(desc.Tag) + + typ := reflect.TypeOf(desc.ExtensionType) + if err := p.checkForColon(props, typ); err != nil { + return err + } + + rep := desc.repeated() + + // Read the extension structure, and set it in + // the value we're constructing. + var ext reflect.Value + if !rep { + ext = reflect.New(typ).Elem() + } else { + ext = reflect.New(typ.Elem()).Elem() + } + if err := p.readAny(ext, props); err != nil { + if _, ok := err.(*RequiredNotSetError); !ok { + return err + } + reqFieldErr = err + } + ep := sv.Addr().Interface().(extendableProto) + if !rep { + SetExtension(ep, desc, ext.Interface()) + } else { + old, err := GetExtension(ep, desc) + var sl reflect.Value + if err == nil { + sl = reflect.ValueOf(old) // existing slice + } else { + sl = reflect.MakeSlice(typ, 0, 1) + } + sl = reflect.Append(sl, ext) + SetExtension(ep, desc, sl.Interface()) + } + } else { + // This is a normal, non-extension field. + name := tok.value + fi, props, ok := structFieldByName(st, name) + if !ok { + return p.errorf("unknown field name %q in %v", name, st) + } + + dst := sv.Field(fi) + + if dst.Kind() == reflect.Map { + // Consume any colon. + if err := p.checkForColon(props, dst.Type()); err != nil { + return err + } + + // Construct the map if it doesn't already exist. + if dst.IsNil() { + dst.Set(reflect.MakeMap(dst.Type())) + } + key := reflect.New(dst.Type().Key()).Elem() + val := reflect.New(dst.Type().Elem()).Elem() + + // The map entry should be this sequence of tokens: + // < key : KEY value : VALUE > + // Technically the "key" and "value" could come in any order, + // but in practice they won't. + + tok := p.next() + var terminator string + switch tok.value { + case "<": + terminator = ">" + case "{": + terminator = "}" + default: + return p.errorf("expected '{' or '<', found %q", tok.value) + } + if err := p.consumeToken("key"); err != nil { + return err + } + if err := p.consumeToken(":"); err != nil { + return err + } + if err := p.readAny(key, props.mkeyprop); err != nil { + return err + } + if err := p.consumeOptionalSeparator(); err != nil { + return err + } + if err := p.consumeToken("value"); err != nil { + return err + } + if err := p.checkForColon(props.mvalprop, dst.Type().Elem()); err != nil { + return err + } + if err := p.readAny(val, props.mvalprop); err != nil { + return err + } + if err := p.consumeOptionalSeparator(); err != nil { + return err + } + if err := p.consumeToken(terminator); err != nil { + return err + } + + dst.SetMapIndex(key, val) + continue + } + + // Check that it's not already set if it's not a repeated field. + if !props.Repeated && fieldSet[name] { + return p.errorf("non-repeated field %q was repeated", name) + } + + if err := p.checkForColon(props, st.Field(fi).Type); err != nil { + return err + } + + // Parse into the field. + fieldSet[name] = true + if err := p.readAny(dst, props); err != nil { + if _, ok := err.(*RequiredNotSetError); !ok { + return err + } + reqFieldErr = err + } else if props.Required { + reqCount-- + } + } + + if err := p.consumeOptionalSeparator(); err != nil { + return err + } + + } + + if reqCount > 0 { + return p.missingRequiredFieldError(sv) + } + return reqFieldErr +} + +// consumeOptionalSeparator consumes an optional semicolon or comma. +// It is used in readStruct to provide backward compatibility. +func (p *textParser) consumeOptionalSeparator() error { + tok := p.next() + if tok.err != nil { + return tok.err + } + if tok.value != ";" && tok.value != "," { + p.back() + } + return nil +} + +func (p *textParser) readAny(v reflect.Value, props *Properties) error { + tok := p.next() + if tok.err != nil { + return tok.err + } + if tok.value == "" { + return p.errorf("unexpected EOF") + } + + switch fv := v; fv.Kind() { + case reflect.Slice: + at := v.Type() + if at.Elem().Kind() == reflect.Uint8 { + // Special case for []byte + if tok.value[0] != '"' && tok.value[0] != '\'' { + // Deliberately written out here, as the error after + // this switch statement would write "invalid []byte: ...", + // which is not as user-friendly. + return p.errorf("invalid string: %v", tok.value) + } + bytes := []byte(tok.unquoted) + fv.Set(reflect.ValueOf(bytes)) + return nil + } + // Repeated field. May already exist. + flen := fv.Len() + if flen == fv.Cap() { + nav := reflect.MakeSlice(at, flen, 2*flen+1) + reflect.Copy(nav, fv) + fv.Set(nav) + } + fv.SetLen(flen + 1) + + // Read one. + p.back() + return p.readAny(fv.Index(flen), props) + case reflect.Bool: + // Either "true", "false", 1 or 0. + switch tok.value { + case "true", "1": + fv.SetBool(true) + return nil + case "false", "0": + fv.SetBool(false) + return nil + } + case reflect.Float32, reflect.Float64: + v := tok.value + // Ignore 'f' for compatibility with output generated by C++, but don't + // remove 'f' when the value is "-inf" or "inf". + if strings.HasSuffix(v, "f") && tok.value != "-inf" && tok.value != "inf" { + v = v[:len(v)-1] + } + if f, err := strconv.ParseFloat(v, fv.Type().Bits()); err == nil { + fv.SetFloat(f) + return nil + } + case reflect.Int32: + if x, err := strconv.ParseInt(tok.value, 0, 32); err == nil { + fv.SetInt(x) + return nil + } + + if len(props.Enum) == 0 { + break + } + m, ok := enumValueMaps[props.Enum] + if !ok { + break + } + x, ok := m[tok.value] + if !ok { + break + } + fv.SetInt(int64(x)) + return nil + case reflect.Int64: + if x, err := strconv.ParseInt(tok.value, 0, 64); err == nil { + fv.SetInt(x) + return nil + } + + case reflect.Ptr: + // A basic field (indirected through pointer), or a repeated message/group + p.back() + fv.Set(reflect.New(fv.Type().Elem())) + return p.readAny(fv.Elem(), props) + case reflect.String: + if tok.value[0] == '"' || tok.value[0] == '\'' { + fv.SetString(tok.unquoted) + return nil + } + case reflect.Struct: + var terminator string + switch tok.value { + case "{": + terminator = "}" + case "<": + terminator = ">" + default: + return p.errorf("expected '{' or '<', found %q", tok.value) + } + // TODO: Handle nested messages which implement encoding.TextUnmarshaler. + return p.readStruct(fv, terminator) + case reflect.Uint32: + if x, err := strconv.ParseUint(tok.value, 0, 32); err == nil { + fv.SetUint(uint64(x)) + return nil + } + case reflect.Uint64: + if x, err := strconv.ParseUint(tok.value, 0, 64); err == nil { + fv.SetUint(x) + return nil + } + } + return p.errorf("invalid %v: %v", v.Type(), tok.value) +} + +// UnmarshalText reads a protocol buffer in Text format. UnmarshalText resets pb +// before starting to unmarshal, so any existing data in pb is always removed. +// If a required field is not set and no other error occurs, +// UnmarshalText returns *RequiredNotSetError. +func UnmarshalText(s string, pb Message) error { + if um, ok := pb.(encoding.TextUnmarshaler); ok { + err := um.UnmarshalText([]byte(s)) + return err + } + pb.Reset() + v := reflect.ValueOf(pb) + if pe := newTextParser(s).readStruct(v.Elem(), ""); pe != nil { + return pe + } + return nil +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/text_parser_test.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/text_parser_test.go new file mode 100644 index 0000000000000..0754b2626c7c2 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/text_parser_test.go @@ -0,0 +1,511 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "math" + "reflect" + "testing" + + . "github.com/golang/protobuf/proto" + proto3pb "github.com/golang/protobuf/proto/proto3_proto" + . "github.com/golang/protobuf/proto/testdata" +) + +type UnmarshalTextTest struct { + in string + err string // if "", no error expected + out *MyMessage +} + +func buildExtStructTest(text string) UnmarshalTextTest { + msg := &MyMessage{ + Count: Int32(42), + } + SetExtension(msg, E_Ext_More, &Ext{ + Data: String("Hello, world!"), + }) + return UnmarshalTextTest{in: text, out: msg} +} + +func buildExtDataTest(text string) UnmarshalTextTest { + msg := &MyMessage{ + Count: Int32(42), + } + SetExtension(msg, E_Ext_Text, String("Hello, world!")) + SetExtension(msg, E_Ext_Number, Int32(1729)) + return UnmarshalTextTest{in: text, out: msg} +} + +func buildExtRepStringTest(text string) UnmarshalTextTest { + msg := &MyMessage{ + Count: Int32(42), + } + if err := SetExtension(msg, E_Greeting, []string{"bula", "hola"}); err != nil { + panic(err) + } + return UnmarshalTextTest{in: text, out: msg} +} + +var unMarshalTextTests = []UnmarshalTextTest{ + // Basic + { + in: " count:42\n name:\"Dave\" ", + out: &MyMessage{ + Count: Int32(42), + Name: String("Dave"), + }, + }, + + // Empty quoted string + { + in: `count:42 name:""`, + out: &MyMessage{ + Count: Int32(42), + Name: String(""), + }, + }, + + // Quoted string concatenation + { + in: `count:42 name: "My name is "` + "\n" + `"elsewhere"`, + out: &MyMessage{ + Count: Int32(42), + Name: String("My name is elsewhere"), + }, + }, + + // Quoted string with escaped apostrophe + { + in: `count:42 name: "HOLIDAY - New Year\'s Day"`, + out: &MyMessage{ + Count: Int32(42), + Name: String("HOLIDAY - New Year's Day"), + }, + }, + + // Quoted string with single quote + { + in: `count:42 name: 'Roger "The Ramster" Ramjet'`, + out: &MyMessage{ + Count: Int32(42), + Name: String(`Roger "The Ramster" Ramjet`), + }, + }, + + // Quoted string with all the accepted special characters from the C++ test + { + in: `count:42 name: ` + "\"\\\"A string with \\' characters \\n and \\r newlines and \\t tabs and \\001 slashes \\\\ and multiple spaces\"", + out: &MyMessage{ + Count: Int32(42), + Name: String("\"A string with ' characters \n and \r newlines and \t tabs and \001 slashes \\ and multiple spaces"), + }, + }, + + // Quoted string with quoted backslash + { + in: `count:42 name: "\\'xyz"`, + out: &MyMessage{ + Count: Int32(42), + Name: String(`\'xyz`), + }, + }, + + // Quoted string with UTF-8 bytes. + { + in: "count:42 name: '\303\277\302\201\xAB'", + out: &MyMessage{ + Count: Int32(42), + Name: String("\303\277\302\201\xAB"), + }, + }, + + // Bad quoted string + { + in: `inner: < host: "\0" >` + "\n", + err: `line 1.15: invalid quoted string "\0": \0 requires 2 following digits`, + }, + + // Number too large for int64 + { + in: "count: 1 others { key: 123456789012345678901 }", + err: "line 1.23: invalid int64: 123456789012345678901", + }, + + // Number too large for int32 + { + in: "count: 1234567890123", + err: "line 1.7: invalid int32: 1234567890123", + }, + + // Number in hexadecimal + { + in: "count: 0x2beef", + out: &MyMessage{ + Count: Int32(0x2beef), + }, + }, + + // Number in octal + { + in: "count: 024601", + out: &MyMessage{ + Count: Int32(024601), + }, + }, + + // Floating point number with "f" suffix + { + in: "count: 4 others:< weight: 17.0f >", + out: &MyMessage{ + Count: Int32(4), + Others: []*OtherMessage{ + { + Weight: Float32(17), + }, + }, + }, + }, + + // Floating point positive infinity + { + in: "count: 4 bigfloat: inf", + out: &MyMessage{ + Count: Int32(4), + Bigfloat: Float64(math.Inf(1)), + }, + }, + + // Floating point negative infinity + { + in: "count: 4 bigfloat: -inf", + out: &MyMessage{ + Count: Int32(4), + Bigfloat: Float64(math.Inf(-1)), + }, + }, + + // Number too large for float32 + { + in: "others:< weight: 12345678901234567890123456789012345678901234567890 >", + err: "line 1.17: invalid float32: 12345678901234567890123456789012345678901234567890", + }, + + // Number posing as a quoted string + { + in: `inner: < host: 12 >` + "\n", + err: `line 1.15: invalid string: 12`, + }, + + // Quoted string posing as int32 + { + in: `count: "12"`, + err: `line 1.7: invalid int32: "12"`, + }, + + // Quoted string posing a float32 + { + in: `others:< weight: "17.4" >`, + err: `line 1.17: invalid float32: "17.4"`, + }, + + // Enum + { + in: `count:42 bikeshed: BLUE`, + out: &MyMessage{ + Count: Int32(42), + Bikeshed: MyMessage_BLUE.Enum(), + }, + }, + + // Repeated field + { + in: `count:42 pet: "horsey" pet:"bunny"`, + out: &MyMessage{ + Count: Int32(42), + Pet: []string{"horsey", "bunny"}, + }, + }, + + // Repeated message with/without colon and <>/{} + { + in: `count:42 others:{} others{} others:<> others:{}`, + out: &MyMessage{ + Count: Int32(42), + Others: []*OtherMessage{ + {}, + {}, + {}, + {}, + }, + }, + }, + + // Missing colon for inner message + { + in: `count:42 inner < host: "cauchy.syd" >`, + out: &MyMessage{ + Count: Int32(42), + Inner: &InnerMessage{ + Host: String("cauchy.syd"), + }, + }, + }, + + // Missing colon for string field + { + in: `name "Dave"`, + err: `line 1.5: expected ':', found "\"Dave\""`, + }, + + // Missing colon for int32 field + { + in: `count 42`, + err: `line 1.6: expected ':', found "42"`, + }, + + // Missing required field + { + in: `name: "Pawel"`, + err: `proto: required field "testdata.MyMessage.count" not set`, + out: &MyMessage{ + Name: String("Pawel"), + }, + }, + + // Repeated non-repeated field + { + in: `name: "Rob" name: "Russ"`, + err: `line 1.12: non-repeated field "name" was repeated`, + }, + + // Group + { + in: `count: 17 SomeGroup { group_field: 12 }`, + out: &MyMessage{ + Count: Int32(17), + Somegroup: &MyMessage_SomeGroup{ + GroupField: Int32(12), + }, + }, + }, + + // Semicolon between fields + { + in: `count:3;name:"Calvin"`, + out: &MyMessage{ + Count: Int32(3), + Name: String("Calvin"), + }, + }, + // Comma between fields + { + in: `count:4,name:"Ezekiel"`, + out: &MyMessage{ + Count: Int32(4), + Name: String("Ezekiel"), + }, + }, + + // Extension + buildExtStructTest(`count: 42 [testdata.Ext.more]:`), + buildExtStructTest(`count: 42 [testdata.Ext.more] {data:"Hello, world!"}`), + buildExtDataTest(`count: 42 [testdata.Ext.text]:"Hello, world!" [testdata.Ext.number]:1729`), + buildExtRepStringTest(`count: 42 [testdata.greeting]:"bula" [testdata.greeting]:"hola"`), + + // Big all-in-one + { + in: "count:42 # Meaning\n" + + `name:"Dave" ` + + `quote:"\"I didn't want to go.\"" ` + + `pet:"bunny" ` + + `pet:"kitty" ` + + `pet:"horsey" ` + + `inner:<` + + ` host:"footrest.syd" ` + + ` port:7001 ` + + ` connected:true ` + + `> ` + + `others:<` + + ` key:3735928559 ` + + ` value:"\x01A\a\f" ` + + `> ` + + `others:<` + + " weight:58.9 # Atomic weight of Co\n" + + ` inner:<` + + ` host:"lesha.mtv" ` + + ` port:8002 ` + + ` >` + + `>`, + out: &MyMessage{ + Count: Int32(42), + Name: String("Dave"), + Quote: String(`"I didn't want to go."`), + Pet: []string{"bunny", "kitty", "horsey"}, + Inner: &InnerMessage{ + Host: String("footrest.syd"), + Port: Int32(7001), + Connected: Bool(true), + }, + Others: []*OtherMessage{ + { + Key: Int64(3735928559), + Value: []byte{0x1, 'A', '\a', '\f'}, + }, + { + Weight: Float32(58.9), + Inner: &InnerMessage{ + Host: String("lesha.mtv"), + Port: Int32(8002), + }, + }, + }, + }, + }, +} + +func TestUnmarshalText(t *testing.T) { + for i, test := range unMarshalTextTests { + pb := new(MyMessage) + err := UnmarshalText(test.in, pb) + if test.err == "" { + // We don't expect failure. + if err != nil { + t.Errorf("Test %d: Unexpected error: %v", i, err) + } else if !reflect.DeepEqual(pb, test.out) { + t.Errorf("Test %d: Incorrect populated \nHave: %v\nWant: %v", + i, pb, test.out) + } + } else { + // We do expect failure. + if err == nil { + t.Errorf("Test %d: Didn't get expected error: %v", i, test.err) + } else if err.Error() != test.err { + t.Errorf("Test %d: Incorrect error.\nHave: %v\nWant: %v", + i, err.Error(), test.err) + } else if _, ok := err.(*RequiredNotSetError); ok && test.out != nil && !reflect.DeepEqual(pb, test.out) { + t.Errorf("Test %d: Incorrect populated \nHave: %v\nWant: %v", + i, pb, test.out) + } + } + } +} + +func TestUnmarshalTextCustomMessage(t *testing.T) { + msg := &textMessage{} + if err := UnmarshalText("custom", msg); err != nil { + t.Errorf("Unexpected error from custom unmarshal: %v", err) + } + if UnmarshalText("not custom", msg) == nil { + t.Errorf("Didn't get expected error from custom unmarshal") + } +} + +// Regression test; this caused a panic. +func TestRepeatedEnum(t *testing.T) { + pb := new(RepeatedEnum) + if err := UnmarshalText("color: RED", pb); err != nil { + t.Fatal(err) + } + exp := &RepeatedEnum{ + Color: []RepeatedEnum_Color{RepeatedEnum_RED}, + } + if !Equal(pb, exp) { + t.Errorf("Incorrect populated \nHave: %v\nWant: %v", pb, exp) + } +} + +func TestProto3TextParsing(t *testing.T) { + m := new(proto3pb.Message) + const in = `name: "Wallace" true_scotsman: true` + want := &proto3pb.Message{ + Name: "Wallace", + TrueScotsman: true, + } + if err := UnmarshalText(in, m); err != nil { + t.Fatal(err) + } + if !Equal(m, want) { + t.Errorf("\n got %v\nwant %v", m, want) + } +} + +func TestMapParsing(t *testing.T) { + m := new(MessageWithMap) + const in = `name_mapping: name_mapping:` + + `msg_mapping:,>` + // separating commas are okay + `msg_mapping>` + // no colon after "value" + `byte_mapping:` + want := &MessageWithMap{ + NameMapping: map[int32]string{ + 1: "Beatles", + 1234: "Feist", + }, + MsgMapping: map[int64]*FloatingPoint{ + -4: {F: Float64(2.0)}, + -2: {F: Float64(4.0)}, + }, + ByteMapping: map[bool][]byte{ + true: []byte("so be it"), + }, + } + if err := UnmarshalText(in, m); err != nil { + t.Fatal(err) + } + if !Equal(m, want) { + t.Errorf("\n got %v\nwant %v", m, want) + } +} + +var benchInput string + +func init() { + benchInput = "count: 4\n" + for i := 0; i < 1000; i++ { + benchInput += "pet: \"fido\"\n" + } + + // Check it is valid input. + pb := new(MyMessage) + err := UnmarshalText(benchInput, pb) + if err != nil { + panic("Bad benchmark input: " + err.Error()) + } +} + +func BenchmarkUnmarshalText(b *testing.B) { + pb := new(MyMessage) + for i := 0; i < b.N; i++ { + UnmarshalText(benchInput, pb) + } + b.SetBytes(int64(len(benchInput))) +} diff --git a/Godeps/_workspace/src/github.com/golang/protobuf/proto/text_test.go b/Godeps/_workspace/src/github.com/golang/protobuf/proto/text_test.go new file mode 100644 index 0000000000000..64579e94d787a --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/protobuf/proto/text_test.go @@ -0,0 +1,450 @@ +// Go support for Protocol Buffers - Google's data interchange format +// +// Copyright 2010 The Go Authors. All rights reserved. +// https://github.com/golang/protobuf +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +package proto_test + +import ( + "bytes" + "errors" + "io/ioutil" + "math" + "strings" + "testing" + + "github.com/golang/protobuf/proto" + + proto3pb "github.com/golang/protobuf/proto/proto3_proto" + pb "github.com/golang/protobuf/proto/testdata" +) + +// textMessage implements the methods that allow it to marshal and unmarshal +// itself as text. +type textMessage struct { +} + +func (*textMessage) MarshalText() ([]byte, error) { + return []byte("custom"), nil +} + +func (*textMessage) UnmarshalText(bytes []byte) error { + if string(bytes) != "custom" { + return errors.New("expected 'custom'") + } + return nil +} + +func (*textMessage) Reset() {} +func (*textMessage) String() string { return "" } +func (*textMessage) ProtoMessage() {} + +func newTestMessage() *pb.MyMessage { + msg := &pb.MyMessage{ + Count: proto.Int32(42), + Name: proto.String("Dave"), + Quote: proto.String(`"I didn't want to go."`), + Pet: []string{"bunny", "kitty", "horsey"}, + Inner: &pb.InnerMessage{ + Host: proto.String("footrest.syd"), + Port: proto.Int32(7001), + Connected: proto.Bool(true), + }, + Others: []*pb.OtherMessage{ + { + Key: proto.Int64(0xdeadbeef), + Value: []byte{1, 65, 7, 12}, + }, + { + Weight: proto.Float32(6.022), + Inner: &pb.InnerMessage{ + Host: proto.String("lesha.mtv"), + Port: proto.Int32(8002), + }, + }, + }, + Bikeshed: pb.MyMessage_BLUE.Enum(), + Somegroup: &pb.MyMessage_SomeGroup{ + GroupField: proto.Int32(8), + }, + // One normally wouldn't do this. + // This is an undeclared tag 13, as a varint (wire type 0) with value 4. + XXX_unrecognized: []byte{13<<3 | 0, 4}, + } + ext := &pb.Ext{ + Data: proto.String("Big gobs for big rats"), + } + if err := proto.SetExtension(msg, pb.E_Ext_More, ext); err != nil { + panic(err) + } + greetings := []string{"adg", "easy", "cow"} + if err := proto.SetExtension(msg, pb.E_Greeting, greetings); err != nil { + panic(err) + } + + // Add an unknown extension. We marshal a pb.Ext, and fake the ID. + b, err := proto.Marshal(&pb.Ext{Data: proto.String("3G skiing")}) + if err != nil { + panic(err) + } + b = append(proto.EncodeVarint(201<<3|proto.WireBytes), b...) + proto.SetRawExtension(msg, 201, b) + + // Extensions can be plain fields, too, so let's test that. + b = append(proto.EncodeVarint(202<<3|proto.WireVarint), 19) + proto.SetRawExtension(msg, 202, b) + + return msg +} + +const text = `count: 42 +name: "Dave" +quote: "\"I didn't want to go.\"" +pet: "bunny" +pet: "kitty" +pet: "horsey" +inner: < + host: "footrest.syd" + port: 7001 + connected: true +> +others: < + key: 3735928559 + value: "\001A\007\014" +> +others: < + weight: 6.022 + inner: < + host: "lesha.mtv" + port: 8002 + > +> +bikeshed: BLUE +SomeGroup { + group_field: 8 +} +/* 2 unknown bytes */ +13: 4 +[testdata.Ext.more]: < + data: "Big gobs for big rats" +> +[testdata.greeting]: "adg" +[testdata.greeting]: "easy" +[testdata.greeting]: "cow" +/* 13 unknown bytes */ +201: "\t3G skiing" +/* 3 unknown bytes */ +202: 19 +` + +func TestMarshalText(t *testing.T) { + buf := new(bytes.Buffer) + if err := proto.MarshalText(buf, newTestMessage()); err != nil { + t.Fatalf("proto.MarshalText: %v", err) + } + s := buf.String() + if s != text { + t.Errorf("Got:\n===\n%v===\nExpected:\n===\n%v===\n", s, text) + } +} + +func TestMarshalTextCustomMessage(t *testing.T) { + buf := new(bytes.Buffer) + if err := proto.MarshalText(buf, &textMessage{}); err != nil { + t.Fatalf("proto.MarshalText: %v", err) + } + s := buf.String() + if s != "custom" { + t.Errorf("Got %q, expected %q", s, "custom") + } +} +func TestMarshalTextNil(t *testing.T) { + want := "" + tests := []proto.Message{nil, (*pb.MyMessage)(nil)} + for i, test := range tests { + buf := new(bytes.Buffer) + if err := proto.MarshalText(buf, test); err != nil { + t.Fatal(err) + } + if got := buf.String(); got != want { + t.Errorf("%d: got %q want %q", i, got, want) + } + } +} + +func TestMarshalTextUnknownEnum(t *testing.T) { + // The Color enum only specifies values 0-2. + m := &pb.MyMessage{Bikeshed: pb.MyMessage_Color(3).Enum()} + got := m.String() + const want = `bikeshed:3 ` + if got != want { + t.Errorf("\n got %q\nwant %q", got, want) + } +} + +func BenchmarkMarshalTextBuffered(b *testing.B) { + buf := new(bytes.Buffer) + m := newTestMessage() + for i := 0; i < b.N; i++ { + buf.Reset() + proto.MarshalText(buf, m) + } +} + +func BenchmarkMarshalTextUnbuffered(b *testing.B) { + w := ioutil.Discard + m := newTestMessage() + for i := 0; i < b.N; i++ { + proto.MarshalText(w, m) + } +} + +func compact(src string) string { + // s/[ \n]+/ /g; s/ $//; + dst := make([]byte, len(src)) + space, comment := false, false + j := 0 + for i := 0; i < len(src); i++ { + if strings.HasPrefix(src[i:], "/*") { + comment = true + i++ + continue + } + if comment && strings.HasPrefix(src[i:], "*/") { + comment = false + i++ + continue + } + if comment { + continue + } + c := src[i] + if c == ' ' || c == '\n' { + space = true + continue + } + if j > 0 && (dst[j-1] == ':' || dst[j-1] == '<' || dst[j-1] == '{') { + space = false + } + if c == '{' { + space = false + } + if space { + dst[j] = ' ' + j++ + space = false + } + dst[j] = c + j++ + } + if space { + dst[j] = ' ' + j++ + } + return string(dst[0:j]) +} + +var compactText = compact(text) + +func TestCompactText(t *testing.T) { + s := proto.CompactTextString(newTestMessage()) + if s != compactText { + t.Errorf("Got:\n===\n%v===\nExpected:\n===\n%v\n===\n", s, compactText) + } +} + +func TestStringEscaping(t *testing.T) { + testCases := []struct { + in *pb.Strings + out string + }{ + { + // Test data from C++ test (TextFormatTest.StringEscape). + // Single divergence: we don't escape apostrophes. + &pb.Strings{StringField: proto.String("\"A string with ' characters \n and \r newlines and \t tabs and \001 slashes \\ and multiple spaces")}, + "string_field: \"\\\"A string with ' characters \\n and \\r newlines and \\t tabs and \\001 slashes \\\\ and multiple spaces\"\n", + }, + { + // Test data from the same C++ test. + &pb.Strings{StringField: proto.String("\350\260\267\346\255\214")}, + "string_field: \"\\350\\260\\267\\346\\255\\214\"\n", + }, + { + // Some UTF-8. + &pb.Strings{StringField: proto.String("\x00\x01\xff\x81")}, + `string_field: "\000\001\377\201"` + "\n", + }, + } + + for i, tc := range testCases { + var buf bytes.Buffer + if err := proto.MarshalText(&buf, tc.in); err != nil { + t.Errorf("proto.MarsalText: %v", err) + continue + } + s := buf.String() + if s != tc.out { + t.Errorf("#%d: Got:\n%s\nExpected:\n%s\n", i, s, tc.out) + continue + } + + // Check round-trip. + pb := new(pb.Strings) + if err := proto.UnmarshalText(s, pb); err != nil { + t.Errorf("#%d: UnmarshalText: %v", i, err) + continue + } + if !proto.Equal(pb, tc.in) { + t.Errorf("#%d: Round-trip failed:\nstart: %v\n end: %v", i, tc.in, pb) + } + } +} + +// A limitedWriter accepts some output before it fails. +// This is a proxy for something like a nearly-full or imminently-failing disk, +// or a network connection that is about to die. +type limitedWriter struct { + b bytes.Buffer + limit int +} + +var outOfSpace = errors.New("proto: insufficient space") + +func (w *limitedWriter) Write(p []byte) (n int, err error) { + var avail = w.limit - w.b.Len() + if avail <= 0 { + return 0, outOfSpace + } + if len(p) <= avail { + return w.b.Write(p) + } + n, _ = w.b.Write(p[:avail]) + return n, outOfSpace +} + +func TestMarshalTextFailing(t *testing.T) { + // Try lots of different sizes to exercise more error code-paths. + for lim := 0; lim < len(text); lim++ { + buf := new(limitedWriter) + buf.limit = lim + err := proto.MarshalText(buf, newTestMessage()) + // We expect a certain error, but also some partial results in the buffer. + if err != outOfSpace { + t.Errorf("Got:\n===\n%v===\nExpected:\n===\n%v===\n", err, outOfSpace) + } + s := buf.b.String() + x := text[:buf.limit] + if s != x { + t.Errorf("Got:\n===\n%v===\nExpected:\n===\n%v===\n", s, x) + } + } +} + +func TestFloats(t *testing.T) { + tests := []struct { + f float64 + want string + }{ + {0, "0"}, + {4.7, "4.7"}, + {math.Inf(1), "inf"}, + {math.Inf(-1), "-inf"}, + {math.NaN(), "nan"}, + } + for _, test := range tests { + msg := &pb.FloatingPoint{F: &test.f} + got := strings.TrimSpace(msg.String()) + want := `f:` + test.want + if got != want { + t.Errorf("f=%f: got %q, want %q", test.f, got, want) + } + } +} + +func TestRepeatedNilText(t *testing.T) { + m := &pb.MessageList{ + Message: []*pb.MessageList_Message{ + nil, + &pb.MessageList_Message{ + Name: proto.String("Horse"), + }, + nil, + }, + } + want := `Message +Message { + name: "Horse" +} +Message +` + if s := proto.MarshalTextString(m); s != want { + t.Errorf(" got: %s\nwant: %s", s, want) + } +} + +func TestProto3Text(t *testing.T) { + tests := []struct { + m proto.Message + want string + }{ + // zero message + {&proto3pb.Message{}, ``}, + // zero message except for an empty byte slice + {&proto3pb.Message{Data: []byte{}}, ``}, + // trivial case + {&proto3pb.Message{Name: "Rob", HeightInCm: 175}, `name:"Rob" height_in_cm:175`}, + // empty map + {&pb.MessageWithMap{}, ``}, + // non-empty map; map format is the same as a repeated struct, + // and they are sorted by key (numerically for numeric keys). + { + &pb.MessageWithMap{NameMapping: map[int32]string{ + -1: "Negatory", + 7: "Lucky", + 1234: "Feist", + 6345789: "Otis", + }}, + `name_mapping: ` + + `name_mapping: ` + + `name_mapping: ` + + `name_mapping:`, + }, + // map with nil value; not well-defined, but we shouldn't crash + { + &pb.MessageWithMap{MsgMapping: map[int64]*pb.FloatingPoint{7: nil}}, + `msg_mapping:`, + }, + } + for _, test := range tests { + got := strings.TrimSpace(test.m.String()) + if got != test.want { + t.Errorf("\n got %s\nwant %s", got, test.want) + } + } +} diff --git a/Godeps/_workspace/src/github.com/golang/snappy/AUTHORS b/Godeps/_workspace/src/github.com/golang/snappy/AUTHORS new file mode 100644 index 0000000000000..824bf2e1481e5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/snappy/AUTHORS @@ -0,0 +1,14 @@ +# This is the official list of Snappy-Go authors for copyright purposes. +# This file is distinct from the CONTRIBUTORS files. +# See the latter for an explanation. + +# Names should be added to this file as +# Name or Organization +# The email address is not required for organizations. + +# Please keep the list sorted. + +Damian Gryski +Google Inc. +Jan Mercl <0xjnml@gmail.com> +Sebastien Binet diff --git a/Godeps/_workspace/src/github.com/golang/snappy/CONTRIBUTORS b/Godeps/_workspace/src/github.com/golang/snappy/CONTRIBUTORS new file mode 100644 index 0000000000000..9f54f21ff7da5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/snappy/CONTRIBUTORS @@ -0,0 +1,36 @@ +# This is the official list of people who can contribute +# (and typically have contributed) code to the Snappy-Go repository. +# The AUTHORS file lists the copyright holders; this file +# lists people. For example, Google employees are listed here +# but not in AUTHORS, because Google holds the copyright. +# +# The submission process automatically checks to make sure +# that people submitting code are listed in this file (by email address). +# +# Names should be added to this file only after verifying that +# the individual or the individual's organization has agreed to +# the appropriate Contributor License Agreement, found here: +# +# http://code.google.com/legal/individual-cla-v1.0.html +# http://code.google.com/legal/corporate-cla-v1.0.html +# +# The agreement for individuals can be filled out on the web. +# +# When adding J Random Contributor's name to this file, +# either J's name or J's organization's name should be +# added to the AUTHORS file, depending on whether the +# individual or corporate CLA was used. + +# Names should be added to this file like so: +# Name + +# Please keep the list sorted. + +Damian Gryski +Jan Mercl <0xjnml@gmail.com> +Kai Backman +Marc-Antoine Ruel +Nigel Tao +Rob Pike +Russ Cox +Sebastien Binet diff --git a/Godeps/_workspace/src/github.com/golang/snappy/LICENSE b/Godeps/_workspace/src/github.com/golang/snappy/LICENSE new file mode 100644 index 0000000000000..6050c10f4c8b4 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/snappy/LICENSE @@ -0,0 +1,27 @@ +Copyright (c) 2011 The Snappy-Go Authors. All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are +met: + + * Redistributions of source code must retain the above copyright +notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above +copyright notice, this list of conditions and the following disclaimer +in the documentation and/or other materials provided with the +distribution. + * Neither the name of Google Inc. nor the names of its +contributors may be used to endorse or promote products derived from +this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/Godeps/_workspace/src/github.com/golang/snappy/README b/Godeps/_workspace/src/github.com/golang/snappy/README new file mode 100644 index 0000000000000..5074bbab8dee3 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/snappy/README @@ -0,0 +1,7 @@ +The Snappy compression format in the Go programming language. + +To download and install from source: +$ go get github.com/golang/snappy + +Unless otherwise noted, the Snappy-Go source files are distributed +under the BSD-style license found in the LICENSE file. diff --git a/Godeps/_workspace/src/github.com/golang/snappy/decode.go b/Godeps/_workspace/src/github.com/golang/snappy/decode.go new file mode 100644 index 0000000000000..e7f1259a34aba --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/snappy/decode.go @@ -0,0 +1,294 @@ +// Copyright 2011 The Snappy-Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package snappy + +import ( + "encoding/binary" + "errors" + "io" +) + +var ( + // ErrCorrupt reports that the input is invalid. + ErrCorrupt = errors.New("snappy: corrupt input") + // ErrTooLarge reports that the uncompressed length is too large. + ErrTooLarge = errors.New("snappy: decoded block is too large") + // ErrUnsupported reports that the input isn't supported. + ErrUnsupported = errors.New("snappy: unsupported input") +) + +// DecodedLen returns the length of the decoded block. +func DecodedLen(src []byte) (int, error) { + v, _, err := decodedLen(src) + return v, err +} + +// decodedLen returns the length of the decoded block and the number of bytes +// that the length header occupied. +func decodedLen(src []byte) (blockLen, headerLen int, err error) { + v, n := binary.Uvarint(src) + if n <= 0 || v > 0xffffffff { + return 0, 0, ErrCorrupt + } + + const wordSize = 32 << (^uint(0) >> 32 & 1) + if wordSize == 32 && v > 0x7fffffff { + return 0, 0, ErrTooLarge + } + return int(v), n, nil +} + +// Decode returns the decoded form of src. The returned slice may be a sub- +// slice of dst if dst was large enough to hold the entire decoded block. +// Otherwise, a newly allocated slice will be returned. +// It is valid to pass a nil dst. +func Decode(dst, src []byte) ([]byte, error) { + dLen, s, err := decodedLen(src) + if err != nil { + return nil, err + } + if len(dst) < dLen { + dst = make([]byte, dLen) + } + + var d, offset, length int + for s < len(src) { + switch src[s] & 0x03 { + case tagLiteral: + x := uint(src[s] >> 2) + switch { + case x < 60: + s++ + case x == 60: + s += 2 + if s > len(src) { + return nil, ErrCorrupt + } + x = uint(src[s-1]) + case x == 61: + s += 3 + if s > len(src) { + return nil, ErrCorrupt + } + x = uint(src[s-2]) | uint(src[s-1])<<8 + case x == 62: + s += 4 + if s > len(src) { + return nil, ErrCorrupt + } + x = uint(src[s-3]) | uint(src[s-2])<<8 | uint(src[s-1])<<16 + case x == 63: + s += 5 + if s > len(src) { + return nil, ErrCorrupt + } + x = uint(src[s-4]) | uint(src[s-3])<<8 | uint(src[s-2])<<16 | uint(src[s-1])<<24 + } + length = int(x + 1) + if length <= 0 { + return nil, errors.New("snappy: unsupported literal length") + } + if length > len(dst)-d || length > len(src)-s { + return nil, ErrCorrupt + } + copy(dst[d:], src[s:s+length]) + d += length + s += length + continue + + case tagCopy1: + s += 2 + if s > len(src) { + return nil, ErrCorrupt + } + length = 4 + int(src[s-2])>>2&0x7 + offset = int(src[s-2])&0xe0<<3 | int(src[s-1]) + + case tagCopy2: + s += 3 + if s > len(src) { + return nil, ErrCorrupt + } + length = 1 + int(src[s-3])>>2 + offset = int(src[s-2]) | int(src[s-1])<<8 + + case tagCopy4: + return nil, errors.New("snappy: unsupported COPY_4 tag") + } + + end := d + length + if offset > d || end > len(dst) { + return nil, ErrCorrupt + } + for ; d < end; d++ { + dst[d] = dst[d-offset] + } + } + if d != dLen { + return nil, ErrCorrupt + } + return dst[:d], nil +} + +// NewReader returns a new Reader that decompresses from r, using the framing +// format described at +// https://github.com/google/snappy/blob/master/framing_format.txt +func NewReader(r io.Reader) *Reader { + return &Reader{ + r: r, + decoded: make([]byte, maxUncompressedChunkLen), + buf: make([]byte, MaxEncodedLen(maxUncompressedChunkLen)+checksumSize), + } +} + +// Reader is an io.Reader than can read Snappy-compressed bytes. +type Reader struct { + r io.Reader + err error + decoded []byte + buf []byte + // decoded[i:j] contains decoded bytes that have not yet been passed on. + i, j int + readHeader bool +} + +// Reset discards any buffered data, resets all state, and switches the Snappy +// reader to read from r. This permits reusing a Reader rather than allocating +// a new one. +func (r *Reader) Reset(reader io.Reader) { + r.r = reader + r.err = nil + r.i = 0 + r.j = 0 + r.readHeader = false +} + +func (r *Reader) readFull(p []byte) (ok bool) { + if _, r.err = io.ReadFull(r.r, p); r.err != nil { + if r.err == io.ErrUnexpectedEOF { + r.err = ErrCorrupt + } + return false + } + return true +} + +// Read satisfies the io.Reader interface. +func (r *Reader) Read(p []byte) (int, error) { + if r.err != nil { + return 0, r.err + } + for { + if r.i < r.j { + n := copy(p, r.decoded[r.i:r.j]) + r.i += n + return n, nil + } + if !r.readFull(r.buf[:4]) { + return 0, r.err + } + chunkType := r.buf[0] + if !r.readHeader { + if chunkType != chunkTypeStreamIdentifier { + r.err = ErrCorrupt + return 0, r.err + } + r.readHeader = true + } + chunkLen := int(r.buf[1]) | int(r.buf[2])<<8 | int(r.buf[3])<<16 + if chunkLen > len(r.buf) { + r.err = ErrUnsupported + return 0, r.err + } + + // The chunk types are specified at + // https://github.com/google/snappy/blob/master/framing_format.txt + switch chunkType { + case chunkTypeCompressedData: + // Section 4.2. Compressed data (chunk type 0x00). + if chunkLen < checksumSize { + r.err = ErrCorrupt + return 0, r.err + } + buf := r.buf[:chunkLen] + if !r.readFull(buf) { + return 0, r.err + } + checksum := uint32(buf[0]) | uint32(buf[1])<<8 | uint32(buf[2])<<16 | uint32(buf[3])<<24 + buf = buf[checksumSize:] + + n, err := DecodedLen(buf) + if err != nil { + r.err = err + return 0, r.err + } + if n > len(r.decoded) { + r.err = ErrCorrupt + return 0, r.err + } + if _, err := Decode(r.decoded, buf); err != nil { + r.err = err + return 0, r.err + } + if crc(r.decoded[:n]) != checksum { + r.err = ErrCorrupt + return 0, r.err + } + r.i, r.j = 0, n + continue + + case chunkTypeUncompressedData: + // Section 4.3. Uncompressed data (chunk type 0x01). + if chunkLen < checksumSize { + r.err = ErrCorrupt + return 0, r.err + } + buf := r.buf[:checksumSize] + if !r.readFull(buf) { + return 0, r.err + } + checksum := uint32(buf[0]) | uint32(buf[1])<<8 | uint32(buf[2])<<16 | uint32(buf[3])<<24 + // Read directly into r.decoded instead of via r.buf. + n := chunkLen - checksumSize + if !r.readFull(r.decoded[:n]) { + return 0, r.err + } + if crc(r.decoded[:n]) != checksum { + r.err = ErrCorrupt + return 0, r.err + } + r.i, r.j = 0, n + continue + + case chunkTypeStreamIdentifier: + // Section 4.1. Stream identifier (chunk type 0xff). + if chunkLen != len(magicBody) { + r.err = ErrCorrupt + return 0, r.err + } + if !r.readFull(r.buf[:len(magicBody)]) { + return 0, r.err + } + for i := 0; i < len(magicBody); i++ { + if r.buf[i] != magicBody[i] { + r.err = ErrCorrupt + return 0, r.err + } + } + continue + } + + if chunkType <= 0x7f { + // Section 4.5. Reserved unskippable chunks (chunk types 0x02-0x7f). + r.err = ErrUnsupported + return 0, r.err + } + // Section 4.4 Padding (chunk type 0xfe). + // Section 4.6. Reserved skippable chunks (chunk types 0x80-0xfd). + if !r.readFull(r.buf[:chunkLen]) { + return 0, r.err + } + } +} diff --git a/Godeps/_workspace/src/github.com/golang/snappy/encode.go b/Godeps/_workspace/src/github.com/golang/snappy/encode.go new file mode 100644 index 0000000000000..f3b5484bc7e0f --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/snappy/encode.go @@ -0,0 +1,254 @@ +// Copyright 2011 The Snappy-Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package snappy + +import ( + "encoding/binary" + "io" +) + +// We limit how far copy back-references can go, the same as the C++ code. +const maxOffset = 1 << 15 + +// emitLiteral writes a literal chunk and returns the number of bytes written. +func emitLiteral(dst, lit []byte) int { + i, n := 0, uint(len(lit)-1) + switch { + case n < 60: + dst[0] = uint8(n)<<2 | tagLiteral + i = 1 + case n < 1<<8: + dst[0] = 60<<2 | tagLiteral + dst[1] = uint8(n) + i = 2 + case n < 1<<16: + dst[0] = 61<<2 | tagLiteral + dst[1] = uint8(n) + dst[2] = uint8(n >> 8) + i = 3 + case n < 1<<24: + dst[0] = 62<<2 | tagLiteral + dst[1] = uint8(n) + dst[2] = uint8(n >> 8) + dst[3] = uint8(n >> 16) + i = 4 + case int64(n) < 1<<32: + dst[0] = 63<<2 | tagLiteral + dst[1] = uint8(n) + dst[2] = uint8(n >> 8) + dst[3] = uint8(n >> 16) + dst[4] = uint8(n >> 24) + i = 5 + default: + panic("snappy: source buffer is too long") + } + if copy(dst[i:], lit) != len(lit) { + panic("snappy: destination buffer is too short") + } + return i + len(lit) +} + +// emitCopy writes a copy chunk and returns the number of bytes written. +func emitCopy(dst []byte, offset, length int) int { + i := 0 + for length > 0 { + x := length - 4 + if 0 <= x && x < 1<<3 && offset < 1<<11 { + dst[i+0] = uint8(offset>>8)&0x07<<5 | uint8(x)<<2 | tagCopy1 + dst[i+1] = uint8(offset) + i += 2 + break + } + + x = length + if x > 1<<6 { + x = 1 << 6 + } + dst[i+0] = uint8(x-1)<<2 | tagCopy2 + dst[i+1] = uint8(offset) + dst[i+2] = uint8(offset >> 8) + i += 3 + length -= x + } + return i +} + +// Encode returns the encoded form of src. The returned slice may be a sub- +// slice of dst if dst was large enough to hold the entire encoded block. +// Otherwise, a newly allocated slice will be returned. +// It is valid to pass a nil dst. +func Encode(dst, src []byte) []byte { + if n := MaxEncodedLen(len(src)); len(dst) < n { + dst = make([]byte, n) + } + + // The block starts with the varint-encoded length of the decompressed bytes. + d := binary.PutUvarint(dst, uint64(len(src))) + + // Return early if src is short. + if len(src) <= 4 { + if len(src) != 0 { + d += emitLiteral(dst[d:], src) + } + return dst[:d] + } + + // Initialize the hash table. Its size ranges from 1<<8 to 1<<14 inclusive. + const maxTableSize = 1 << 14 + shift, tableSize := uint(32-8), 1<<8 + for tableSize < maxTableSize && tableSize < len(src) { + shift-- + tableSize *= 2 + } + var table [maxTableSize]int + + // Iterate over the source bytes. + var ( + s int // The iterator position. + t int // The last position with the same hash as s. + lit int // The start position of any pending literal bytes. + ) + for s+3 < len(src) { + // Update the hash table. + b0, b1, b2, b3 := src[s], src[s+1], src[s+2], src[s+3] + h := uint32(b0) | uint32(b1)<<8 | uint32(b2)<<16 | uint32(b3)<<24 + p := &table[(h*0x1e35a7bd)>>shift] + // We need to to store values in [-1, inf) in table. To save + // some initialization time, (re)use the table's zero value + // and shift the values against this zero: add 1 on writes, + // subtract 1 on reads. + t, *p = *p-1, s+1 + // If t is invalid or src[s:s+4] differs from src[t:t+4], accumulate a literal byte. + if t < 0 || s-t >= maxOffset || b0 != src[t] || b1 != src[t+1] || b2 != src[t+2] || b3 != src[t+3] { + s++ + continue + } + // Otherwise, we have a match. First, emit any pending literal bytes. + if lit != s { + d += emitLiteral(dst[d:], src[lit:s]) + } + // Extend the match to be as long as possible. + s0 := s + s, t = s+4, t+4 + for s < len(src) && src[s] == src[t] { + s++ + t++ + } + // Emit the copied bytes. + d += emitCopy(dst[d:], s-t, s-s0) + lit = s + } + + // Emit any final pending literal bytes and return. + if lit != len(src) { + d += emitLiteral(dst[d:], src[lit:]) + } + return dst[:d] +} + +// MaxEncodedLen returns the maximum length of a snappy block, given its +// uncompressed length. +func MaxEncodedLen(srcLen int) int { + // Compressed data can be defined as: + // compressed := item* literal* + // item := literal* copy + // + // The trailing literal sequence has a space blowup of at most 62/60 + // since a literal of length 60 needs one tag byte + one extra byte + // for length information. + // + // Item blowup is trickier to measure. Suppose the "copy" op copies + // 4 bytes of data. Because of a special check in the encoding code, + // we produce a 4-byte copy only if the offset is < 65536. Therefore + // the copy op takes 3 bytes to encode, and this type of item leads + // to at most the 62/60 blowup for representing literals. + // + // Suppose the "copy" op copies 5 bytes of data. If the offset is big + // enough, it will take 5 bytes to encode the copy op. Therefore the + // worst case here is a one-byte literal followed by a five-byte copy. + // That is, 6 bytes of input turn into 7 bytes of "compressed" data. + // + // This last factor dominates the blowup, so the final estimate is: + return 32 + srcLen + srcLen/6 +} + +// NewWriter returns a new Writer that compresses to w, using the framing +// format described at +// https://github.com/google/snappy/blob/master/framing_format.txt +func NewWriter(w io.Writer) *Writer { + return &Writer{ + w: w, + enc: make([]byte, MaxEncodedLen(maxUncompressedChunkLen)), + } +} + +// Writer is an io.Writer than can write Snappy-compressed bytes. +type Writer struct { + w io.Writer + err error + enc []byte + buf [checksumSize + chunkHeaderSize]byte + wroteHeader bool +} + +// Reset discards the writer's state and switches the Snappy writer to write to +// w. This permits reusing a Writer rather than allocating a new one. +func (w *Writer) Reset(writer io.Writer) { + w.w = writer + w.err = nil + w.wroteHeader = false +} + +// Write satisfies the io.Writer interface. +func (w *Writer) Write(p []byte) (n int, errRet error) { + if w.err != nil { + return 0, w.err + } + if !w.wroteHeader { + copy(w.enc, magicChunk) + if _, err := w.w.Write(w.enc[:len(magicChunk)]); err != nil { + w.err = err + return n, err + } + w.wroteHeader = true + } + for len(p) > 0 { + var uncompressed []byte + if len(p) > maxUncompressedChunkLen { + uncompressed, p = p[:maxUncompressedChunkLen], p[maxUncompressedChunkLen:] + } else { + uncompressed, p = p, nil + } + checksum := crc(uncompressed) + + // Compress the buffer, discarding the result if the improvement + // isn't at least 12.5%. + chunkType := uint8(chunkTypeCompressedData) + chunkBody := Encode(w.enc, uncompressed) + if len(chunkBody) >= len(uncompressed)-len(uncompressed)/8 { + chunkType, chunkBody = chunkTypeUncompressedData, uncompressed + } + + chunkLen := 4 + len(chunkBody) + w.buf[0] = chunkType + w.buf[1] = uint8(chunkLen >> 0) + w.buf[2] = uint8(chunkLen >> 8) + w.buf[3] = uint8(chunkLen >> 16) + w.buf[4] = uint8(checksum >> 0) + w.buf[5] = uint8(checksum >> 8) + w.buf[6] = uint8(checksum >> 16) + w.buf[7] = uint8(checksum >> 24) + if _, err := w.w.Write(w.buf[:]); err != nil { + w.err = err + return n, err + } + if _, err := w.w.Write(chunkBody); err != nil { + w.err = err + return n, err + } + n += len(uncompressed) + } + return n, nil +} diff --git a/Godeps/_workspace/src/github.com/golang/snappy/snappy.go b/Godeps/_workspace/src/github.com/golang/snappy/snappy.go new file mode 100644 index 0000000000000..e98653acff4ca --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/snappy/snappy.go @@ -0,0 +1,68 @@ +// Copyright 2011 The Snappy-Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// Package snappy implements the snappy block-based compression format. +// It aims for very high speeds and reasonable compression. +// +// The C++ snappy implementation is at https://github.com/google/snappy +package snappy + +import ( + "hash/crc32" +) + +/* +Each encoded block begins with the varint-encoded length of the decoded data, +followed by a sequence of chunks. Chunks begin and end on byte boundaries. The +first byte of each chunk is broken into its 2 least and 6 most significant bits +called l and m: l ranges in [0, 4) and m ranges in [0, 64). l is the chunk tag. +Zero means a literal tag. All other values mean a copy tag. + +For literal tags: + - If m < 60, the next 1 + m bytes are literal bytes. + - Otherwise, let n be the little-endian unsigned integer denoted by the next + m - 59 bytes. The next 1 + n bytes after that are literal bytes. + +For copy tags, length bytes are copied from offset bytes ago, in the style of +Lempel-Ziv compression algorithms. In particular: + - For l == 1, the offset ranges in [0, 1<<11) and the length in [4, 12). + The length is 4 + the low 3 bits of m. The high 3 bits of m form bits 8-10 + of the offset. The next byte is bits 0-7 of the offset. + - For l == 2, the offset ranges in [0, 1<<16) and the length in [1, 65). + The length is 1 + m. The offset is the little-endian unsigned integer + denoted by the next 2 bytes. + - For l == 3, this tag is a legacy format that is no longer supported. +*/ +const ( + tagLiteral = 0x00 + tagCopy1 = 0x01 + tagCopy2 = 0x02 + tagCopy4 = 0x03 +) + +const ( + checksumSize = 4 + chunkHeaderSize = 4 + magicChunk = "\xff\x06\x00\x00" + magicBody + magicBody = "sNaPpY" + // https://github.com/google/snappy/blob/master/framing_format.txt says + // that "the uncompressed data in a chunk must be no longer than 65536 bytes". + maxUncompressedChunkLen = 65536 +) + +const ( + chunkTypeCompressedData = 0x00 + chunkTypeUncompressedData = 0x01 + chunkTypePadding = 0xfe + chunkTypeStreamIdentifier = 0xff +) + +var crcTable = crc32.MakeTable(crc32.Castagnoli) + +// crc implements the checksum specified in section 3 of +// https://github.com/google/snappy/blob/master/framing_format.txt +func crc(b []byte) uint32 { + c := crc32.Update(0, crcTable, b) + return uint32(c>>15|c<<17) + 0xa282ead8 +} diff --git a/Godeps/_workspace/src/github.com/golang/snappy/snappy_test.go b/Godeps/_workspace/src/github.com/golang/snappy/snappy_test.go new file mode 100644 index 0000000000000..f8188f11eba82 --- /dev/null +++ b/Godeps/_workspace/src/github.com/golang/snappy/snappy_test.go @@ -0,0 +1,377 @@ +// Copyright 2011 The Snappy-Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package snappy + +import ( + "bytes" + "flag" + "fmt" + "io" + "io/ioutil" + "math/rand" + "net/http" + "os" + "path/filepath" + "strings" + "testing" +) + +var ( + download = flag.Bool("download", false, "If true, download any missing files before running benchmarks") + testdata = flag.String("testdata", "testdata", "Directory containing the test data") +) + +func roundtrip(b, ebuf, dbuf []byte) error { + d, err := Decode(dbuf, Encode(ebuf, b)) + if err != nil { + return fmt.Errorf("decoding error: %v", err) + } + if !bytes.Equal(b, d) { + return fmt.Errorf("roundtrip mismatch:\n\twant %v\n\tgot %v", b, d) + } + return nil +} + +func TestEmpty(t *testing.T) { + if err := roundtrip(nil, nil, nil); err != nil { + t.Fatal(err) + } +} + +func TestSmallCopy(t *testing.T) { + for _, ebuf := range [][]byte{nil, make([]byte, 20), make([]byte, 64)} { + for _, dbuf := range [][]byte{nil, make([]byte, 20), make([]byte, 64)} { + for i := 0; i < 32; i++ { + s := "aaaa" + strings.Repeat("b", i) + "aaaabbbb" + if err := roundtrip([]byte(s), ebuf, dbuf); err != nil { + t.Errorf("len(ebuf)=%d, len(dbuf)=%d, i=%d: %v", len(ebuf), len(dbuf), i, err) + } + } + } + } +} + +func TestSmallRand(t *testing.T) { + rng := rand.New(rand.NewSource(27354294)) + for n := 1; n < 20000; n += 23 { + b := make([]byte, n) + for i := range b { + b[i] = uint8(rng.Uint32()) + } + if err := roundtrip(b, nil, nil); err != nil { + t.Fatal(err) + } + } +} + +func TestSmallRegular(t *testing.T) { + for n := 1; n < 20000; n += 23 { + b := make([]byte, n) + for i := range b { + b[i] = uint8(i%10 + 'a') + } + if err := roundtrip(b, nil, nil); err != nil { + t.Fatal(err) + } + } +} + +func TestInvalidVarint(t *testing.T) { + data := []byte("\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\x00") + if _, err := DecodedLen(data); err != ErrCorrupt { + t.Errorf("DecodedLen: got %v, want ErrCorrupt", err) + } + if _, err := Decode(nil, data); err != ErrCorrupt { + t.Errorf("Decode: got %v, want ErrCorrupt", err) + } + + // The encoded varint overflows 32 bits + data = []byte("\xff\xff\xff\xff\xff\x00") + + if _, err := DecodedLen(data); err != ErrCorrupt { + t.Errorf("DecodedLen: got %v, want ErrCorrupt", err) + } + if _, err := Decode(nil, data); err != ErrCorrupt { + t.Errorf("Decode: got %v, want ErrCorrupt", err) + } +} + +func cmp(a, b []byte) error { + if len(a) != len(b) { + return fmt.Errorf("got %d bytes, want %d", len(a), len(b)) + } + for i := range a { + if a[i] != b[i] { + return fmt.Errorf("byte #%d: got 0x%02x, want 0x%02x", i, a[i], b[i]) + } + } + return nil +} + +func TestFramingFormat(t *testing.T) { + // src is comprised of alternating 1e5-sized sequences of random + // (incompressible) bytes and repeated (compressible) bytes. 1e5 was chosen + // because it is larger than maxUncompressedChunkLen (64k). + src := make([]byte, 1e6) + rng := rand.New(rand.NewSource(1)) + for i := 0; i < 10; i++ { + if i%2 == 0 { + for j := 0; j < 1e5; j++ { + src[1e5*i+j] = uint8(rng.Intn(256)) + } + } else { + for j := 0; j < 1e5; j++ { + src[1e5*i+j] = uint8(i) + } + } + } + + buf := new(bytes.Buffer) + if _, err := NewWriter(buf).Write(src); err != nil { + t.Fatalf("Write: encoding: %v", err) + } + dst, err := ioutil.ReadAll(NewReader(buf)) + if err != nil { + t.Fatalf("ReadAll: decoding: %v", err) + } + if err := cmp(dst, src); err != nil { + t.Fatal(err) + } +} + +func TestReaderReset(t *testing.T) { + gold := bytes.Repeat([]byte("All that is gold does not glitter,\n"), 10000) + buf := new(bytes.Buffer) + if _, err := NewWriter(buf).Write(gold); err != nil { + t.Fatalf("Write: %v", err) + } + encoded, invalid, partial := buf.String(), "invalid", "partial" + r := NewReader(nil) + for i, s := range []string{encoded, invalid, partial, encoded, partial, invalid, encoded, encoded} { + if s == partial { + r.Reset(strings.NewReader(encoded)) + if _, err := r.Read(make([]byte, 101)); err != nil { + t.Errorf("#%d: %v", i, err) + continue + } + continue + } + r.Reset(strings.NewReader(s)) + got, err := ioutil.ReadAll(r) + switch s { + case encoded: + if err != nil { + t.Errorf("#%d: %v", i, err) + continue + } + if err := cmp(got, gold); err != nil { + t.Errorf("#%d: %v", i, err) + continue + } + case invalid: + if err == nil { + t.Errorf("#%d: got nil error, want non-nil", i) + continue + } + } + } +} + +func TestWriterReset(t *testing.T) { + gold := bytes.Repeat([]byte("Not all those who wander are lost;\n"), 10000) + var gots, wants [][]byte + const n = 20 + w, failed := NewWriter(nil), false + for i := 0; i <= n; i++ { + buf := new(bytes.Buffer) + w.Reset(buf) + want := gold[:len(gold)*i/n] + if _, err := w.Write(want); err != nil { + t.Errorf("#%d: Write: %v", i, err) + failed = true + continue + } + got, err := ioutil.ReadAll(NewReader(buf)) + if err != nil { + t.Errorf("#%d: ReadAll: %v", i, err) + failed = true + continue + } + gots = append(gots, got) + wants = append(wants, want) + } + if failed { + return + } + for i := range gots { + if err := cmp(gots[i], wants[i]); err != nil { + t.Errorf("#%d: %v", i, err) + } + } +} + +func benchDecode(b *testing.B, src []byte) { + encoded := Encode(nil, src) + // Bandwidth is in amount of uncompressed data. + b.SetBytes(int64(len(src))) + b.ResetTimer() + for i := 0; i < b.N; i++ { + Decode(src, encoded) + } +} + +func benchEncode(b *testing.B, src []byte) { + // Bandwidth is in amount of uncompressed data. + b.SetBytes(int64(len(src))) + dst := make([]byte, MaxEncodedLen(len(src))) + b.ResetTimer() + for i := 0; i < b.N; i++ { + Encode(dst, src) + } +} + +func readFile(b testing.TB, filename string) []byte { + src, err := ioutil.ReadFile(filename) + if err != nil { + b.Skipf("skipping benchmark: %v", err) + } + if len(src) == 0 { + b.Fatalf("%s has zero length", filename) + } + return src +} + +// expand returns a slice of length n containing repeated copies of src. +func expand(src []byte, n int) []byte { + dst := make([]byte, n) + for x := dst; len(x) > 0; { + i := copy(x, src) + x = x[i:] + } + return dst +} + +func benchWords(b *testing.B, n int, decode bool) { + // Note: the file is OS-language dependent so the resulting values are not + // directly comparable for non-US-English OS installations. + data := expand(readFile(b, "/usr/share/dict/words"), n) + if decode { + benchDecode(b, data) + } else { + benchEncode(b, data) + } +} + +func BenchmarkWordsDecode1e3(b *testing.B) { benchWords(b, 1e3, true) } +func BenchmarkWordsDecode1e4(b *testing.B) { benchWords(b, 1e4, true) } +func BenchmarkWordsDecode1e5(b *testing.B) { benchWords(b, 1e5, true) } +func BenchmarkWordsDecode1e6(b *testing.B) { benchWords(b, 1e6, true) } +func BenchmarkWordsEncode1e3(b *testing.B) { benchWords(b, 1e3, false) } +func BenchmarkWordsEncode1e4(b *testing.B) { benchWords(b, 1e4, false) } +func BenchmarkWordsEncode1e5(b *testing.B) { benchWords(b, 1e5, false) } +func BenchmarkWordsEncode1e6(b *testing.B) { benchWords(b, 1e6, false) } + +// testFiles' values are copied directly from +// https://raw.githubusercontent.com/google/snappy/master/snappy_unittest.cc +// The label field is unused in snappy-go. +var testFiles = []struct { + label string + filename string +}{ + {"html", "html"}, + {"urls", "urls.10K"}, + {"jpg", "fireworks.jpeg"}, + {"jpg_200", "fireworks.jpeg"}, + {"pdf", "paper-100k.pdf"}, + {"html4", "html_x_4"}, + {"txt1", "alice29.txt"}, + {"txt2", "asyoulik.txt"}, + {"txt3", "lcet10.txt"}, + {"txt4", "plrabn12.txt"}, + {"pb", "geo.protodata"}, + {"gaviota", "kppkn.gtb"}, +} + +// The test data files are present at this canonical URL. +const baseURL = "https://raw.githubusercontent.com/google/snappy/master/testdata/" + +func downloadTestdata(b *testing.B, basename string) (errRet error) { + filename := filepath.Join(*testdata, basename) + if stat, err := os.Stat(filename); err == nil && stat.Size() != 0 { + return nil + } + + if !*download { + b.Skipf("test data not found; skipping benchmark without the -download flag") + } + // Download the official snappy C++ implementation reference test data + // files for benchmarking. + if err := os.Mkdir(*testdata, 0777); err != nil && !os.IsExist(err) { + return fmt.Errorf("failed to create testdata: %s", err) + } + + f, err := os.Create(filename) + if err != nil { + return fmt.Errorf("failed to create %s: %s", filename, err) + } + defer f.Close() + defer func() { + if errRet != nil { + os.Remove(filename) + } + }() + url := baseURL + basename + resp, err := http.Get(url) + if err != nil { + return fmt.Errorf("failed to download %s: %s", url, err) + } + defer resp.Body.Close() + if s := resp.StatusCode; s != http.StatusOK { + return fmt.Errorf("downloading %s: HTTP status code %d (%s)", url, s, http.StatusText(s)) + } + _, err = io.Copy(f, resp.Body) + if err != nil { + return fmt.Errorf("failed to download %s to %s: %s", url, filename, err) + } + return nil +} + +func benchFile(b *testing.B, n int, decode bool) { + if err := downloadTestdata(b, testFiles[n].filename); err != nil { + b.Fatalf("failed to download testdata: %s", err) + } + data := readFile(b, filepath.Join(*testdata, testFiles[n].filename)) + if decode { + benchDecode(b, data) + } else { + benchEncode(b, data) + } +} + +// Naming convention is kept similar to what snappy's C++ implementation uses. +func Benchmark_UFlat0(b *testing.B) { benchFile(b, 0, true) } +func Benchmark_UFlat1(b *testing.B) { benchFile(b, 1, true) } +func Benchmark_UFlat2(b *testing.B) { benchFile(b, 2, true) } +func Benchmark_UFlat3(b *testing.B) { benchFile(b, 3, true) } +func Benchmark_UFlat4(b *testing.B) { benchFile(b, 4, true) } +func Benchmark_UFlat5(b *testing.B) { benchFile(b, 5, true) } +func Benchmark_UFlat6(b *testing.B) { benchFile(b, 6, true) } +func Benchmark_UFlat7(b *testing.B) { benchFile(b, 7, true) } +func Benchmark_UFlat8(b *testing.B) { benchFile(b, 8, true) } +func Benchmark_UFlat9(b *testing.B) { benchFile(b, 9, true) } +func Benchmark_UFlat10(b *testing.B) { benchFile(b, 10, true) } +func Benchmark_UFlat11(b *testing.B) { benchFile(b, 11, true) } +func Benchmark_ZFlat0(b *testing.B) { benchFile(b, 0, false) } +func Benchmark_ZFlat1(b *testing.B) { benchFile(b, 1, false) } +func Benchmark_ZFlat2(b *testing.B) { benchFile(b, 2, false) } +func Benchmark_ZFlat3(b *testing.B) { benchFile(b, 3, false) } +func Benchmark_ZFlat4(b *testing.B) { benchFile(b, 4, false) } +func Benchmark_ZFlat5(b *testing.B) { benchFile(b, 5, false) } +func Benchmark_ZFlat6(b *testing.B) { benchFile(b, 6, false) } +func Benchmark_ZFlat7(b *testing.B) { benchFile(b, 7, false) } +func Benchmark_ZFlat8(b *testing.B) { benchFile(b, 8, false) } +func Benchmark_ZFlat9(b *testing.B) { benchFile(b, 9, false) } +func Benchmark_ZFlat10(b *testing.B) { benchFile(b, 10, false) } +func Benchmark_ZFlat11(b *testing.B) { benchFile(b, 11, false) } diff --git a/Godeps/_workspace/src/github.com/gonuts/go-shellquote/README b/Godeps/_workspace/src/github.com/gonuts/go-shellquote/README new file mode 100644 index 0000000000000..4d34e87afc33e --- /dev/null +++ b/Godeps/_workspace/src/github.com/gonuts/go-shellquote/README @@ -0,0 +1,36 @@ +PACKAGE + +package shellquote + import "github.com/kballard/go-shellquote" + + Shellquote provides utilities for joining/splitting strings using sh's + word-splitting rules. + +VARIABLES + +var ( + UnterminatedSingleQuoteError = errors.New("Unterminated single-quoted string") + UnterminatedDoubleQuoteError = errors.New("Unterminated double-quoted string") + UnterminatedEscapeError = errors.New("Unterminated backslash-escape") +) + + +FUNCTIONS + +func Join(args ...string) string + Join quotes each argument and joins them with a space. If passed to + /bin/sh, the resulting string will be split back into the original + arguments. + +func Split(input string) (words []string, err error) + Split splits a string according to /bin/sh's word-splitting rules. It + supports backslash-escapes, single-quotes, and double-quotes. Notably it + does not support the $'' style of quoting. It also doesn't attempt to + perform any other sort of expansion, including brace expansion, shell + expansion, or pathname expansion. + + If the given input has an unterminated quoted string or ends in a + backslash-escape, one of UnterminatedSingleQuoteError, + UnterminatedDoubleQuoteError, or UnterminatedEscapeError is returned. + + diff --git a/Godeps/_workspace/src/github.com/gonuts/go-shellquote/both_test.go b/Godeps/_workspace/src/github.com/gonuts/go-shellquote/both_test.go new file mode 100644 index 0000000000000..9cba3c84919da --- /dev/null +++ b/Godeps/_workspace/src/github.com/gonuts/go-shellquote/both_test.go @@ -0,0 +1,29 @@ +package shellquote + +import ( + "reflect" + "testing" + "testing/quick" +) + +// this is called bothtest because it tests Split and Join together + +func TestJoinSplit(t *testing.T) { + f := func(strs []string) bool { + // Join, then split, the input + combined := Join(strs...) + split, err := Split(combined) + if err != nil { + t.Logf("Error splitting %#v: %v", combined, err) + return false + } + if !reflect.DeepEqual(strs, split) { + t.Logf("Input %q did not match output %q", strs, split) + return false + } + return true + } + if err := quick.Check(f, nil); err != nil { + t.Error(err) + } +} diff --git a/Godeps/_workspace/src/github.com/gonuts/go-shellquote/doc.go b/Godeps/_workspace/src/github.com/gonuts/go-shellquote/doc.go new file mode 100644 index 0000000000000..9445fa4ad990e --- /dev/null +++ b/Godeps/_workspace/src/github.com/gonuts/go-shellquote/doc.go @@ -0,0 +1,3 @@ +// Shellquote provides utilities for joining/splitting strings using sh's +// word-splitting rules. +package shellquote diff --git a/Godeps/_workspace/src/github.com/gonuts/go-shellquote/quote.go b/Godeps/_workspace/src/github.com/gonuts/go-shellquote/quote.go new file mode 100644 index 0000000000000..f6cacee0f9a22 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gonuts/go-shellquote/quote.go @@ -0,0 +1,102 @@ +package shellquote + +import ( + "bytes" + "strings" + "unicode/utf8" +) + +// Join quotes each argument and joins them with a space. +// If passed to /bin/sh, the resulting string will be split back into the +// original arguments. +func Join(args ...string) string { + var buf bytes.Buffer + for i, arg := range args { + if i != 0 { + buf.WriteByte(' ') + } + quote(arg, &buf) + } + return buf.String() +} + +const ( + specialChars = "\\'\"`${[|&;<>()*?!" + extraSpecialChars = " \t\n" + prefixChars = "~" +) + +func quote(word string, buf *bytes.Buffer) { + // We want to try to produce a "nice" output. As such, we will + // backslash-escape most characters, but if we encounter a space, or if we + // encounter an extra-special char (which doesn't work with + // backslash-escaping) we switch over to quoting the whole word. We do this + // with a space because it's typically easier for people to read multi-word + // arguments when quoted with a space rather than with ugly backslashes + // everywhere. + origLen := buf.Len() + + if len(word) == 0 { + // oops, no content + buf.WriteString("''") + return + } + + cur, prev := word, word + atStart := true + for len(cur) > 0 { + c, l := utf8.DecodeRuneInString(cur) + cur = cur[l:] + if strings.ContainsRune(specialChars, c) || (atStart && strings.ContainsRune(prefixChars, c)) { + // copy the non-special chars up to this point + if len(cur) < len(prev) { + buf.WriteString(word[0 : len(prev)-len(cur)-l]) + } + buf.WriteByte('\\') + buf.WriteRune(c) + prev = cur + } else if strings.ContainsRune(extraSpecialChars, c) { + // start over in quote mode + buf.Truncate(origLen) + goto quote + } + atStart = false + } + if len(prev) > 0 { + buf.WriteString(prev) + } + return + +quote: + // quote mode + // Use single-quotes, but if we find a single-quote in the word, we need + // to terminate the string, emit an escaped quote, and start the string up + // again + inQuote := false + for len(word) > 0 { + i := strings.IndexRune(word, '\'') + if i == -1 { + break + } + if i > 0 { + if !inQuote { + buf.WriteByte('\'') + inQuote = true + } + buf.WriteString(word[0:i]) + word = word[i+1:] + } + if inQuote { + buf.WriteByte('\'') + inQuote = false + } + buf.WriteString("\\'") + } + if len(word) > 0 { + if !inQuote { + buf.WriteByte('\'') + } + buf.WriteString(word) + buf.WriteByte('\'') + } +} diff --git a/Godeps/_workspace/src/github.com/gonuts/go-shellquote/quote_test.go b/Godeps/_workspace/src/github.com/gonuts/go-shellquote/quote_test.go new file mode 100644 index 0000000000000..a4d2d82fb2872 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gonuts/go-shellquote/quote_test.go @@ -0,0 +1,28 @@ +package shellquote + +import ( + "testing" +) + +func TestSimpleJoin(t *testing.T) { + for _, elem := range simpleJoinTest { + output := Join(elem.input...) + if output != elem.output { + t.Errorf("Input %q, got %q, expected %q", elem.input, output, elem.output) + } + } +} + +var simpleJoinTest = []struct { + input []string + output string +}{ + {[]string{"test"}, "test"}, + {[]string{"hello goodbye"}, "'hello goodbye'"}, + {[]string{"hello", "goodbye"}, "hello goodbye"}, + {[]string{"don't you know the dewey decimal system?"}, "'don'\\''t you know the dewey decimal system?'"}, + {[]string{"don't", "you", "know", "the", "dewey", "decimal", "system?"}, "don\\'t you know the dewey decimal system\\?"}, + {[]string{"~user", "u~ser", " ~user", "!~user"}, "\\~user u~ser ' ~user' \\!~user"}, + {[]string{"foo*", "M{ovies,usic}", "ab[cd]", "%3"}, "foo\\* M\\{ovies,usic} ab\\[cd] %3"}, + {[]string{"one", "", "three"}, "one '' three"}, +} diff --git a/Godeps/_workspace/src/github.com/gonuts/go-shellquote/unquote.go b/Godeps/_workspace/src/github.com/gonuts/go-shellquote/unquote.go new file mode 100644 index 0000000000000..ba3a0f2271b97 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gonuts/go-shellquote/unquote.go @@ -0,0 +1,144 @@ +package shellquote + +import ( + "bytes" + "errors" + "strings" + "unicode/utf8" +) + +var ( + UnterminatedSingleQuoteError = errors.New("Unterminated single-quoted string") + UnterminatedDoubleQuoteError = errors.New("Unterminated double-quoted string") + UnterminatedEscapeError = errors.New("Unterminated backslash-escape") +) + +var ( + splitChars = " \n\t" + singleChar = '\'' + doubleChar = '"' + escapeChar = '\\' + doubleEscapeChars = "$`\"\n\\" +) + +// Split splits a string according to /bin/sh's word-splitting rules. It +// supports backslash-escapes, single-quotes, and double-quotes. Notably it does +// not support the $'' style of quoting. It also doesn't attempt to perform any +// other sort of expansion, including brace expansion, shell expansion, or +// pathname expansion. +// +// If the given input has an unterminated quoted string or ends in a +// backslash-escape, one of UnterminatedSingleQuoteError, +// UnterminatedDoubleQuoteError, or UnterminatedEscapeError is returned. +func Split(input string) (words []string, err error) { + var buf bytes.Buffer + words = make([]string, 0) + + for len(input) > 0 { + // skip any splitChars at the start + c, l := utf8.DecodeRuneInString(input) + if strings.ContainsRune(splitChars, c) { + input = input[l:] + continue + } + + var word string + word, input, err = splitWord(input, &buf) + if err != nil { + return + } + words = append(words, word) + } + return +} + +func splitWord(input string, buf *bytes.Buffer) (word string, remainder string, err error) { + buf.Reset() + +raw: + { + cur := input + for len(cur) > 0 { + c, l := utf8.DecodeRuneInString(cur) + cur = cur[l:] + if c == singleChar { + buf.WriteString(input[0 : len(input)-len(cur)-l]) + input = cur + goto single + } else if c == doubleChar { + buf.WriteString(input[0 : len(input)-len(cur)-l]) + input = cur + goto double + } else if c == escapeChar { + buf.WriteString(input[0 : len(input)-len(cur)-l]) + input = cur + goto escape + } else if strings.ContainsRune(splitChars, c) { + buf.WriteString(input[0 : len(input)-len(cur)-l]) + return buf.String(), cur, nil + } + } + if len(input) > 0 { + buf.WriteString(input) + input = "" + } + goto done + } + +escape: + { + if len(input) == 0 { + return "", "", UnterminatedEscapeError + } + c, l := utf8.DecodeRuneInString(input) + if c == '\n' { + // a backslash-escaped newline is elided from the output entirely + } else { + buf.WriteString(input[:l]) + } + input = input[l:] + } + goto raw + +single: + { + i := strings.IndexRune(input, singleChar) + if i == -1 { + return "", "", UnterminatedSingleQuoteError + } + buf.WriteString(input[0:i]) + input = input[i+1:] + goto raw + } + +double: + { + cur := input + for len(cur) > 0 { + c, l := utf8.DecodeRuneInString(cur) + cur = cur[l:] + if c == doubleChar { + buf.WriteString(input[0 : len(input)-len(cur)-l]) + input = cur + goto raw + } else if c == escapeChar { + // bash only supports certain escapes in double-quoted strings + c2, l2 := utf8.DecodeRuneInString(cur) + cur = cur[l2:] + if strings.ContainsRune(doubleEscapeChars, c2) { + buf.WriteString(input[0 : len(input)-len(cur)-l-l2]) + if c2 == '\n' { + // newline is special, skip the backslash entirely + } else { + buf.WriteRune(c2) + } + input = cur + } + } + } + return "", "", UnterminatedDoubleQuoteError + } + +done: + return buf.String(), input, nil +} diff --git a/Godeps/_workspace/src/github.com/gonuts/go-shellquote/unquote_test.go b/Godeps/_workspace/src/github.com/gonuts/go-shellquote/unquote_test.go new file mode 100644 index 0000000000000..32ea5144bb4a6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/gonuts/go-shellquote/unquote_test.go @@ -0,0 +1,52 @@ +package shellquote + +import ( + "reflect" + "testing" +) + +func TestSimpleSplit(t *testing.T) { + for _, elem := range simpleSplitTest { + output, err := Split(elem.input) + if err != nil { + t.Errorf("Input %q, got error %#v", elem.input, err) + } else if !reflect.DeepEqual(output, elem.output) { + t.Errorf("Input %q, got %q, expected %q", elem.input, output, elem.output) + } + } +} + +func TestErrorSplit(t *testing.T) { + for _, elem := range errorSplitTest { + _, err := Split(elem.input) + if err != elem.error { + t.Errorf("Input %q, got error %#v, expected error %#v", elem.input, err, elem.error) + } + } +} + +var simpleSplitTest = []struct { + input string + output []string +}{ + {"hello", []string{"hello"}}, + {"hello goodbye", []string{"hello", "goodbye"}}, + {"hello goodbye", []string{"hello", "goodbye"}}, + {"glob* test?", []string{"glob*", "test?"}}, + {"don\\'t you know the dewey decimal system\\?", []string{"don't", "you", "know", "the", "dewey", "decimal", "system?"}}, + {"'don'\\''t you know the dewey decimal system?'", []string{"don't you know the dewey decimal system?"}}, + {"one '' two", []string{"one", "", "two"}}, + {"text with\\\na newline", []string{"text", "witha", "newline"}}, + {"\"quoted\\d\\\\\\\" text with a\\\nnewline\"", []string{"quoted\\d\\\" text with anewline"}}, + {"foo\"bar\"baz", []string{"foobarbaz"}}, +} + +var errorSplitTest = []struct { + input string + error error +}{ + {"don't worry", UnterminatedSingleQuoteError}, + {"'test'\\''ing", UnterminatedSingleQuoteError}, + {"\"foo'bar", UnterminatedDoubleQuoteError}, + {"foo\\", UnterminatedEscapeError}, +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/0doc.go b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/0doc.go new file mode 100644 index 0000000000000..c14d810a73e81 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/0doc.go @@ -0,0 +1,143 @@ +// Copyright (c) 2012, 2013 Ugorji Nwoke. All rights reserved. +// Use of this source code is governed by a BSD-style license found in the LICENSE file. + +/* +High Performance, Feature-Rich Idiomatic Go encoding library for msgpack and binc . + +Supported Serialization formats are: + + - msgpack: [https://github.com/msgpack/msgpack] + - binc: [http://github.com/ugorji/binc] + +To install: + + go get github.com/ugorji/go/codec + +The idiomatic Go support is as seen in other encoding packages in +the standard library (ie json, xml, gob, etc). + +Rich Feature Set includes: + + - Simple but extremely powerful and feature-rich API + - Very High Performance. + Our extensive benchmarks show us outperforming Gob, Json and Bson by 2-4X. + This was achieved by taking extreme care on: + - managing allocation + - function frame size (important due to Go's use of split stacks), + - reflection use (and by-passing reflection for common types) + - recursion implications + - zero-copy mode (encoding/decoding to byte slice without using temp buffers) + - Correct. + Care was taken to precisely handle corner cases like: + overflows, nil maps and slices, nil value in stream, etc. + - Efficient zero-copying into temporary byte buffers + when encoding into or decoding from a byte slice. + - Standard field renaming via tags + - Encoding from any value + (struct, slice, map, primitives, pointers, interface{}, etc) + - Decoding into pointer to any non-nil typed value + (struct, slice, map, int, float32, bool, string, reflect.Value, etc) + - Supports extension functions to handle the encode/decode of custom types + - Support Go 1.2 encoding.BinaryMarshaler/BinaryUnmarshaler + - Schema-less decoding + (decode into a pointer to a nil interface{} as opposed to a typed non-nil value). + Includes Options to configure what specific map or slice type to use + when decoding an encoded list or map into a nil interface{} + - Provides a RPC Server and Client Codec for net/rpc communication protocol. + - Msgpack Specific: + - Provides extension functions to handle spec-defined extensions (binary, timestamp) + - Options to resolve ambiguities in handling raw bytes (as string or []byte) + during schema-less decoding (decoding into a nil interface{}) + - RPC Server/Client Codec for msgpack-rpc protocol defined at: + https://github.com/msgpack-rpc/msgpack-rpc/blob/master/spec.md + - Fast Paths for some container types: + For some container types, we circumvent reflection and its associated overhead + and allocation costs, and encode/decode directly. These types are: + []interface{} + []int + []string + map[interface{}]interface{} + map[int]interface{} + map[string]interface{} + +Extension Support + +Users can register a function to handle the encoding or decoding of +their custom types. + +There are no restrictions on what the custom type can be. Some examples: + + type BisSet []int + type BitSet64 uint64 + type UUID string + type MyStructWithUnexportedFields struct { a int; b bool; c []int; } + type GifImage struct { ... } + +As an illustration, MyStructWithUnexportedFields would normally be +encoded as an empty map because it has no exported fields, while UUID +would be encoded as a string. However, with extension support, you can +encode any of these however you like. + +RPC + +RPC Client and Server Codecs are implemented, so the codecs can be used +with the standard net/rpc package. + +Usage + +Typical usage model: + + // create and configure Handle + var ( + bh codec.BincHandle + mh codec.MsgpackHandle + ) + + mh.MapType = reflect.TypeOf(map[string]interface{}(nil)) + + // configure extensions + // e.g. for msgpack, define functions and enable Time support for tag 1 + // mh.AddExt(reflect.TypeOf(time.Time{}), 1, myMsgpackTimeEncodeExtFn, myMsgpackTimeDecodeExtFn) + + // create and use decoder/encoder + var ( + r io.Reader + w io.Writer + b []byte + h = &bh // or mh to use msgpack + ) + + dec = codec.NewDecoder(r, h) + dec = codec.NewDecoderBytes(b, h) + err = dec.Decode(&v) + + enc = codec.NewEncoder(w, h) + enc = codec.NewEncoderBytes(&b, h) + err = enc.Encode(v) + + //RPC Server + go func() { + for { + conn, err := listener.Accept() + rpcCodec := codec.GoRpc.ServerCodec(conn, h) + //OR rpcCodec := codec.MsgpackSpecRpc.ServerCodec(conn, h) + rpc.ServeCodec(rpcCodec) + } + }() + + //RPC Communication (client side) + conn, err = net.Dial("tcp", "localhost:5555") + rpcCodec := codec.GoRpc.ClientCodec(conn, h) + //OR rpcCodec := codec.MsgpackSpecRpc.ClientCodec(conn, h) + client := rpc.NewClientWithCodec(rpcCodec) + +Representative Benchmark Results + +Run the benchmark suite using: + go test -bi -bench=. -benchmem + +To run full benchmark suite (including against vmsgpack and bson), +see notes in ext_dep_test.go + +*/ +package codec diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/README.md b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/README.md new file mode 100644 index 0000000000000..6c95d1bfd2081 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/README.md @@ -0,0 +1,174 @@ +# Codec + +High Performance and Feature-Rich Idiomatic Go Library providing +encode/decode support for different serialization formats. + +Supported Serialization formats are: + + - msgpack: [https://github.com/msgpack/msgpack] + - binc: [http://github.com/ugorji/binc] + +To install: + + go get github.com/ugorji/go/codec + +Online documentation: [http://godoc.org/github.com/ugorji/go/codec] + +The idiomatic Go support is as seen in other encoding packages in +the standard library (ie json, xml, gob, etc). + +Rich Feature Set includes: + + - Simple but extremely powerful and feature-rich API + - Very High Performance. + Our extensive benchmarks show us outperforming Gob, Json and Bson by 2-4X. + This was achieved by taking extreme care on: + - managing allocation + - function frame size (important due to Go's use of split stacks), + - reflection use (and by-passing reflection for common types) + - recursion implications + - zero-copy mode (encoding/decoding to byte slice without using temp buffers) + - Correct. + Care was taken to precisely handle corner cases like: + overflows, nil maps and slices, nil value in stream, etc. + - Efficient zero-copying into temporary byte buffers + when encoding into or decoding from a byte slice. + - Standard field renaming via tags + - Encoding from any value + (struct, slice, map, primitives, pointers, interface{}, etc) + - Decoding into pointer to any non-nil typed value + (struct, slice, map, int, float32, bool, string, reflect.Value, etc) + - Supports extension functions to handle the encode/decode of custom types + - Support Go 1.2 encoding.BinaryMarshaler/BinaryUnmarshaler + - Schema-less decoding + (decode into a pointer to a nil interface{} as opposed to a typed non-nil value). + Includes Options to configure what specific map or slice type to use + when decoding an encoded list or map into a nil interface{} + - Provides a RPC Server and Client Codec for net/rpc communication protocol. + - Msgpack Specific: + - Provides extension functions to handle spec-defined extensions (binary, timestamp) + - Options to resolve ambiguities in handling raw bytes (as string or []byte) + during schema-less decoding (decoding into a nil interface{}) + - RPC Server/Client Codec for msgpack-rpc protocol defined at: + https://github.com/msgpack-rpc/msgpack-rpc/blob/master/spec.md + - Fast Paths for some container types: + For some container types, we circumvent reflection and its associated overhead + and allocation costs, and encode/decode directly. These types are: + []interface{} + []int + []string + map[interface{}]interface{} + map[int]interface{} + map[string]interface{} + +## Extension Support + +Users can register a function to handle the encoding or decoding of +their custom types. + +There are no restrictions on what the custom type can be. Some examples: + + type BisSet []int + type BitSet64 uint64 + type UUID string + type MyStructWithUnexportedFields struct { a int; b bool; c []int; } + type GifImage struct { ... } + +As an illustration, MyStructWithUnexportedFields would normally be +encoded as an empty map because it has no exported fields, while UUID +would be encoded as a string. However, with extension support, you can +encode any of these however you like. + +## RPC + +RPC Client and Server Codecs are implemented, so the codecs can be used +with the standard net/rpc package. + +## Usage + +Typical usage model: + + // create and configure Handle + var ( + bh codec.BincHandle + mh codec.MsgpackHandle + ) + + mh.MapType = reflect.TypeOf(map[string]interface{}(nil)) + + // configure extensions + // e.g. for msgpack, define functions and enable Time support for tag 1 + // mh.AddExt(reflect.TypeOf(time.Time{}), 1, myMsgpackTimeEncodeExtFn, myMsgpackTimeDecodeExtFn) + + // create and use decoder/encoder + var ( + r io.Reader + w io.Writer + b []byte + h = &bh // or mh to use msgpack + ) + + dec = codec.NewDecoder(r, h) + dec = codec.NewDecoderBytes(b, h) + err = dec.Decode(&v) + + enc = codec.NewEncoder(w, h) + enc = codec.NewEncoderBytes(&b, h) + err = enc.Encode(v) + + //RPC Server + go func() { + for { + conn, err := listener.Accept() + rpcCodec := codec.GoRpc.ServerCodec(conn, h) + //OR rpcCodec := codec.MsgpackSpecRpc.ServerCodec(conn, h) + rpc.ServeCodec(rpcCodec) + } + }() + + //RPC Communication (client side) + conn, err = net.Dial("tcp", "localhost:5555") + rpcCodec := codec.GoRpc.ClientCodec(conn, h) + //OR rpcCodec := codec.MsgpackSpecRpc.ClientCodec(conn, h) + client := rpc.NewClientWithCodec(rpcCodec) + +## Representative Benchmark Results + +A sample run of benchmark using "go test -bi -bench=. -benchmem": + + /proc/cpuinfo: Intel(R) Core(TM) i7-2630QM CPU @ 2.00GHz (HT) + + .............................................. + BENCHMARK INIT: 2013-10-16 11:02:50.345970786 -0400 EDT + To run full benchmark comparing encodings (MsgPack, Binc, JSON, GOB, etc), use: "go test -bench=." + Benchmark: + Struct recursive Depth: 1 + ApproxDeepSize Of benchmark Struct: 4694 bytes + Benchmark One-Pass Run: + v-msgpack: len: 1600 bytes + bson: len: 3025 bytes + msgpack: len: 1560 bytes + binc: len: 1187 bytes + gob: len: 1972 bytes + json: len: 2538 bytes + .............................................. + PASS + Benchmark__Msgpack____Encode 50000 54359 ns/op 14953 B/op 83 allocs/op + Benchmark__Msgpack____Decode 10000 106531 ns/op 14990 B/op 410 allocs/op + Benchmark__Binc_NoSym_Encode 50000 53956 ns/op 14966 B/op 83 allocs/op + Benchmark__Binc_NoSym_Decode 10000 103751 ns/op 14529 B/op 386 allocs/op + Benchmark__Binc_Sym___Encode 50000 65961 ns/op 17130 B/op 88 allocs/op + Benchmark__Binc_Sym___Decode 10000 106310 ns/op 15857 B/op 287 allocs/op + Benchmark__Gob________Encode 10000 135944 ns/op 21189 B/op 237 allocs/op + Benchmark__Gob________Decode 5000 405390 ns/op 83460 B/op 1841 allocs/op + Benchmark__Json_______Encode 20000 79412 ns/op 13874 B/op 102 allocs/op + Benchmark__Json_______Decode 10000 247979 ns/op 14202 B/op 493 allocs/op + Benchmark__Bson_______Encode 10000 121762 ns/op 27814 B/op 514 allocs/op + Benchmark__Bson_______Decode 10000 162126 ns/op 16514 B/op 789 allocs/op + Benchmark__VMsgpack___Encode 50000 69155 ns/op 12370 B/op 344 allocs/op + Benchmark__VMsgpack___Decode 10000 151609 ns/op 20307 B/op 571 allocs/op + ok ugorji.net/codec 30.827s + +To run full benchmark suite (including against vmsgpack and bson), +see notes in ext\_dep\_test.go + diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/bench_test.go b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/bench_test.go new file mode 100644 index 0000000000000..4d437035e0faf --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/bench_test.go @@ -0,0 +1,319 @@ +// Copyright (c) 2012, 2013 Ugorji Nwoke. All rights reserved. +// Use of this source code is governed by a BSD-style license found in the LICENSE file. + +package codec + +import ( + "bytes" + "encoding/gob" + "encoding/json" + "flag" + "fmt" + "reflect" + "runtime" + "testing" + "time" +) + +// Sample way to run: +// go test -bi -bv -bd=1 -benchmem -bench=. + +var ( + _ = fmt.Printf + benchTs *TestStruc + + approxSize int + + benchDoInitBench bool + benchVerify bool + benchUnscientificRes bool = false + //depth of 0 maps to ~400bytes json-encoded string, 1 maps to ~1400 bytes, etc + //For depth>1, we likely trigger stack growth for encoders, making benchmarking unreliable. + benchDepth int + benchInitDebug bool + benchCheckers []benchChecker +) + +type benchEncFn func(interface{}) ([]byte, error) +type benchDecFn func([]byte, interface{}) error +type benchIntfFn func() interface{} + +type benchChecker struct { + name string + encodefn benchEncFn + decodefn benchDecFn +} + +func benchInitFlags() { + flag.BoolVar(&benchInitDebug, "bg", false, "Bench Debug") + flag.IntVar(&benchDepth, "bd", 1, "Bench Depth: If >1, potential unreliable results due to stack growth") + flag.BoolVar(&benchDoInitBench, "bi", false, "Run Bench Init") + flag.BoolVar(&benchVerify, "bv", false, "Verify Decoded Value during Benchmark") + flag.BoolVar(&benchUnscientificRes, "bu", false, "Show Unscientific Results during Benchmark") +} + +func benchInit() { + benchTs = newTestStruc(benchDepth, true) + approxSize = approxDataSize(reflect.ValueOf(benchTs)) + bytesLen := 1024 * 4 * (benchDepth + 1) * (benchDepth + 1) + if bytesLen < approxSize { + bytesLen = approxSize + } + + benchCheckers = append(benchCheckers, + benchChecker{"msgpack", fnMsgpackEncodeFn, fnMsgpackDecodeFn}, + benchChecker{"binc-nosym", fnBincNoSymEncodeFn, fnBincNoSymDecodeFn}, + benchChecker{"binc-sym", fnBincSymEncodeFn, fnBincSymDecodeFn}, + benchChecker{"simple", fnSimpleEncodeFn, fnSimpleDecodeFn}, + benchChecker{"gob", fnGobEncodeFn, fnGobDecodeFn}, + benchChecker{"json", fnJsonEncodeFn, fnJsonDecodeFn}, + ) + if benchDoInitBench { + runBenchInit() + } +} + +func runBenchInit() { + logT(nil, "..............................................") + logT(nil, "BENCHMARK INIT: %v", time.Now()) + logT(nil, "To run full benchmark comparing encodings (MsgPack, Binc, Simple, JSON, GOB, etc), "+ + "use: \"go test -bench=.\"") + logT(nil, "Benchmark: ") + logT(nil, "\tStruct recursive Depth: %d", benchDepth) + if approxSize > 0 { + logT(nil, "\tApproxDeepSize Of benchmark Struct: %d bytes", approxSize) + } + if benchUnscientificRes { + logT(nil, "Benchmark One-Pass Run (with Unscientific Encode/Decode times): ") + } else { + logT(nil, "Benchmark One-Pass Run:") + } + for _, bc := range benchCheckers { + doBenchCheck(bc.name, bc.encodefn, bc.decodefn) + } + logT(nil, "..............................................") + if benchInitDebug { + logT(nil, "<<<<====>>>> depth: %v, ts: %#v\n", benchDepth, benchTs) + } +} + +func fnBenchNewTs() interface{} { + return new(TestStruc) +} + +func doBenchCheck(name string, encfn benchEncFn, decfn benchDecFn) { + runtime.GC() + tnow := time.Now() + buf, err := encfn(benchTs) + if err != nil { + logT(nil, "\t%10s: **** Error encoding benchTs: %v", name, err) + } + encDur := time.Now().Sub(tnow) + encLen := len(buf) + runtime.GC() + if !benchUnscientificRes { + logT(nil, "\t%10s: len: %d bytes\n", name, encLen) + return + } + tnow = time.Now() + if err = decfn(buf, new(TestStruc)); err != nil { + logT(nil, "\t%10s: **** Error decoding into new TestStruc: %v", name, err) + } + decDur := time.Now().Sub(tnow) + logT(nil, "\t%10s: len: %d bytes, encode: %v, decode: %v\n", name, encLen, encDur, decDur) +} + +func fnBenchmarkEncode(b *testing.B, encName string, ts interface{}, encfn benchEncFn) { + runtime.GC() + b.ResetTimer() + for i := 0; i < b.N; i++ { + _, err := encfn(ts) + if err != nil { + logT(b, "Error encoding benchTs: %s: %v", encName, err) + b.FailNow() + } + } +} + +func fnBenchmarkDecode(b *testing.B, encName string, ts interface{}, + encfn benchEncFn, decfn benchDecFn, newfn benchIntfFn, +) { + buf, err := encfn(ts) + if err != nil { + logT(b, "Error encoding benchTs: %s: %v", encName, err) + b.FailNow() + } + runtime.GC() + b.ResetTimer() + for i := 0; i < b.N; i++ { + ts = newfn() + if err = decfn(buf, ts); err != nil { + logT(b, "Error decoding into new TestStruc: %s: %v", encName, err) + b.FailNow() + } + if benchVerify { + if vts, vok := ts.(*TestStruc); vok { + verifyTsTree(b, vts) + } + } + } +} + +func verifyTsTree(b *testing.B, ts *TestStruc) { + var ts0, ts1m, ts2m, ts1s, ts2s *TestStruc + ts0 = ts + + if benchDepth > 0 { + ts1m, ts1s = verifyCheckAndGet(b, ts0) + } + + if benchDepth > 1 { + ts2m, ts2s = verifyCheckAndGet(b, ts1m) + } + for _, tsx := range []*TestStruc{ts0, ts1m, ts2m, ts1s, ts2s} { + if tsx != nil { + verifyOneOne(b, tsx) + } + } +} + +func verifyCheckAndGet(b *testing.B, ts0 *TestStruc) (ts1m *TestStruc, ts1s *TestStruc) { + // if len(ts1m.Ms) <= 2 { + // logT(b, "Error: ts1m.Ms len should be > 2. Got: %v", len(ts1m.Ms)) + // b.FailNow() + // } + if len(ts0.Its) == 0 { + logT(b, "Error: ts0.Islice len should be > 0. Got: %v", len(ts0.Its)) + b.FailNow() + } + ts1m = ts0.Mtsptr["0"] + ts1s = ts0.Its[0] + if ts1m == nil || ts1s == nil { + logT(b, "Error: At benchDepth 1, No *TestStruc found") + b.FailNow() + } + return +} + +func verifyOneOne(b *testing.B, ts *TestStruc) { + if ts.I64slice[2] != int64(3) { + logT(b, "Error: Decode failed by checking values") + b.FailNow() + } +} + +func fnMsgpackEncodeFn(ts interface{}) (bs []byte, err error) { + err = NewEncoderBytes(&bs, testMsgpackH).Encode(ts) + return +} + +func fnMsgpackDecodeFn(buf []byte, ts interface{}) error { + return NewDecoderBytes(buf, testMsgpackH).Decode(ts) +} + +func fnBincEncodeFn(ts interface{}, sym AsSymbolFlag) (bs []byte, err error) { + tSym := testBincH.AsSymbols + testBincH.AsSymbols = sym + err = NewEncoderBytes(&bs, testBincH).Encode(ts) + testBincH.AsSymbols = tSym + return +} + +func fnBincDecodeFn(buf []byte, ts interface{}, sym AsSymbolFlag) (err error) { + tSym := testBincH.AsSymbols + testBincH.AsSymbols = sym + err = NewDecoderBytes(buf, testBincH).Decode(ts) + testBincH.AsSymbols = tSym + return +} + +func fnBincNoSymEncodeFn(ts interface{}) (bs []byte, err error) { + return fnBincEncodeFn(ts, AsSymbolNone) +} + +func fnBincNoSymDecodeFn(buf []byte, ts interface{}) error { + return fnBincDecodeFn(buf, ts, AsSymbolNone) +} + +func fnBincSymEncodeFn(ts interface{}) (bs []byte, err error) { + return fnBincEncodeFn(ts, AsSymbolAll) +} + +func fnBincSymDecodeFn(buf []byte, ts interface{}) error { + return fnBincDecodeFn(buf, ts, AsSymbolAll) +} + +func fnSimpleEncodeFn(ts interface{}) (bs []byte, err error) { + err = NewEncoderBytes(&bs, testSimpleH).Encode(ts) + return +} + +func fnSimpleDecodeFn(buf []byte, ts interface{}) error { + return NewDecoderBytes(buf, testSimpleH).Decode(ts) +} + +func fnGobEncodeFn(ts interface{}) ([]byte, error) { + bbuf := new(bytes.Buffer) + err := gob.NewEncoder(bbuf).Encode(ts) + return bbuf.Bytes(), err +} + +func fnGobDecodeFn(buf []byte, ts interface{}) error { + return gob.NewDecoder(bytes.NewBuffer(buf)).Decode(ts) +} + +func fnJsonEncodeFn(ts interface{}) ([]byte, error) { + return json.Marshal(ts) +} + +func fnJsonDecodeFn(buf []byte, ts interface{}) error { + return json.Unmarshal(buf, ts) +} + +func Benchmark__Msgpack____Encode(b *testing.B) { + fnBenchmarkEncode(b, "msgpack", benchTs, fnMsgpackEncodeFn) +} + +func Benchmark__Msgpack____Decode(b *testing.B) { + fnBenchmarkDecode(b, "msgpack", benchTs, fnMsgpackEncodeFn, fnMsgpackDecodeFn, fnBenchNewTs) +} + +func Benchmark__Binc_NoSym_Encode(b *testing.B) { + fnBenchmarkEncode(b, "binc", benchTs, fnBincNoSymEncodeFn) +} + +func Benchmark__Binc_NoSym_Decode(b *testing.B) { + fnBenchmarkDecode(b, "binc", benchTs, fnBincNoSymEncodeFn, fnBincNoSymDecodeFn, fnBenchNewTs) +} + +func Benchmark__Binc_Sym___Encode(b *testing.B) { + fnBenchmarkEncode(b, "binc", benchTs, fnBincSymEncodeFn) +} + +func Benchmark__Binc_Sym___Decode(b *testing.B) { + fnBenchmarkDecode(b, "binc", benchTs, fnBincSymEncodeFn, fnBincSymDecodeFn, fnBenchNewTs) +} + +func Benchmark__Simple____Encode(b *testing.B) { + fnBenchmarkEncode(b, "simple", benchTs, fnSimpleEncodeFn) +} + +func Benchmark__Simple____Decode(b *testing.B) { + fnBenchmarkDecode(b, "simple", benchTs, fnSimpleEncodeFn, fnSimpleDecodeFn, fnBenchNewTs) +} + +func Benchmark__Gob________Encode(b *testing.B) { + fnBenchmarkEncode(b, "gob", benchTs, fnGobEncodeFn) +} + +func Benchmark__Gob________Decode(b *testing.B) { + fnBenchmarkDecode(b, "gob", benchTs, fnGobEncodeFn, fnGobDecodeFn, fnBenchNewTs) +} + +func Benchmark__Json_______Encode(b *testing.B) { + fnBenchmarkEncode(b, "json", benchTs, fnJsonEncodeFn) +} + +func Benchmark__Json_______Decode(b *testing.B) { + fnBenchmarkDecode(b, "json", benchTs, fnJsonEncodeFn, fnJsonDecodeFn, fnBenchNewTs) +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/binc.go b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/binc.go new file mode 100644 index 0000000000000..2bb5e8fee8548 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/binc.go @@ -0,0 +1,786 @@ +// Copyright (c) 2012, 2013 Ugorji Nwoke. All rights reserved. +// Use of this source code is governed by a BSD-style license found in the LICENSE file. + +package codec + +import ( + "math" + // "reflect" + // "sync/atomic" + "time" + //"fmt" +) + +const bincDoPrune = true // No longer needed. Needed before as C lib did not support pruning. + +//var _ = fmt.Printf + +// vd as low 4 bits (there are 16 slots) +const ( + bincVdSpecial byte = iota + bincVdPosInt + bincVdNegInt + bincVdFloat + + bincVdString + bincVdByteArray + bincVdArray + bincVdMap + + bincVdTimestamp + bincVdSmallInt + bincVdUnicodeOther + bincVdSymbol + + bincVdDecimal + _ // open slot + _ // open slot + bincVdCustomExt = 0x0f +) + +const ( + bincSpNil byte = iota + bincSpFalse + bincSpTrue + bincSpNan + bincSpPosInf + bincSpNegInf + bincSpZeroFloat + bincSpZero + bincSpNegOne +) + +const ( + bincFlBin16 byte = iota + bincFlBin32 + _ // bincFlBin32e + bincFlBin64 + _ // bincFlBin64e + // others not currently supported +) + +type bincEncDriver struct { + w encWriter + m map[string]uint16 // symbols + s uint32 // symbols sequencer + b [8]byte +} + +func (e *bincEncDriver) isBuiltinType(rt uintptr) bool { + return rt == timeTypId +} + +func (e *bincEncDriver) encodeBuiltin(rt uintptr, v interface{}) { + switch rt { + case timeTypId: + bs := encodeTime(v.(time.Time)) + e.w.writen1(bincVdTimestamp<<4 | uint8(len(bs))) + e.w.writeb(bs) + } +} + +func (e *bincEncDriver) encodeNil() { + e.w.writen1(bincVdSpecial<<4 | bincSpNil) +} + +func (e *bincEncDriver) encodeBool(b bool) { + if b { + e.w.writen1(bincVdSpecial<<4 | bincSpTrue) + } else { + e.w.writen1(bincVdSpecial<<4 | bincSpFalse) + } +} + +func (e *bincEncDriver) encodeFloat32(f float32) { + if f == 0 { + e.w.writen1(bincVdSpecial<<4 | bincSpZeroFloat) + return + } + e.w.writen1(bincVdFloat<<4 | bincFlBin32) + e.w.writeUint32(math.Float32bits(f)) +} + +func (e *bincEncDriver) encodeFloat64(f float64) { + if f == 0 { + e.w.writen1(bincVdSpecial<<4 | bincSpZeroFloat) + return + } + bigen.PutUint64(e.b[:], math.Float64bits(f)) + if bincDoPrune { + i := 7 + for ; i >= 0 && (e.b[i] == 0); i-- { + } + i++ + if i <= 6 { + e.w.writen1(bincVdFloat<<4 | 0x8 | bincFlBin64) + e.w.writen1(byte(i)) + e.w.writeb(e.b[:i]) + return + } + } + e.w.writen1(bincVdFloat<<4 | bincFlBin64) + e.w.writeb(e.b[:]) +} + +func (e *bincEncDriver) encIntegerPrune(bd byte, pos bool, v uint64, lim uint8) { + if lim == 4 { + bigen.PutUint32(e.b[:lim], uint32(v)) + } else { + bigen.PutUint64(e.b[:lim], v) + } + if bincDoPrune { + i := pruneSignExt(e.b[:lim], pos) + e.w.writen1(bd | lim - 1 - byte(i)) + e.w.writeb(e.b[i:lim]) + } else { + e.w.writen1(bd | lim - 1) + e.w.writeb(e.b[:lim]) + } +} + +func (e *bincEncDriver) encodeInt(v int64) { + const nbd byte = bincVdNegInt << 4 + switch { + case v >= 0: + e.encUint(bincVdPosInt<<4, true, uint64(v)) + case v == -1: + e.w.writen1(bincVdSpecial<<4 | bincSpNegOne) + default: + e.encUint(bincVdNegInt<<4, false, uint64(-v)) + } +} + +func (e *bincEncDriver) encodeUint(v uint64) { + e.encUint(bincVdPosInt<<4, true, v) +} + +func (e *bincEncDriver) encUint(bd byte, pos bool, v uint64) { + switch { + case v == 0: + e.w.writen1(bincVdSpecial<<4 | bincSpZero) + case pos && v >= 1 && v <= 16: + e.w.writen1(bincVdSmallInt<<4 | byte(v-1)) + case v <= math.MaxUint8: + e.w.writen2(bd|0x0, byte(v)) + case v <= math.MaxUint16: + e.w.writen1(bd | 0x01) + e.w.writeUint16(uint16(v)) + case v <= math.MaxUint32: + e.encIntegerPrune(bd, pos, v, 4) + default: + e.encIntegerPrune(bd, pos, v, 8) + } +} + +func (e *bincEncDriver) encodeExtPreamble(xtag byte, length int) { + e.encLen(bincVdCustomExt<<4, uint64(length)) + e.w.writen1(xtag) +} + +func (e *bincEncDriver) encodeArrayPreamble(length int) { + e.encLen(bincVdArray<<4, uint64(length)) +} + +func (e *bincEncDriver) encodeMapPreamble(length int) { + e.encLen(bincVdMap<<4, uint64(length)) +} + +func (e *bincEncDriver) encodeString(c charEncoding, v string) { + l := uint64(len(v)) + e.encBytesLen(c, l) + if l > 0 { + e.w.writestr(v) + } +} + +func (e *bincEncDriver) encodeSymbol(v string) { + // if WriteSymbolsNoRefs { + // e.encodeString(c_UTF8, v) + // return + // } + + //symbols only offer benefit when string length > 1. + //This is because strings with length 1 take only 2 bytes to store + //(bd with embedded length, and single byte for string val). + + l := len(v) + switch l { + case 0: + e.encBytesLen(c_UTF8, 0) + return + case 1: + e.encBytesLen(c_UTF8, 1) + e.w.writen1(v[0]) + return + } + if e.m == nil { + e.m = make(map[string]uint16, 16) + } + ui, ok := e.m[v] + if ok { + if ui <= math.MaxUint8 { + e.w.writen2(bincVdSymbol<<4, byte(ui)) + } else { + e.w.writen1(bincVdSymbol<<4 | 0x8) + e.w.writeUint16(ui) + } + } else { + e.s++ + ui = uint16(e.s) + //ui = uint16(atomic.AddUint32(&e.s, 1)) + e.m[v] = ui + var lenprec uint8 + switch { + case l <= math.MaxUint8: + // lenprec = 0 + case l <= math.MaxUint16: + lenprec = 1 + case int64(l) <= math.MaxUint32: + lenprec = 2 + default: + lenprec = 3 + } + if ui <= math.MaxUint8 { + e.w.writen2(bincVdSymbol<<4|0x0|0x4|lenprec, byte(ui)) + } else { + e.w.writen1(bincVdSymbol<<4 | 0x8 | 0x4 | lenprec) + e.w.writeUint16(ui) + } + switch lenprec { + case 0: + e.w.writen1(byte(l)) + case 1: + e.w.writeUint16(uint16(l)) + case 2: + e.w.writeUint32(uint32(l)) + default: + e.w.writeUint64(uint64(l)) + } + e.w.writestr(v) + } +} + +func (e *bincEncDriver) encodeStringBytes(c charEncoding, v []byte) { + l := uint64(len(v)) + e.encBytesLen(c, l) + if l > 0 { + e.w.writeb(v) + } +} + +func (e *bincEncDriver) encBytesLen(c charEncoding, length uint64) { + //TODO: support bincUnicodeOther (for now, just use string or bytearray) + if c == c_RAW { + e.encLen(bincVdByteArray<<4, length) + } else { + e.encLen(bincVdString<<4, length) + } +} + +func (e *bincEncDriver) encLen(bd byte, l uint64) { + if l < 12 { + e.w.writen1(bd | uint8(l+4)) + } else { + e.encLenNumber(bd, l) + } +} + +func (e *bincEncDriver) encLenNumber(bd byte, v uint64) { + switch { + case v <= math.MaxUint8: + e.w.writen2(bd, byte(v)) + case v <= math.MaxUint16: + e.w.writen1(bd | 0x01) + e.w.writeUint16(uint16(v)) + case v <= math.MaxUint32: + e.w.writen1(bd | 0x02) + e.w.writeUint32(uint32(v)) + default: + e.w.writen1(bd | 0x03) + e.w.writeUint64(uint64(v)) + } +} + +//------------------------------------ + +type bincDecDriver struct { + r decReader + bdRead bool + bdType valueType + bd byte + vd byte + vs byte + b [8]byte + m map[uint32]string // symbols (use uint32 as key, as map optimizes for it) +} + +func (d *bincDecDriver) initReadNext() { + if d.bdRead { + return + } + d.bd = d.r.readn1() + d.vd = d.bd >> 4 + d.vs = d.bd & 0x0f + d.bdRead = true + d.bdType = valueTypeUnset +} + +func (d *bincDecDriver) currentEncodedType() valueType { + if d.bdType == valueTypeUnset { + switch d.vd { + case bincVdSpecial: + switch d.vs { + case bincSpNil: + d.bdType = valueTypeNil + case bincSpFalse, bincSpTrue: + d.bdType = valueTypeBool + case bincSpNan, bincSpNegInf, bincSpPosInf, bincSpZeroFloat: + d.bdType = valueTypeFloat + case bincSpZero: + d.bdType = valueTypeUint + case bincSpNegOne: + d.bdType = valueTypeInt + default: + decErr("currentEncodedType: Unrecognized special value 0x%x", d.vs) + } + case bincVdSmallInt: + d.bdType = valueTypeUint + case bincVdPosInt: + d.bdType = valueTypeUint + case bincVdNegInt: + d.bdType = valueTypeInt + case bincVdFloat: + d.bdType = valueTypeFloat + case bincVdString: + d.bdType = valueTypeString + case bincVdSymbol: + d.bdType = valueTypeSymbol + case bincVdByteArray: + d.bdType = valueTypeBytes + case bincVdTimestamp: + d.bdType = valueTypeTimestamp + case bincVdCustomExt: + d.bdType = valueTypeExt + case bincVdArray: + d.bdType = valueTypeArray + case bincVdMap: + d.bdType = valueTypeMap + default: + decErr("currentEncodedType: Unrecognized d.vd: 0x%x", d.vd) + } + } + return d.bdType +} + +func (d *bincDecDriver) tryDecodeAsNil() bool { + if d.bd == bincVdSpecial<<4|bincSpNil { + d.bdRead = false + return true + } + return false +} + +func (d *bincDecDriver) isBuiltinType(rt uintptr) bool { + return rt == timeTypId +} + +func (d *bincDecDriver) decodeBuiltin(rt uintptr, v interface{}) { + switch rt { + case timeTypId: + if d.vd != bincVdTimestamp { + decErr("Invalid d.vd. Expecting 0x%x. Received: 0x%x", bincVdTimestamp, d.vd) + } + tt, err := decodeTime(d.r.readn(int(d.vs))) + if err != nil { + panic(err) + } + var vt *time.Time = v.(*time.Time) + *vt = tt + d.bdRead = false + } +} + +func (d *bincDecDriver) decFloatPre(vs, defaultLen byte) { + if vs&0x8 == 0 { + d.r.readb(d.b[0:defaultLen]) + } else { + l := d.r.readn1() + if l > 8 { + decErr("At most 8 bytes used to represent float. Received: %v bytes", l) + } + for i := l; i < 8; i++ { + d.b[i] = 0 + } + d.r.readb(d.b[0:l]) + } +} + +func (d *bincDecDriver) decFloat() (f float64) { + //if true { f = math.Float64frombits(d.r.readUint64()); break; } + switch vs := d.vs; vs & 0x7 { + case bincFlBin32: + d.decFloatPre(vs, 4) + f = float64(math.Float32frombits(bigen.Uint32(d.b[0:4]))) + case bincFlBin64: + d.decFloatPre(vs, 8) + f = math.Float64frombits(bigen.Uint64(d.b[0:8])) + default: + decErr("only float32 and float64 are supported. d.vd: 0x%x, d.vs: 0x%x", d.vd, d.vs) + } + return +} + +func (d *bincDecDriver) decUint() (v uint64) { + // need to inline the code (interface conversion and type assertion expensive) + switch d.vs { + case 0: + v = uint64(d.r.readn1()) + case 1: + d.r.readb(d.b[6:]) + v = uint64(bigen.Uint16(d.b[6:])) + case 2: + d.b[4] = 0 + d.r.readb(d.b[5:]) + v = uint64(bigen.Uint32(d.b[4:])) + case 3: + d.r.readb(d.b[4:]) + v = uint64(bigen.Uint32(d.b[4:])) + case 4, 5, 6: + lim := int(7 - d.vs) + d.r.readb(d.b[lim:]) + for i := 0; i < lim; i++ { + d.b[i] = 0 + } + v = uint64(bigen.Uint64(d.b[:])) + case 7: + d.r.readb(d.b[:]) + v = uint64(bigen.Uint64(d.b[:])) + default: + decErr("unsigned integers with greater than 64 bits of precision not supported") + } + return +} + +func (d *bincDecDriver) decIntAny() (ui uint64, i int64, neg bool) { + switch d.vd { + case bincVdPosInt: + ui = d.decUint() + i = int64(ui) + case bincVdNegInt: + ui = d.decUint() + i = -(int64(ui)) + neg = true + case bincVdSmallInt: + i = int64(d.vs) + 1 + ui = uint64(d.vs) + 1 + case bincVdSpecial: + switch d.vs { + case bincSpZero: + //i = 0 + case bincSpNegOne: + neg = true + ui = 1 + i = -1 + default: + decErr("numeric decode fails for special value: d.vs: 0x%x", d.vs) + } + default: + decErr("number can only be decoded from uint or int values. d.bd: 0x%x, d.vd: 0x%x", d.bd, d.vd) + } + return +} + +func (d *bincDecDriver) decodeInt(bitsize uint8) (i int64) { + _, i, _ = d.decIntAny() + checkOverflow(0, i, bitsize) + d.bdRead = false + return +} + +func (d *bincDecDriver) decodeUint(bitsize uint8) (ui uint64) { + ui, i, neg := d.decIntAny() + if neg { + decErr("Assigning negative signed value: %v, to unsigned type", i) + } + checkOverflow(ui, 0, bitsize) + d.bdRead = false + return +} + +func (d *bincDecDriver) decodeFloat(chkOverflow32 bool) (f float64) { + switch d.vd { + case bincVdSpecial: + d.bdRead = false + switch d.vs { + case bincSpNan: + return math.NaN() + case bincSpPosInf: + return math.Inf(1) + case bincSpZeroFloat, bincSpZero: + return + case bincSpNegInf: + return math.Inf(-1) + default: + decErr("Invalid d.vs decoding float where d.vd=bincVdSpecial: %v", d.vs) + } + case bincVdFloat: + f = d.decFloat() + default: + _, i, _ := d.decIntAny() + f = float64(i) + } + checkOverflowFloat32(f, chkOverflow32) + d.bdRead = false + return +} + +// bool can be decoded from bool only (single byte). +func (d *bincDecDriver) decodeBool() (b bool) { + switch d.bd { + case (bincVdSpecial | bincSpFalse): + // b = false + case (bincVdSpecial | bincSpTrue): + b = true + default: + decErr("Invalid single-byte value for bool: %s: %x", msgBadDesc, d.bd) + } + d.bdRead = false + return +} + +func (d *bincDecDriver) readMapLen() (length int) { + if d.vd != bincVdMap { + decErr("Invalid d.vd for map. Expecting 0x%x. Got: 0x%x", bincVdMap, d.vd) + } + length = d.decLen() + d.bdRead = false + return +} + +func (d *bincDecDriver) readArrayLen() (length int) { + if d.vd != bincVdArray { + decErr("Invalid d.vd for array. Expecting 0x%x. Got: 0x%x", bincVdArray, d.vd) + } + length = d.decLen() + d.bdRead = false + return +} + +func (d *bincDecDriver) decLen() int { + if d.vs <= 3 { + return int(d.decUint()) + } + return int(d.vs - 4) +} + +func (d *bincDecDriver) decodeString() (s string) { + switch d.vd { + case bincVdString, bincVdByteArray: + if length := d.decLen(); length > 0 { + s = string(d.r.readn(length)) + } + case bincVdSymbol: + //from vs: extract numSymbolBytes, containsStringVal, strLenPrecision, + //extract symbol + //if containsStringVal, read it and put in map + //else look in map for string value + var symbol uint32 + vs := d.vs + //fmt.Printf(">>>> d.vs: 0b%b, & 0x8: %v, & 0x4: %v\n", d.vs, vs & 0x8, vs & 0x4) + if vs&0x8 == 0 { + symbol = uint32(d.r.readn1()) + } else { + symbol = uint32(d.r.readUint16()) + } + if d.m == nil { + d.m = make(map[uint32]string, 16) + } + + if vs&0x4 == 0 { + s = d.m[symbol] + } else { + var slen int + switch vs & 0x3 { + case 0: + slen = int(d.r.readn1()) + case 1: + slen = int(d.r.readUint16()) + case 2: + slen = int(d.r.readUint32()) + case 3: + slen = int(d.r.readUint64()) + } + s = string(d.r.readn(slen)) + d.m[symbol] = s + } + default: + decErr("Invalid d.vd for string. Expecting string:0x%x, bytearray:0x%x or symbol: 0x%x. Got: 0x%x", + bincVdString, bincVdByteArray, bincVdSymbol, d.vd) + } + d.bdRead = false + return +} + +func (d *bincDecDriver) decodeBytes(bs []byte) (bsOut []byte, changed bool) { + var clen int + switch d.vd { + case bincVdString, bincVdByteArray: + clen = d.decLen() + default: + decErr("Invalid d.vd for bytes. Expecting string:0x%x or bytearray:0x%x. Got: 0x%x", + bincVdString, bincVdByteArray, d.vd) + } + if clen > 0 { + // if no contents in stream, don't update the passed byteslice + if len(bs) != clen { + if len(bs) > clen { + bs = bs[:clen] + } else { + bs = make([]byte, clen) + } + bsOut = bs + changed = true + } + d.r.readb(bs) + } + d.bdRead = false + return +} + +func (d *bincDecDriver) decodeExt(verifyTag bool, tag byte) (xtag byte, xbs []byte) { + switch d.vd { + case bincVdCustomExt: + l := d.decLen() + xtag = d.r.readn1() + if verifyTag && xtag != tag { + decErr("Wrong extension tag. Got %b. Expecting: %v", xtag, tag) + } + xbs = d.r.readn(l) + case bincVdByteArray: + xbs, _ = d.decodeBytes(nil) + default: + decErr("Invalid d.vd for extensions (Expecting extensions or byte array). Got: 0x%x", d.vd) + } + d.bdRead = false + return +} + +func (d *bincDecDriver) decodeNaked() (v interface{}, vt valueType, decodeFurther bool) { + d.initReadNext() + + switch d.vd { + case bincVdSpecial: + switch d.vs { + case bincSpNil: + vt = valueTypeNil + case bincSpFalse: + vt = valueTypeBool + v = false + case bincSpTrue: + vt = valueTypeBool + v = true + case bincSpNan: + vt = valueTypeFloat + v = math.NaN() + case bincSpPosInf: + vt = valueTypeFloat + v = math.Inf(1) + case bincSpNegInf: + vt = valueTypeFloat + v = math.Inf(-1) + case bincSpZeroFloat: + vt = valueTypeFloat + v = float64(0) + case bincSpZero: + vt = valueTypeUint + v = int64(0) // int8(0) + case bincSpNegOne: + vt = valueTypeInt + v = int64(-1) // int8(-1) + default: + decErr("decodeNaked: Unrecognized special value 0x%x", d.vs) + } + case bincVdSmallInt: + vt = valueTypeUint + v = uint64(int8(d.vs)) + 1 // int8(d.vs) + 1 + case bincVdPosInt: + vt = valueTypeUint + v = d.decUint() + case bincVdNegInt: + vt = valueTypeInt + v = -(int64(d.decUint())) + case bincVdFloat: + vt = valueTypeFloat + v = d.decFloat() + case bincVdSymbol: + vt = valueTypeSymbol + v = d.decodeString() + case bincVdString: + vt = valueTypeString + v = d.decodeString() + case bincVdByteArray: + vt = valueTypeBytes + v, _ = d.decodeBytes(nil) + case bincVdTimestamp: + vt = valueTypeTimestamp + tt, err := decodeTime(d.r.readn(int(d.vs))) + if err != nil { + panic(err) + } + v = tt + case bincVdCustomExt: + vt = valueTypeExt + l := d.decLen() + var re RawExt + re.Tag = d.r.readn1() + re.Data = d.r.readn(l) + v = &re + vt = valueTypeExt + case bincVdArray: + vt = valueTypeArray + decodeFurther = true + case bincVdMap: + vt = valueTypeMap + decodeFurther = true + default: + decErr("decodeNaked: Unrecognized d.vd: 0x%x", d.vd) + } + + if !decodeFurther { + d.bdRead = false + } + return +} + +//------------------------------------ + +//BincHandle is a Handle for the Binc Schema-Free Encoding Format +//defined at https://github.com/ugorji/binc . +// +//BincHandle currently supports all Binc features with the following EXCEPTIONS: +// - only integers up to 64 bits of precision are supported. +// big integers are unsupported. +// - Only IEEE 754 binary32 and binary64 floats are supported (ie Go float32 and float64 types). +// extended precision and decimal IEEE 754 floats are unsupported. +// - Only UTF-8 strings supported. +// Unicode_Other Binc types (UTF16, UTF32) are currently unsupported. +//Note that these EXCEPTIONS are temporary and full support is possible and may happen soon. +type BincHandle struct { + BasicHandle +} + +func (h *BincHandle) newEncDriver(w encWriter) encDriver { + return &bincEncDriver{w: w} +} + +func (h *BincHandle) newDecDriver(r decReader) decDriver { + return &bincDecDriver{r: r} +} + +func (_ *BincHandle) writeExt() bool { + return true +} + +func (h *BincHandle) getBasicHandle() *BasicHandle { + return &h.BasicHandle +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/codecs_test.go b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/codecs_test.go new file mode 100644 index 0000000000000..cb184491f1b68 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/codecs_test.go @@ -0,0 +1,1002 @@ +// Copyright (c) 2012, 2013 Ugorji Nwoke. All rights reserved. +// Use of this source code is governed by a BSD-style license found in the LICENSE file. + +package codec + +// Test works by using a slice of interfaces. +// It can test for encoding/decoding into/from a nil interface{} +// or passing the object to encode/decode into. +// +// There are basically 2 main tests here. +// First test internally encodes and decodes things and verifies that +// the artifact was as expected. +// Second test will use python msgpack to create a bunch of golden files, +// read those files, and compare them to what it should be. It then +// writes those files back out and compares the byte streams. +// +// Taken together, the tests are pretty extensive. + +import ( + "bytes" + "encoding/gob" + "flag" + "fmt" + "io/ioutil" + "math" + "net" + "net/rpc" + "os" + "os/exec" + "path/filepath" + "reflect" + "runtime" + "strconv" + "sync/atomic" + "testing" + "time" +) + +type testVerifyArg int + +const ( + testVerifyMapTypeSame testVerifyArg = iota + testVerifyMapTypeStrIntf + testVerifyMapTypeIntfIntf + // testVerifySliceIntf + testVerifyForPython +) + +var ( + testInitDebug bool + testUseIoEncDec bool + testStructToArray bool + testWriteNoSymbols bool + + _ = fmt.Printf + skipVerifyVal interface{} = &(struct{}{}) + + // For Go Time, do not use a descriptive timezone. + // It's unnecessary, and makes it harder to do a reflect.DeepEqual. + // The Offset already tells what the offset should be, if not on UTC and unknown zone name. + timeLoc = time.FixedZone("", -8*60*60) // UTC-08:00 //time.UTC-8 + timeToCompare1 = time.Date(2012, 2, 2, 2, 2, 2, 2000, timeLoc) + timeToCompare2 = time.Date(1900, 2, 2, 2, 2, 2, 2000, timeLoc) + timeToCompare3 = time.Unix(0, 0).UTC() + timeToCompare4 = time.Time{}.UTC() + + table []interface{} // main items we encode + tableVerify []interface{} // we verify encoded things against this after decode + tableTestNilVerify []interface{} // for nil interface, use this to verify (rules are different) + tablePythonVerify []interface{} // for verifying for python, since Python sometimes + // will encode a float32 as float64, or large int as uint + testRpcInt = new(TestRpcInt) + testMsgpackH = &MsgpackHandle{} + testBincH = &BincHandle{} + testSimpleH = &SimpleHandle{} +) + +func testInitFlags() { + // delete(testDecOpts.ExtFuncs, timeTyp) + flag.BoolVar(&testInitDebug, "tg", false, "Test Debug") + flag.BoolVar(&testUseIoEncDec, "ti", false, "Use IO Reader/Writer for Marshal/Unmarshal") + flag.BoolVar(&testStructToArray, "ts", false, "Set StructToArray option") + flag.BoolVar(&testWriteNoSymbols, "tn", false, "Set NoSymbols option") +} + +type AnonInTestStruc struct { + AS string + AI64 int64 + AI16 int16 + AUi64 uint64 + ASslice []string + AI64slice []int64 +} + +type TestStruc struct { + S string + I64 int64 + I16 int16 + Ui64 uint64 + Ui8 uint8 + B bool + By byte + + Sslice []string + I64slice []int64 + I16slice []int16 + Ui64slice []uint64 + Ui8slice []uint8 + Bslice []bool + Byslice []byte + + Islice []interface{} + Iptrslice []*int64 + + AnonInTestStruc + + //M map[interface{}]interface{} `json:"-",bson:"-"` + Ms map[string]interface{} + Msi64 map[string]int64 + + Nintf interface{} //don't set this, so we can test for nil + T time.Time + Nmap map[string]bool //don't set this, so we can test for nil + Nslice []byte //don't set this, so we can test for nil + Nint64 *int64 //don't set this, so we can test for nil + Mtsptr map[string]*TestStruc + Mts map[string]TestStruc + Its []*TestStruc + Nteststruc *TestStruc +} + +type TestABC struct { + A, B, C string +} + +type TestRpcInt struct { + i int +} + +func (r *TestRpcInt) Update(n int, res *int) error { r.i = n; *res = r.i; return nil } +func (r *TestRpcInt) Square(ignore int, res *int) error { *res = r.i * r.i; return nil } +func (r *TestRpcInt) Mult(n int, res *int) error { *res = r.i * n; return nil } +func (r *TestRpcInt) EchoStruct(arg TestABC, res *string) error { + *res = fmt.Sprintf("%#v", arg) + return nil +} +func (r *TestRpcInt) Echo123(args []string, res *string) error { + *res = fmt.Sprintf("%#v", args) + return nil +} + +func testVerifyVal(v interface{}, arg testVerifyArg) (v2 interface{}) { + //for python msgpack, + // - all positive integers are unsigned 64-bit ints + // - all floats are float64 + switch iv := v.(type) { + case int8: + if iv > 0 { + v2 = uint64(iv) + } else { + v2 = int64(iv) + } + case int16: + if iv > 0 { + v2 = uint64(iv) + } else { + v2 = int64(iv) + } + case int32: + if iv > 0 { + v2 = uint64(iv) + } else { + v2 = int64(iv) + } + case int64: + if iv > 0 { + v2 = uint64(iv) + } else { + v2 = int64(iv) + } + case uint8: + v2 = uint64(iv) + case uint16: + v2 = uint64(iv) + case uint32: + v2 = uint64(iv) + case uint64: + v2 = uint64(iv) + case float32: + v2 = float64(iv) + case float64: + v2 = float64(iv) + case []interface{}: + m2 := make([]interface{}, len(iv)) + for j, vj := range iv { + m2[j] = testVerifyVal(vj, arg) + } + v2 = m2 + case map[string]bool: + switch arg { + case testVerifyMapTypeSame: + m2 := make(map[string]bool) + for kj, kv := range iv { + m2[kj] = kv + } + v2 = m2 + case testVerifyMapTypeStrIntf, testVerifyForPython: + m2 := make(map[string]interface{}) + for kj, kv := range iv { + m2[kj] = kv + } + v2 = m2 + case testVerifyMapTypeIntfIntf: + m2 := make(map[interface{}]interface{}) + for kj, kv := range iv { + m2[kj] = kv + } + v2 = m2 + } + case map[string]interface{}: + switch arg { + case testVerifyMapTypeSame: + m2 := make(map[string]interface{}) + for kj, kv := range iv { + m2[kj] = testVerifyVal(kv, arg) + } + v2 = m2 + case testVerifyMapTypeStrIntf, testVerifyForPython: + m2 := make(map[string]interface{}) + for kj, kv := range iv { + m2[kj] = testVerifyVal(kv, arg) + } + v2 = m2 + case testVerifyMapTypeIntfIntf: + m2 := make(map[interface{}]interface{}) + for kj, kv := range iv { + m2[kj] = testVerifyVal(kv, arg) + } + v2 = m2 + } + case map[interface{}]interface{}: + m2 := make(map[interface{}]interface{}) + for kj, kv := range iv { + m2[testVerifyVal(kj, arg)] = testVerifyVal(kv, arg) + } + v2 = m2 + case time.Time: + switch arg { + case testVerifyForPython: + if iv2 := iv.UnixNano(); iv2 > 0 { + v2 = uint64(iv2) + } else { + v2 = int64(iv2) + } + default: + v2 = v + } + default: + v2 = v + } + return +} + +func testInit() { + gob.Register(new(TestStruc)) + if testInitDebug { + ts0 := newTestStruc(2, false) + fmt.Printf("====> depth: %v, ts: %#v\n", 2, ts0) + } + + testBincH.StructToArray = testStructToArray + if testWriteNoSymbols { + testBincH.AsSymbols = AsSymbolNone + } else { + testBincH.AsSymbols = AsSymbolAll + } + testMsgpackH.StructToArray = testStructToArray + testMsgpackH.RawToString = true + // testMsgpackH.AddExt(byteSliceTyp, 0, testMsgpackH.BinaryEncodeExt, testMsgpackH.BinaryDecodeExt) + // testMsgpackH.AddExt(timeTyp, 1, testMsgpackH.TimeEncodeExt, testMsgpackH.TimeDecodeExt) + timeEncExt := func(rv reflect.Value) ([]byte, error) { + return encodeTime(rv.Interface().(time.Time)), nil + } + timeDecExt := func(rv reflect.Value, bs []byte) error { + tt, err := decodeTime(bs) + if err == nil { + rv.Set(reflect.ValueOf(tt)) + } + return err + } + + // add extensions for msgpack, simple for time.Time, so we can encode/decode same way. + testMsgpackH.AddExt(timeTyp, 1, timeEncExt, timeDecExt) + testSimpleH.AddExt(timeTyp, 1, timeEncExt, timeDecExt) + + primitives := []interface{}{ + int8(-8), + int16(-1616), + int32(-32323232), + int64(-6464646464646464), + uint8(192), + uint16(1616), + uint32(32323232), + uint64(6464646464646464), + byte(192), + float32(-3232.0), + float64(-6464646464.0), + float32(3232.0), + float64(6464646464.0), + false, + true, + nil, + "someday", + "", + "bytestring", + timeToCompare1, + timeToCompare2, + timeToCompare3, + timeToCompare4, + } + mapsAndStrucs := []interface{}{ + map[string]bool{ + "true": true, + "false": false, + }, + map[string]interface{}{ + "true": "True", + "false": false, + "uint16(1616)": uint16(1616), + }, + //add a complex combo map in here. (map has list which has map) + //note that after the first thing, everything else should be generic. + map[string]interface{}{ + "list": []interface{}{ + int16(1616), + int32(32323232), + true, + float32(-3232.0), + map[string]interface{}{ + "TRUE": true, + "FALSE": false, + }, + []interface{}{true, false}, + }, + "int32": int32(32323232), + "bool": true, + "LONG STRING": "123456789012345678901234567890123456789012345678901234567890", + "SHORT STRING": "1234567890", + }, + map[interface{}]interface{}{ + true: "true", + uint8(138): false, + "false": uint8(200), + }, + newTestStruc(0, false), + } + + table = []interface{}{} + table = append(table, primitives...) //0-19 are primitives + table = append(table, primitives) //20 is a list of primitives + table = append(table, mapsAndStrucs...) //21-24 are maps. 25 is a *struct + + tableVerify = make([]interface{}, len(table)) + tableTestNilVerify = make([]interface{}, len(table)) + tablePythonVerify = make([]interface{}, len(table)) + + lp := len(primitives) + av := tableVerify + for i, v := range table { + if i == lp+3 { + av[i] = skipVerifyVal + continue + } + //av[i] = testVerifyVal(v, testVerifyMapTypeSame) + switch v.(type) { + case []interface{}: + av[i] = testVerifyVal(v, testVerifyMapTypeSame) + case map[string]interface{}: + av[i] = testVerifyVal(v, testVerifyMapTypeSame) + case map[interface{}]interface{}: + av[i] = testVerifyVal(v, testVerifyMapTypeSame) + default: + av[i] = v + } + } + + av = tableTestNilVerify + for i, v := range table { + if i > lp+3 { + av[i] = skipVerifyVal + continue + } + av[i] = testVerifyVal(v, testVerifyMapTypeStrIntf) + } + + av = tablePythonVerify + for i, v := range table { + if i > lp+3 { + av[i] = skipVerifyVal + continue + } + av[i] = testVerifyVal(v, testVerifyForPython) + } + + tablePythonVerify = tablePythonVerify[:24] +} + +func testUnmarshal(v interface{}, data []byte, h Handle) error { + if testUseIoEncDec { + return NewDecoder(bytes.NewBuffer(data), h).Decode(v) + } + return NewDecoderBytes(data, h).Decode(v) +} + +func testMarshal(v interface{}, h Handle) (bs []byte, err error) { + if testUseIoEncDec { + var buf bytes.Buffer + err = NewEncoder(&buf, h).Encode(v) + bs = buf.Bytes() + return + } + err = NewEncoderBytes(&bs, h).Encode(v) + return +} + +func testMarshalErr(v interface{}, h Handle, t *testing.T, name string) (bs []byte, err error) { + if bs, err = testMarshal(v, h); err != nil { + logT(t, "Error encoding %s: %v, Err: %v", name, v, err) + t.FailNow() + } + return +} + +func testUnmarshalErr(v interface{}, data []byte, h Handle, t *testing.T, name string) (err error) { + if err = testUnmarshal(v, data, h); err != nil { + logT(t, "Error Decoding into %s: %v, Err: %v", name, v, err) + t.FailNow() + } + return +} + +func newTestStruc(depth int, bench bool) (ts *TestStruc) { + var i64a, i64b, i64c, i64d int64 = 64, 6464, 646464, 64646464 + + ts = &TestStruc{ + S: "some string", + I64: math.MaxInt64 * 2 / 3, // 64, + I16: 16, + Ui64: uint64(int64(math.MaxInt64 * 2 / 3)), // 64, //don't use MaxUint64, as bson can't write it + Ui8: 160, + B: true, + By: 5, + + Sslice: []string{"one", "two", "three"}, + I64slice: []int64{1, 2, 3}, + I16slice: []int16{4, 5, 6}, + Ui64slice: []uint64{137, 138, 139}, + Ui8slice: []uint8{210, 211, 212}, + Bslice: []bool{true, false, true, false}, + Byslice: []byte{13, 14, 15}, + + Islice: []interface{}{"true", true, "no", false, uint64(288), float64(0.4)}, + + Ms: map[string]interface{}{ + "true": "true", + "int64(9)": false, + }, + Msi64: map[string]int64{ + "one": 1, + "two": 2, + }, + T: timeToCompare1, + AnonInTestStruc: AnonInTestStruc{ + AS: "A-String", + AI64: 64, + AI16: 16, + AUi64: 64, + ASslice: []string{"Aone", "Atwo", "Athree"}, + AI64slice: []int64{1, 2, 3}, + }, + } + //For benchmarks, some things will not work. + if !bench { + //json and bson require string keys in maps + //ts.M = map[interface{}]interface{}{ + // true: "true", + // int8(9): false, + //} + //gob cannot encode nil in element in array (encodeArray: nil element) + ts.Iptrslice = []*int64{nil, &i64a, nil, &i64b, nil, &i64c, nil, &i64d, nil} + // ts.Iptrslice = nil + } + if depth > 0 { + depth-- + if ts.Mtsptr == nil { + ts.Mtsptr = make(map[string]*TestStruc) + } + if ts.Mts == nil { + ts.Mts = make(map[string]TestStruc) + } + ts.Mtsptr["0"] = newTestStruc(depth, bench) + ts.Mts["0"] = *(ts.Mtsptr["0"]) + ts.Its = append(ts.Its, ts.Mtsptr["0"]) + } + return +} + +// doTestCodecTableOne allows us test for different variations based on arguments passed. +func doTestCodecTableOne(t *testing.T, testNil bool, h Handle, + vs []interface{}, vsVerify []interface{}) { + //if testNil, then just test for when a pointer to a nil interface{} is passed. It should work. + //Current setup allows us test (at least manually) the nil interface or typed interface. + logT(t, "================ TestNil: %v ================\n", testNil) + for i, v0 := range vs { + logT(t, "..............................................") + logT(t, " Testing: #%d:, %T, %#v\n", i, v0, v0) + b0, err := testMarshalErr(v0, h, t, "v0") + if err != nil { + continue + } + logT(t, " Encoded bytes: len: %v, %v\n", len(b0), b0) + + var v1 interface{} + + if testNil { + err = testUnmarshal(&v1, b0, h) + } else { + if v0 != nil { + v0rt := reflect.TypeOf(v0) // ptr + rv1 := reflect.New(v0rt) + err = testUnmarshal(rv1.Interface(), b0, h) + v1 = rv1.Elem().Interface() + // v1 = reflect.Indirect(reflect.ValueOf(v1)).Interface() + } + } + + logT(t, " v1 returned: %T, %#v", v1, v1) + // if v1 != nil { + // logT(t, " v1 returned: %T, %#v", v1, v1) + // //we always indirect, because ptr to typed value may be passed (if not testNil) + // v1 = reflect.Indirect(reflect.ValueOf(v1)).Interface() + // } + if err != nil { + logT(t, "-------- Error: %v. Partial return: %v", err, v1) + failT(t) + continue + } + v0check := vsVerify[i] + if v0check == skipVerifyVal { + logT(t, " Nil Check skipped: Decoded: %T, %#v\n", v1, v1) + continue + } + + if err = deepEqual(v0check, v1); err == nil { + logT(t, "++++++++ Before and After marshal matched\n") + } else { + logT(t, "-------- Before and After marshal do not match: Error: %v"+ + " ====> GOLDEN: (%T) %#v, DECODED: (%T) %#v\n", err, v0check, v0check, v1, v1) + failT(t) + } + } +} + +func testCodecTableOne(t *testing.T, h Handle) { + // func TestMsgpackAllExperimental(t *testing.T) { + // dopts := testDecOpts(nil, nil, false, true, true), + + switch v := h.(type) { + case *MsgpackHandle: + var oldWriteExt, oldRawToString bool + oldWriteExt, v.WriteExt = v.WriteExt, true + oldRawToString, v.RawToString = v.RawToString, true + doTestCodecTableOne(t, false, h, table, tableVerify) + v.WriteExt, v.RawToString = oldWriteExt, oldRawToString + default: + doTestCodecTableOne(t, false, h, table, tableVerify) + } + // func TestMsgpackAll(t *testing.T) { + idxTime, numPrim, numMap := 19, 23, 4 + + //skip []interface{} containing time.Time + doTestCodecTableOne(t, false, h, table[:numPrim], tableVerify[:numPrim]) + doTestCodecTableOne(t, false, h, table[numPrim+1:], tableVerify[numPrim+1:]) + // func TestMsgpackNilStringMap(t *testing.T) { + var oldMapType reflect.Type + v := h.getBasicHandle() + oldMapType, v.MapType = v.MapType, mapStrIntfTyp + + //skip time.Time, []interface{} containing time.Time, last map, and newStruc + doTestCodecTableOne(t, true, h, table[:idxTime], tableTestNilVerify[:idxTime]) + doTestCodecTableOne(t, true, h, table[numPrim+1:numPrim+numMap], tableTestNilVerify[numPrim+1:numPrim+numMap]) + + v.MapType = oldMapType + + // func TestMsgpackNilIntf(t *testing.T) { + + //do newTestStruc and last element of map + doTestCodecTableOne(t, true, h, table[numPrim+numMap:], tableTestNilVerify[numPrim+numMap:]) + //TODO? What is this one? + //doTestCodecTableOne(t, true, h, table[17:18], tableTestNilVerify[17:18]) +} + +func testCodecMiscOne(t *testing.T, h Handle) { + b, err := testMarshalErr(32, h, t, "32") + // Cannot do this nil one, because faster type assertion decoding will panic + // var i *int32 + // if err = testUnmarshal(b, i, nil); err == nil { + // logT(t, "------- Expecting error because we cannot unmarshal to int32 nil ptr") + // t.FailNow() + // } + var i2 int32 = 0 + err = testUnmarshalErr(&i2, b, h, t, "int32-ptr") + if i2 != int32(32) { + logT(t, "------- didn't unmarshal to 32: Received: %d", i2) + t.FailNow() + } + + // func TestMsgpackDecodePtr(t *testing.T) { + ts := newTestStruc(0, false) + b, err = testMarshalErr(ts, h, t, "pointer-to-struct") + if len(b) < 40 { + logT(t, "------- Size must be > 40. Size: %d", len(b)) + t.FailNow() + } + logT(t, "------- b: %v", b) + ts2 := new(TestStruc) + err = testUnmarshalErr(ts2, b, h, t, "pointer-to-struct") + if ts2.I64 != math.MaxInt64*2/3 { + logT(t, "------- Unmarshal wrong. Expect I64 = 64. Got: %v", ts2.I64) + t.FailNow() + } + + // func TestMsgpackIntfDecode(t *testing.T) { + m := map[string]int{"A": 2, "B": 3} + p := []interface{}{m} + bs, err := testMarshalErr(p, h, t, "p") + + m2 := map[string]int{} + p2 := []interface{}{m2} + err = testUnmarshalErr(&p2, bs, h, t, "&p2") + + if m2["A"] != 2 || m2["B"] != 3 { + logT(t, "m2 not as expected: expecting: %v, got: %v", m, m2) + t.FailNow() + } + // log("m: %v, m2: %v, p: %v, p2: %v", m, m2, p, p2) + checkEqualT(t, p, p2, "p=p2") + checkEqualT(t, m, m2, "m=m2") + if err = deepEqual(p, p2); err == nil { + logT(t, "p and p2 match") + } else { + logT(t, "Not Equal: %v. p: %v, p2: %v", err, p, p2) + t.FailNow() + } + if err = deepEqual(m, m2); err == nil { + logT(t, "m and m2 match") + } else { + logT(t, "Not Equal: %v. m: %v, m2: %v", err, m, m2) + t.FailNow() + } + + // func TestMsgpackDecodeStructSubset(t *testing.T) { + // test that we can decode a subset of the stream + mm := map[string]interface{}{"A": 5, "B": 99, "C": 333} + bs, err = testMarshalErr(mm, h, t, "mm") + type ttt struct { + A uint8 + C int32 + } + var t2 ttt + testUnmarshalErr(&t2, bs, h, t, "t2") + t3 := ttt{5, 333} + checkEqualT(t, t2, t3, "t2=t3") + + // println(">>>>>") + // test simple arrays, non-addressable arrays, slices + type tarr struct { + A int64 + B [3]int64 + C []byte + D [3]byte + } + var tarr0 = tarr{1, [3]int64{2, 3, 4}, []byte{4, 5, 6}, [3]byte{7, 8, 9}} + // test both pointer and non-pointer (value) + for _, tarr1 := range []interface{}{tarr0, &tarr0} { + bs, err = testMarshalErr(tarr1, h, t, "tarr1") + var tarr2 tarr + testUnmarshalErr(&tarr2, bs, h, t, "tarr2") + checkEqualT(t, tarr0, tarr2, "tarr0=tarr2") + // fmt.Printf(">>>> err: %v. tarr1: %v, tarr2: %v\n", err, tarr0, tarr2) + } + + // test byte array, even if empty (msgpack only) + if h == testMsgpackH { + type ystruct struct { + Anarray []byte + } + var ya = ystruct{} + testUnmarshalErr(&ya, []byte{0x91, 0x90}, h, t, "ya") + } +} + +func testCodecEmbeddedPointer(t *testing.T, h Handle) { + type Z int + type A struct { + AnInt int + } + type B struct { + *Z + *A + MoreInt int + } + var z Z = 4 + x1 := &B{&z, &A{5}, 6} + bs, err := testMarshalErr(x1, h, t, "x1") + // fmt.Printf("buf: len(%v): %x\n", buf.Len(), buf.Bytes()) + var x2 = new(B) + err = testUnmarshalErr(x2, bs, h, t, "x2") + err = checkEqualT(t, x1, x2, "x1=x2") + _ = err +} + +func doTestRpcOne(t *testing.T, rr Rpc, h Handle, doRequest bool, exitSleepMs time.Duration, +) (port int) { + // rpc needs EOF, which is sent via a panic, and so must be recovered. + if !recoverPanicToErr { + logT(t, "EXPECTED. set recoverPanicToErr=true, since rpc needs EOF") + t.FailNow() + } + srv := rpc.NewServer() + srv.Register(testRpcInt) + ln, err := net.Listen("tcp", "127.0.0.1:0") + // log("listener: %v", ln.Addr()) + checkErrT(t, err) + port = (ln.Addr().(*net.TCPAddr)).Port + // var opts *DecoderOptions + // opts := testDecOpts + // opts.MapType = mapStrIntfTyp + // opts.RawToString = false + serverExitChan := make(chan bool, 1) + var serverExitFlag uint64 = 0 + serverFn := func() { + for { + conn1, err1 := ln.Accept() + // if err1 != nil { + // //fmt.Printf("accept err1: %v\n", err1) + // continue + // } + if atomic.LoadUint64(&serverExitFlag) == 1 { + serverExitChan <- true + conn1.Close() + return // exit serverFn goroutine + } + if err1 == nil { + var sc rpc.ServerCodec = rr.ServerCodec(conn1, h) + srv.ServeCodec(sc) + } + } + } + + clientFn := func(cc rpc.ClientCodec) { + cl := rpc.NewClientWithCodec(cc) + defer cl.Close() + var up, sq, mult int + var rstr string + // log("Calling client") + checkErrT(t, cl.Call("TestRpcInt.Update", 5, &up)) + // log("Called TestRpcInt.Update") + checkEqualT(t, testRpcInt.i, 5, "testRpcInt.i=5") + checkEqualT(t, up, 5, "up=5") + checkErrT(t, cl.Call("TestRpcInt.Square", 1, &sq)) + checkEqualT(t, sq, 25, "sq=25") + checkErrT(t, cl.Call("TestRpcInt.Mult", 20, &mult)) + checkEqualT(t, mult, 100, "mult=100") + checkErrT(t, cl.Call("TestRpcInt.EchoStruct", TestABC{"Aa", "Bb", "Cc"}, &rstr)) + checkEqualT(t, rstr, fmt.Sprintf("%#v", TestABC{"Aa", "Bb", "Cc"}), "rstr=") + checkErrT(t, cl.Call("TestRpcInt.Echo123", []string{"A1", "B2", "C3"}, &rstr)) + checkEqualT(t, rstr, fmt.Sprintf("%#v", []string{"A1", "B2", "C3"}), "rstr=") + } + + connFn := func() (bs net.Conn) { + // log("calling f1") + bs, err2 := net.Dial(ln.Addr().Network(), ln.Addr().String()) + //fmt.Printf("f1. bs: %v, err2: %v\n", bs, err2) + checkErrT(t, err2) + return + } + + exitFn := func() { + atomic.StoreUint64(&serverExitFlag, 1) + bs := connFn() + <-serverExitChan + bs.Close() + // serverExitChan <- true + } + + go serverFn() + runtime.Gosched() + //time.Sleep(100 * time.Millisecond) + if exitSleepMs == 0 { + defer ln.Close() + defer exitFn() + } + if doRequest { + bs := connFn() + cc := rr.ClientCodec(bs, h) + clientFn(cc) + } + if exitSleepMs != 0 { + go func() { + defer ln.Close() + time.Sleep(exitSleepMs) + exitFn() + }() + } + return +} + +// Comprehensive testing that generates data encoded from python msgpack, +// and validates that our code can read and write it out accordingly. +// We keep this unexported here, and put actual test in ext_dep_test.go. +// This way, it can be excluded by excluding file completely. +func doTestMsgpackPythonGenStreams(t *testing.T) { + logT(t, "TestPythonGenStreams") + tmpdir, err := ioutil.TempDir("", "golang-msgpack-test") + if err != nil { + logT(t, "-------- Unable to create temp directory\n") + t.FailNow() + } + defer os.RemoveAll(tmpdir) + logT(t, "tmpdir: %v", tmpdir) + cmd := exec.Command("python", "msgpack_test.py", "testdata", tmpdir) + //cmd.Stdin = strings.NewReader("some input") + //cmd.Stdout = &out + var cmdout []byte + if cmdout, err = cmd.CombinedOutput(); err != nil { + logT(t, "-------- Error running msgpack_test.py testdata. Err: %v", err) + logT(t, " %v", string(cmdout)) + t.FailNow() + } + + oldMapType := testMsgpackH.MapType + for i, v := range tablePythonVerify { + testMsgpackH.MapType = oldMapType + //load up the golden file based on number + //decode it + //compare to in-mem object + //encode it again + //compare to output stream + logT(t, "..............................................") + logT(t, " Testing: #%d: %T, %#v\n", i, v, v) + var bss []byte + bss, err = ioutil.ReadFile(filepath.Join(tmpdir, strconv.Itoa(i)+".golden")) + if err != nil { + logT(t, "-------- Error reading golden file: %d. Err: %v", i, err) + failT(t) + continue + } + testMsgpackH.MapType = mapStrIntfTyp + + var v1 interface{} + if err = testUnmarshal(&v1, bss, testMsgpackH); err != nil { + logT(t, "-------- Error decoding stream: %d: Err: %v", i, err) + failT(t) + continue + } + if v == skipVerifyVal { + continue + } + //no need to indirect, because we pass a nil ptr, so we already have the value + //if v1 != nil { v1 = reflect.Indirect(reflect.ValueOf(v1)).Interface() } + if err = deepEqual(v, v1); err == nil { + logT(t, "++++++++ Objects match") + } else { + logT(t, "-------- Objects do not match: %v. Source: %T. Decoded: %T", err, v, v1) + logT(t, "-------- AGAINST: %#v", v) + logT(t, "-------- DECODED: %#v <====> %#v", v1, reflect.Indirect(reflect.ValueOf(v1)).Interface()) + failT(t) + } + bsb, err := testMarshal(v1, testMsgpackH) + if err != nil { + logT(t, "Error encoding to stream: %d: Err: %v", i, err) + failT(t) + continue + } + if err = deepEqual(bsb, bss); err == nil { + logT(t, "++++++++ Bytes match") + } else { + logT(t, "???????? Bytes do not match. %v.", err) + xs := "--------" + if reflect.ValueOf(v).Kind() == reflect.Map { + xs = " " + logT(t, "%s It's a map. Ok that they don't match (dependent on ordering).", xs) + } else { + logT(t, "%s It's not a map. They should match.", xs) + failT(t) + } + logT(t, "%s FROM_FILE: %4d] %v", xs, len(bss), bss) + logT(t, "%s ENCODED: %4d] %v", xs, len(bsb), bsb) + } + } + testMsgpackH.MapType = oldMapType +} + +// To test MsgpackSpecRpc, we test 3 scenarios: +// - Go Client to Go RPC Service (contained within TestMsgpackRpcSpec) +// - Go client to Python RPC Service (contained within doTestMsgpackRpcSpecGoClientToPythonSvc) +// - Python Client to Go RPC Service (contained within doTestMsgpackRpcSpecPythonClientToGoSvc) +// +// This allows us test the different calling conventions +// - Go Service requires only one argument +// - Python Service allows multiple arguments + +func doTestMsgpackRpcSpecGoClientToPythonSvc(t *testing.T) { + openPort := "6789" + cmd := exec.Command("python", "msgpack_test.py", "rpc-server", openPort, "2") + checkErrT(t, cmd.Start()) + time.Sleep(100 * time.Millisecond) // time for python rpc server to start + bs, err2 := net.Dial("tcp", ":"+openPort) + checkErrT(t, err2) + cc := MsgpackSpecRpc.ClientCodec(bs, testMsgpackH) + cl := rpc.NewClientWithCodec(cc) + defer cl.Close() + var rstr string + checkErrT(t, cl.Call("EchoStruct", TestABC{"Aa", "Bb", "Cc"}, &rstr)) + //checkEqualT(t, rstr, "{'A': 'Aa', 'B': 'Bb', 'C': 'Cc'}") + var mArgs MsgpackSpecRpcMultiArgs = []interface{}{"A1", "B2", "C3"} + checkErrT(t, cl.Call("Echo123", mArgs, &rstr)) + checkEqualT(t, rstr, "1:A1 2:B2 3:C3", "rstr=") +} + +func doTestMsgpackRpcSpecPythonClientToGoSvc(t *testing.T) { + port := doTestRpcOne(t, MsgpackSpecRpc, testMsgpackH, false, 1*time.Second) + //time.Sleep(1000 * time.Millisecond) + cmd := exec.Command("python", "msgpack_test.py", "rpc-client-go-service", strconv.Itoa(port)) + var cmdout []byte + var err error + if cmdout, err = cmd.CombinedOutput(); err != nil { + logT(t, "-------- Error running msgpack_test.py rpc-client-go-service. Err: %v", err) + logT(t, " %v", string(cmdout)) + t.FailNow() + } + checkEqualT(t, string(cmdout), + fmt.Sprintf("%#v\n%#v\n", []string{"A1", "B2", "C3"}, TestABC{"Aa", "Bb", "Cc"}), "cmdout=") +} + +func TestBincCodecsTable(t *testing.T) { + testCodecTableOne(t, testBincH) +} + +func TestBincCodecsMisc(t *testing.T) { + testCodecMiscOne(t, testBincH) +} + +func TestBincCodecsEmbeddedPointer(t *testing.T) { + testCodecEmbeddedPointer(t, testBincH) +} + +func TestSimpleCodecsTable(t *testing.T) { + testCodecTableOne(t, testSimpleH) +} + +func TestSimpleCodecsMisc(t *testing.T) { + testCodecMiscOne(t, testSimpleH) +} + +func TestSimpleCodecsEmbeddedPointer(t *testing.T) { + testCodecEmbeddedPointer(t, testSimpleH) +} + +func TestMsgpackCodecsTable(t *testing.T) { + testCodecTableOne(t, testMsgpackH) +} + +func TestMsgpackCodecsMisc(t *testing.T) { + testCodecMiscOne(t, testMsgpackH) +} + +func TestMsgpackCodecsEmbeddedPointer(t *testing.T) { + testCodecEmbeddedPointer(t, testMsgpackH) +} + +func TestBincRpcGo(t *testing.T) { + doTestRpcOne(t, GoRpc, testBincH, true, 0) +} + +func _TestSimpleRpcGo(t *testing.T) { + doTestRpcOne(t, GoRpc, testSimpleH, true, 0) +} + +func TestMsgpackRpcGo(t *testing.T) { + doTestRpcOne(t, GoRpc, testMsgpackH, true, 0) +} + +func TestMsgpackRpcSpec(t *testing.T) { + doTestRpcOne(t, MsgpackSpecRpc, testMsgpackH, true, 0) +} + +// TODO: +// Add Tests for: +// - decoding empty list/map in stream into a nil slice/map +// - binary(M|Unm)arsher support for time.Time diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/decode.go b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/decode.go new file mode 100644 index 0000000000000..87bef2b93586c --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/decode.go @@ -0,0 +1,1048 @@ +// Copyright (c) 2012, 2013 Ugorji Nwoke. All rights reserved. +// Use of this source code is governed by a BSD-style license found in the LICENSE file. + +package codec + +import ( + "io" + "reflect" + // "runtime/debug" +) + +// Some tagging information for error messages. +const ( + msgTagDec = "codec.decoder" + msgBadDesc = "Unrecognized descriptor byte" + msgDecCannotExpandArr = "cannot expand go array from %v to stream length: %v" +) + +// decReader abstracts the reading source, allowing implementations that can +// read from an io.Reader or directly off a byte slice with zero-copying. +type decReader interface { + readn(n int) []byte + readb([]byte) + readn1() uint8 + readUint16() uint16 + readUint32() uint32 + readUint64() uint64 +} + +type decDriver interface { + initReadNext() + tryDecodeAsNil() bool + currentEncodedType() valueType + isBuiltinType(rt uintptr) bool + decodeBuiltin(rt uintptr, v interface{}) + //decodeNaked: Numbers are decoded as int64, uint64, float64 only (no smaller sized number types). + decodeNaked() (v interface{}, vt valueType, decodeFurther bool) + decodeInt(bitsize uint8) (i int64) + decodeUint(bitsize uint8) (ui uint64) + decodeFloat(chkOverflow32 bool) (f float64) + decodeBool() (b bool) + // decodeString can also decode symbols + decodeString() (s string) + decodeBytes(bs []byte) (bsOut []byte, changed bool) + decodeExt(verifyTag bool, tag byte) (xtag byte, xbs []byte) + readMapLen() int + readArrayLen() int +} + +type DecodeOptions struct { + // An instance of MapType is used during schema-less decoding of a map in the stream. + // If nil, we use map[interface{}]interface{} + MapType reflect.Type + // An instance of SliceType is used during schema-less decoding of an array in the stream. + // If nil, we use []interface{} + SliceType reflect.Type + // ErrorIfNoField controls whether an error is returned when decoding a map + // from a codec stream into a struct, and no matching struct field is found. + ErrorIfNoField bool +} + +// ------------------------------------ + +// ioDecReader is a decReader that reads off an io.Reader +type ioDecReader struct { + r io.Reader + br io.ByteReader + x [8]byte //temp byte array re-used internally for efficiency +} + +func (z *ioDecReader) readn(n int) (bs []byte) { + if n <= 0 { + return + } + bs = make([]byte, n) + if _, err := io.ReadAtLeast(z.r, bs, n); err != nil { + panic(err) + } + return +} + +func (z *ioDecReader) readb(bs []byte) { + if _, err := io.ReadAtLeast(z.r, bs, len(bs)); err != nil { + panic(err) + } +} + +func (z *ioDecReader) readn1() uint8 { + if z.br != nil { + b, err := z.br.ReadByte() + if err != nil { + panic(err) + } + return b + } + z.readb(z.x[:1]) + return z.x[0] +} + +func (z *ioDecReader) readUint16() uint16 { + z.readb(z.x[:2]) + return bigen.Uint16(z.x[:2]) +} + +func (z *ioDecReader) readUint32() uint32 { + z.readb(z.x[:4]) + return bigen.Uint32(z.x[:4]) +} + +func (z *ioDecReader) readUint64() uint64 { + z.readb(z.x[:8]) + return bigen.Uint64(z.x[:8]) +} + +// ------------------------------------ + +// bytesDecReader is a decReader that reads off a byte slice with zero copying +type bytesDecReader struct { + b []byte // data + c int // cursor + a int // available +} + +func (z *bytesDecReader) consume(n int) (oldcursor int) { + if z.a == 0 { + panic(io.EOF) + } + if n > z.a { + decErr("Trying to read %v bytes. Only %v available", n, z.a) + } + // z.checkAvailable(n) + oldcursor = z.c + z.c = oldcursor + n + z.a = z.a - n + return +} + +func (z *bytesDecReader) readn(n int) (bs []byte) { + if n <= 0 { + return + } + c0 := z.consume(n) + bs = z.b[c0:z.c] + return +} + +func (z *bytesDecReader) readb(bs []byte) { + copy(bs, z.readn(len(bs))) +} + +func (z *bytesDecReader) readn1() uint8 { + c0 := z.consume(1) + return z.b[c0] +} + +// Use binaryEncoding helper for 4 and 8 bits, but inline it for 2 bits +// creating temp slice variable and copying it to helper function is expensive +// for just 2 bits. + +func (z *bytesDecReader) readUint16() uint16 { + c0 := z.consume(2) + return uint16(z.b[c0+1]) | uint16(z.b[c0])<<8 +} + +func (z *bytesDecReader) readUint32() uint32 { + c0 := z.consume(4) + return bigen.Uint32(z.b[c0:z.c]) +} + +func (z *bytesDecReader) readUint64() uint64 { + c0 := z.consume(8) + return bigen.Uint64(z.b[c0:z.c]) +} + +// ------------------------------------ + +// decFnInfo has methods for registering handling decoding of a specific type +// based on some characteristics (builtin, extension, reflect Kind, etc) +type decFnInfo struct { + ti *typeInfo + d *Decoder + dd decDriver + xfFn func(reflect.Value, []byte) error + xfTag byte + array bool +} + +func (f *decFnInfo) builtin(rv reflect.Value) { + f.dd.decodeBuiltin(f.ti.rtid, rv.Addr().Interface()) +} + +func (f *decFnInfo) rawExt(rv reflect.Value) { + xtag, xbs := f.dd.decodeExt(false, 0) + rv.Field(0).SetUint(uint64(xtag)) + rv.Field(1).SetBytes(xbs) +} + +func (f *decFnInfo) ext(rv reflect.Value) { + _, xbs := f.dd.decodeExt(true, f.xfTag) + if fnerr := f.xfFn(rv, xbs); fnerr != nil { + panic(fnerr) + } +} + +func (f *decFnInfo) binaryMarshal(rv reflect.Value) { + var bm binaryUnmarshaler + if f.ti.unmIndir == -1 { + bm = rv.Addr().Interface().(binaryUnmarshaler) + } else if f.ti.unmIndir == 0 { + bm = rv.Interface().(binaryUnmarshaler) + } else { + for j, k := int8(0), f.ti.unmIndir; j < k; j++ { + if rv.IsNil() { + rv.Set(reflect.New(rv.Type().Elem())) + } + rv = rv.Elem() + } + bm = rv.Interface().(binaryUnmarshaler) + } + xbs, _ := f.dd.decodeBytes(nil) + if fnerr := bm.UnmarshalBinary(xbs); fnerr != nil { + panic(fnerr) + } +} + +func (f *decFnInfo) kErr(rv reflect.Value) { + decErr("Unhandled value for kind: %v: %s", rv.Kind(), msgBadDesc) +} + +func (f *decFnInfo) kString(rv reflect.Value) { + rv.SetString(f.dd.decodeString()) +} + +func (f *decFnInfo) kBool(rv reflect.Value) { + rv.SetBool(f.dd.decodeBool()) +} + +func (f *decFnInfo) kInt(rv reflect.Value) { + rv.SetInt(f.dd.decodeInt(intBitsize)) +} + +func (f *decFnInfo) kInt64(rv reflect.Value) { + rv.SetInt(f.dd.decodeInt(64)) +} + +func (f *decFnInfo) kInt32(rv reflect.Value) { + rv.SetInt(f.dd.decodeInt(32)) +} + +func (f *decFnInfo) kInt8(rv reflect.Value) { + rv.SetInt(f.dd.decodeInt(8)) +} + +func (f *decFnInfo) kInt16(rv reflect.Value) { + rv.SetInt(f.dd.decodeInt(16)) +} + +func (f *decFnInfo) kFloat32(rv reflect.Value) { + rv.SetFloat(f.dd.decodeFloat(true)) +} + +func (f *decFnInfo) kFloat64(rv reflect.Value) { + rv.SetFloat(f.dd.decodeFloat(false)) +} + +func (f *decFnInfo) kUint8(rv reflect.Value) { + rv.SetUint(f.dd.decodeUint(8)) +} + +func (f *decFnInfo) kUint64(rv reflect.Value) { + rv.SetUint(f.dd.decodeUint(64)) +} + +func (f *decFnInfo) kUint(rv reflect.Value) { + rv.SetUint(f.dd.decodeUint(uintBitsize)) +} + +func (f *decFnInfo) kUint32(rv reflect.Value) { + rv.SetUint(f.dd.decodeUint(32)) +} + +func (f *decFnInfo) kUint16(rv reflect.Value) { + rv.SetUint(f.dd.decodeUint(16)) +} + +// func (f *decFnInfo) kPtr(rv reflect.Value) { +// debugf(">>>>>>> ??? decode kPtr called - shouldn't get called") +// if rv.IsNil() { +// rv.Set(reflect.New(rv.Type().Elem())) +// } +// f.d.decodeValue(rv.Elem()) +// } + +func (f *decFnInfo) kInterface(rv reflect.Value) { + // debugf("\t===> kInterface") + if !rv.IsNil() { + f.d.decodeValue(rv.Elem()) + return + } + // nil interface: + // use some hieristics to set the nil interface to an + // appropriate value based on the first byte read (byte descriptor bd) + v, vt, decodeFurther := f.dd.decodeNaked() + if vt == valueTypeNil { + return + } + // Cannot decode into nil interface with methods (e.g. error, io.Reader, etc) + // if non-nil value in stream. + if num := f.ti.rt.NumMethod(); num > 0 { + decErr("decodeValue: Cannot decode non-nil codec value into nil %v (%v methods)", + f.ti.rt, num) + } + var rvn reflect.Value + var useRvn bool + switch vt { + case valueTypeMap: + if f.d.h.MapType == nil { + var m2 map[interface{}]interface{} + v = &m2 + } else { + rvn = reflect.New(f.d.h.MapType).Elem() + useRvn = true + } + case valueTypeArray: + if f.d.h.SliceType == nil { + var m2 []interface{} + v = &m2 + } else { + rvn = reflect.New(f.d.h.SliceType).Elem() + useRvn = true + } + case valueTypeExt: + re := v.(*RawExt) + var bfn func(reflect.Value, []byte) error + rvn, bfn = f.d.h.getDecodeExtForTag(re.Tag) + if bfn == nil { + rvn = reflect.ValueOf(*re) + } else if fnerr := bfn(rvn, re.Data); fnerr != nil { + panic(fnerr) + } + rv.Set(rvn) + return + } + if decodeFurther { + if useRvn { + f.d.decodeValue(rvn) + } else if v != nil { + // this v is a pointer, so we need to dereference it when done + f.d.decode(v) + rvn = reflect.ValueOf(v).Elem() + useRvn = true + } + } + if useRvn { + rv.Set(rvn) + } else if v != nil { + rv.Set(reflect.ValueOf(v)) + } +} + +func (f *decFnInfo) kStruct(rv reflect.Value) { + fti := f.ti + if currEncodedType := f.dd.currentEncodedType(); currEncodedType == valueTypeMap { + containerLen := f.dd.readMapLen() + if containerLen == 0 { + return + } + tisfi := fti.sfi + for j := 0; j < containerLen; j++ { + // var rvkencname string + // ddecode(&rvkencname) + f.dd.initReadNext() + rvkencname := f.dd.decodeString() + // rvksi := ti.getForEncName(rvkencname) + if k := fti.indexForEncName(rvkencname); k > -1 { + sfik := tisfi[k] + if sfik.i != -1 { + f.d.decodeValue(rv.Field(int(sfik.i))) + } else { + f.d.decEmbeddedField(rv, sfik.is) + } + // f.d.decodeValue(ti.field(k, rv)) + } else { + if f.d.h.ErrorIfNoField { + decErr("No matching struct field found when decoding stream map with key: %v", + rvkencname) + } else { + var nilintf0 interface{} + f.d.decodeValue(reflect.ValueOf(&nilintf0).Elem()) + } + } + } + } else if currEncodedType == valueTypeArray { + containerLen := f.dd.readArrayLen() + if containerLen == 0 { + return + } + for j, si := range fti.sfip { + if j == containerLen { + break + } + if si.i != -1 { + f.d.decodeValue(rv.Field(int(si.i))) + } else { + f.d.decEmbeddedField(rv, si.is) + } + } + if containerLen > len(fti.sfip) { + // read remaining values and throw away + for j := len(fti.sfip); j < containerLen; j++ { + var nilintf0 interface{} + f.d.decodeValue(reflect.ValueOf(&nilintf0).Elem()) + } + } + } else { + decErr("Only encoded map or array can be decoded into a struct. (valueType: %x)", + currEncodedType) + } +} + +func (f *decFnInfo) kSlice(rv reflect.Value) { + // A slice can be set from a map or array in stream. + currEncodedType := f.dd.currentEncodedType() + + switch currEncodedType { + case valueTypeBytes, valueTypeString: + if f.ti.rtid == uint8SliceTypId || f.ti.rt.Elem().Kind() == reflect.Uint8 { + if bs2, changed2 := f.dd.decodeBytes(rv.Bytes()); changed2 { + rv.SetBytes(bs2) + } + return + } + } + + if shortCircuitReflectToFastPath && rv.CanAddr() { + switch f.ti.rtid { + case intfSliceTypId: + f.d.decSliceIntf(rv.Addr().Interface().(*[]interface{}), currEncodedType, f.array) + return + case uint64SliceTypId: + f.d.decSliceUint64(rv.Addr().Interface().(*[]uint64), currEncodedType, f.array) + return + case int64SliceTypId: + f.d.decSliceInt64(rv.Addr().Interface().(*[]int64), currEncodedType, f.array) + return + case strSliceTypId: + f.d.decSliceStr(rv.Addr().Interface().(*[]string), currEncodedType, f.array) + return + } + } + + containerLen, containerLenS := decContLens(f.dd, currEncodedType) + + // an array can never return a nil slice. so no need to check f.array here. + + if rv.IsNil() { + rv.Set(reflect.MakeSlice(f.ti.rt, containerLenS, containerLenS)) + } + + if containerLen == 0 { + return + } + + if rvcap, rvlen := rv.Len(), rv.Cap(); containerLenS > rvcap { + if f.array { // !rv.CanSet() + decErr(msgDecCannotExpandArr, rvcap, containerLenS) + } + rvn := reflect.MakeSlice(f.ti.rt, containerLenS, containerLenS) + if rvlen > 0 { + reflect.Copy(rvn, rv) + } + rv.Set(rvn) + } else if containerLenS > rvlen { + rv.SetLen(containerLenS) + } + + for j := 0; j < containerLenS; j++ { + f.d.decodeValue(rv.Index(j)) + } +} + +func (f *decFnInfo) kArray(rv reflect.Value) { + // f.d.decodeValue(rv.Slice(0, rv.Len())) + f.kSlice(rv.Slice(0, rv.Len())) +} + +func (f *decFnInfo) kMap(rv reflect.Value) { + if shortCircuitReflectToFastPath && rv.CanAddr() { + switch f.ti.rtid { + case mapStrIntfTypId: + f.d.decMapStrIntf(rv.Addr().Interface().(*map[string]interface{})) + return + case mapIntfIntfTypId: + f.d.decMapIntfIntf(rv.Addr().Interface().(*map[interface{}]interface{})) + return + case mapInt64IntfTypId: + f.d.decMapInt64Intf(rv.Addr().Interface().(*map[int64]interface{})) + return + case mapUint64IntfTypId: + f.d.decMapUint64Intf(rv.Addr().Interface().(*map[uint64]interface{})) + return + } + } + + containerLen := f.dd.readMapLen() + + if rv.IsNil() { + rv.Set(reflect.MakeMap(f.ti.rt)) + } + + if containerLen == 0 { + return + } + + ktype, vtype := f.ti.rt.Key(), f.ti.rt.Elem() + ktypeId := reflect.ValueOf(ktype).Pointer() + for j := 0; j < containerLen; j++ { + rvk := reflect.New(ktype).Elem() + f.d.decodeValue(rvk) + + // special case if a byte array. + // if ktype == intfTyp { + if ktypeId == intfTypId { + rvk = rvk.Elem() + if rvk.Type() == uint8SliceTyp { + rvk = reflect.ValueOf(string(rvk.Bytes())) + } + } + rvv := rv.MapIndex(rvk) + if !rvv.IsValid() { + rvv = reflect.New(vtype).Elem() + } + + f.d.decodeValue(rvv) + rv.SetMapIndex(rvk, rvv) + } +} + +// ---------------------------------------- + +type decFn struct { + i *decFnInfo + f func(*decFnInfo, reflect.Value) +} + +// A Decoder reads and decodes an object from an input stream in the codec format. +type Decoder struct { + r decReader + d decDriver + h *BasicHandle + f map[uintptr]decFn + x []uintptr + s []decFn +} + +// NewDecoder returns a Decoder for decoding a stream of bytes from an io.Reader. +// +// For efficiency, Users are encouraged to pass in a memory buffered writer +// (eg bufio.Reader, bytes.Buffer). +func NewDecoder(r io.Reader, h Handle) *Decoder { + z := ioDecReader{ + r: r, + } + z.br, _ = r.(io.ByteReader) + return &Decoder{r: &z, d: h.newDecDriver(&z), h: h.getBasicHandle()} +} + +// NewDecoderBytes returns a Decoder which efficiently decodes directly +// from a byte slice with zero copying. +func NewDecoderBytes(in []byte, h Handle) *Decoder { + z := bytesDecReader{ + b: in, + a: len(in), + } + return &Decoder{r: &z, d: h.newDecDriver(&z), h: h.getBasicHandle()} +} + +// Decode decodes the stream from reader and stores the result in the +// value pointed to by v. v cannot be a nil pointer. v can also be +// a reflect.Value of a pointer. +// +// Note that a pointer to a nil interface is not a nil pointer. +// If you do not know what type of stream it is, pass in a pointer to a nil interface. +// We will decode and store a value in that nil interface. +// +// Sample usages: +// // Decoding into a non-nil typed value +// var f float32 +// err = codec.NewDecoder(r, handle).Decode(&f) +// +// // Decoding into nil interface +// var v interface{} +// dec := codec.NewDecoder(r, handle) +// err = dec.Decode(&v) +// +// When decoding into a nil interface{}, we will decode into an appropriate value based +// on the contents of the stream: +// - Numbers are decoded as float64, int64 or uint64. +// - Other values are decoded appropriately depending on the type: +// bool, string, []byte, time.Time, etc +// - Extensions are decoded as RawExt (if no ext function registered for the tag) +// Configurations exist on the Handle to override defaults +// (e.g. for MapType, SliceType and how to decode raw bytes). +// +// When decoding into a non-nil interface{} value, the mode of encoding is based on the +// type of the value. When a value is seen: +// - If an extension is registered for it, call that extension function +// - If it implements BinaryUnmarshaler, call its UnmarshalBinary(data []byte) error +// - Else decode it based on its reflect.Kind +// +// There are some special rules when decoding into containers (slice/array/map/struct). +// Decode will typically use the stream contents to UPDATE the container. +// - A map can be decoded from a stream map, by updating matching keys. +// - A slice can be decoded from a stream array, +// by updating the first n elements, where n is length of the stream. +// - A slice can be decoded from a stream map, by decoding as if +// it contains a sequence of key-value pairs. +// - A struct can be decoded from a stream map, by updating matching fields. +// - A struct can be decoded from a stream array, +// by updating fields as they occur in the struct (by index). +// +// When decoding a stream map or array with length of 0 into a nil map or slice, +// we reset the destination map or slice to a zero-length value. +// +// However, when decoding a stream nil, we reset the destination container +// to its "zero" value (e.g. nil for slice/map, etc). +// +func (d *Decoder) Decode(v interface{}) (err error) { + defer panicToErr(&err) + d.decode(v) + return +} + +func (d *Decoder) decode(iv interface{}) { + d.d.initReadNext() + + switch v := iv.(type) { + case nil: + decErr("Cannot decode into nil.") + + case reflect.Value: + d.chkPtrValue(v) + d.decodeValue(v.Elem()) + + case *string: + *v = d.d.decodeString() + case *bool: + *v = d.d.decodeBool() + case *int: + *v = int(d.d.decodeInt(intBitsize)) + case *int8: + *v = int8(d.d.decodeInt(8)) + case *int16: + *v = int16(d.d.decodeInt(16)) + case *int32: + *v = int32(d.d.decodeInt(32)) + case *int64: + *v = d.d.decodeInt(64) + case *uint: + *v = uint(d.d.decodeUint(uintBitsize)) + case *uint8: + *v = uint8(d.d.decodeUint(8)) + case *uint16: + *v = uint16(d.d.decodeUint(16)) + case *uint32: + *v = uint32(d.d.decodeUint(32)) + case *uint64: + *v = d.d.decodeUint(64) + case *float32: + *v = float32(d.d.decodeFloat(true)) + case *float64: + *v = d.d.decodeFloat(false) + case *[]byte: + *v, _ = d.d.decodeBytes(*v) + + case *[]interface{}: + d.decSliceIntf(v, valueTypeInvalid, false) + case *[]uint64: + d.decSliceUint64(v, valueTypeInvalid, false) + case *[]int64: + d.decSliceInt64(v, valueTypeInvalid, false) + case *[]string: + d.decSliceStr(v, valueTypeInvalid, false) + case *map[string]interface{}: + d.decMapStrIntf(v) + case *map[interface{}]interface{}: + d.decMapIntfIntf(v) + case *map[uint64]interface{}: + d.decMapUint64Intf(v) + case *map[int64]interface{}: + d.decMapInt64Intf(v) + + case *interface{}: + d.decodeValue(reflect.ValueOf(iv).Elem()) + + default: + rv := reflect.ValueOf(iv) + d.chkPtrValue(rv) + d.decodeValue(rv.Elem()) + } +} + +func (d *Decoder) decodeValue(rv reflect.Value) { + d.d.initReadNext() + + if d.d.tryDecodeAsNil() { + // If value in stream is nil, set the dereferenced value to its "zero" value (if settable) + if rv.Kind() == reflect.Ptr { + if !rv.IsNil() { + rv.Set(reflect.Zero(rv.Type())) + } + return + } + // for rv.Kind() == reflect.Ptr { + // rv = rv.Elem() + // } + if rv.IsValid() { // rv.CanSet() // always settable, except it's invalid + rv.Set(reflect.Zero(rv.Type())) + } + return + } + + // If stream is not containing a nil value, then we can deref to the base + // non-pointer value, and decode into that. + for rv.Kind() == reflect.Ptr { + if rv.IsNil() { + rv.Set(reflect.New(rv.Type().Elem())) + } + rv = rv.Elem() + } + + rt := rv.Type() + rtid := reflect.ValueOf(rt).Pointer() + + // retrieve or register a focus'ed function for this type + // to eliminate need to do the retrieval multiple times + + // if d.f == nil && d.s == nil { debugf("---->Creating new dec f map for type: %v\n", rt) } + var fn decFn + var ok bool + if useMapForCodecCache { + fn, ok = d.f[rtid] + } else { + for i, v := range d.x { + if v == rtid { + fn, ok = d.s[i], true + break + } + } + } + if !ok { + // debugf("\tCreating new dec fn for type: %v\n", rt) + fi := decFnInfo{ti: getTypeInfo(rtid, rt), d: d, dd: d.d} + fn.i = &fi + // An extension can be registered for any type, regardless of the Kind + // (e.g. type BitSet int64, type MyStruct { / * unexported fields * / }, type X []int, etc. + // + // We can't check if it's an extension byte here first, because the user may have + // registered a pointer or non-pointer type, meaning we may have to recurse first + // before matching a mapped type, even though the extension byte is already detected. + // + // NOTE: if decoding into a nil interface{}, we return a non-nil + // value except even if the container registers a length of 0. + if rtid == rawExtTypId { + fn.f = (*decFnInfo).rawExt + } else if d.d.isBuiltinType(rtid) { + fn.f = (*decFnInfo).builtin + } else if xfTag, xfFn := d.h.getDecodeExt(rtid); xfFn != nil { + fi.xfTag, fi.xfFn = xfTag, xfFn + fn.f = (*decFnInfo).ext + } else if supportBinaryMarshal && fi.ti.unm { + fn.f = (*decFnInfo).binaryMarshal + } else { + switch rk := rt.Kind(); rk { + case reflect.String: + fn.f = (*decFnInfo).kString + case reflect.Bool: + fn.f = (*decFnInfo).kBool + case reflect.Int: + fn.f = (*decFnInfo).kInt + case reflect.Int64: + fn.f = (*decFnInfo).kInt64 + case reflect.Int32: + fn.f = (*decFnInfo).kInt32 + case reflect.Int8: + fn.f = (*decFnInfo).kInt8 + case reflect.Int16: + fn.f = (*decFnInfo).kInt16 + case reflect.Float32: + fn.f = (*decFnInfo).kFloat32 + case reflect.Float64: + fn.f = (*decFnInfo).kFloat64 + case reflect.Uint8: + fn.f = (*decFnInfo).kUint8 + case reflect.Uint64: + fn.f = (*decFnInfo).kUint64 + case reflect.Uint: + fn.f = (*decFnInfo).kUint + case reflect.Uint32: + fn.f = (*decFnInfo).kUint32 + case reflect.Uint16: + fn.f = (*decFnInfo).kUint16 + // case reflect.Ptr: + // fn.f = (*decFnInfo).kPtr + case reflect.Interface: + fn.f = (*decFnInfo).kInterface + case reflect.Struct: + fn.f = (*decFnInfo).kStruct + case reflect.Slice: + fn.f = (*decFnInfo).kSlice + case reflect.Array: + fi.array = true + fn.f = (*decFnInfo).kArray + case reflect.Map: + fn.f = (*decFnInfo).kMap + default: + fn.f = (*decFnInfo).kErr + } + } + if useMapForCodecCache { + if d.f == nil { + d.f = make(map[uintptr]decFn, 16) + } + d.f[rtid] = fn + } else { + d.s = append(d.s, fn) + d.x = append(d.x, rtid) + } + } + + fn.f(fn.i, rv) + + return +} + +func (d *Decoder) chkPtrValue(rv reflect.Value) { + // We can only decode into a non-nil pointer + if rv.Kind() == reflect.Ptr && !rv.IsNil() { + return + } + if !rv.IsValid() { + decErr("Cannot decode into a zero (ie invalid) reflect.Value") + } + if !rv.CanInterface() { + decErr("Cannot decode into a value without an interface: %v", rv) + } + rvi := rv.Interface() + decErr("Cannot decode into non-pointer or nil pointer. Got: %v, %T, %v", + rv.Kind(), rvi, rvi) +} + +func (d *Decoder) decEmbeddedField(rv reflect.Value, index []int) { + // d.decodeValue(rv.FieldByIndex(index)) + // nil pointers may be here; so reproduce FieldByIndex logic + enhancements + for _, j := range index { + if rv.Kind() == reflect.Ptr { + if rv.IsNil() { + rv.Set(reflect.New(rv.Type().Elem())) + } + // If a pointer, it must be a pointer to struct (based on typeInfo contract) + rv = rv.Elem() + } + rv = rv.Field(j) + } + d.decodeValue(rv) +} + +// -------------------------------------------------- + +// short circuit functions for common maps and slices + +func (d *Decoder) decSliceIntf(v *[]interface{}, currEncodedType valueType, doNotReset bool) { + _, containerLenS := decContLens(d.d, currEncodedType) + s := *v + if s == nil { + s = make([]interface{}, containerLenS, containerLenS) + } else if containerLenS > cap(s) { + if doNotReset { + decErr(msgDecCannotExpandArr, cap(s), containerLenS) + } + s = make([]interface{}, containerLenS, containerLenS) + copy(s, *v) + } else if containerLenS > len(s) { + s = s[:containerLenS] + } + for j := 0; j < containerLenS; j++ { + d.decode(&s[j]) + } + *v = s +} + +func (d *Decoder) decSliceInt64(v *[]int64, currEncodedType valueType, doNotReset bool) { + _, containerLenS := decContLens(d.d, currEncodedType) + s := *v + if s == nil { + s = make([]int64, containerLenS, containerLenS) + } else if containerLenS > cap(s) { + if doNotReset { + decErr(msgDecCannotExpandArr, cap(s), containerLenS) + } + s = make([]int64, containerLenS, containerLenS) + copy(s, *v) + } else if containerLenS > len(s) { + s = s[:containerLenS] + } + for j := 0; j < containerLenS; j++ { + // d.decode(&s[j]) + d.d.initReadNext() + s[j] = d.d.decodeInt(intBitsize) + } + *v = s +} + +func (d *Decoder) decSliceUint64(v *[]uint64, currEncodedType valueType, doNotReset bool) { + _, containerLenS := decContLens(d.d, currEncodedType) + s := *v + if s == nil { + s = make([]uint64, containerLenS, containerLenS) + } else if containerLenS > cap(s) { + if doNotReset { + decErr(msgDecCannotExpandArr, cap(s), containerLenS) + } + s = make([]uint64, containerLenS, containerLenS) + copy(s, *v) + } else if containerLenS > len(s) { + s = s[:containerLenS] + } + for j := 0; j < containerLenS; j++ { + // d.decode(&s[j]) + d.d.initReadNext() + s[j] = d.d.decodeUint(intBitsize) + } + *v = s +} + +func (d *Decoder) decSliceStr(v *[]string, currEncodedType valueType, doNotReset bool) { + _, containerLenS := decContLens(d.d, currEncodedType) + s := *v + if s == nil { + s = make([]string, containerLenS, containerLenS) + } else if containerLenS > cap(s) { + if doNotReset { + decErr(msgDecCannotExpandArr, cap(s), containerLenS) + } + s = make([]string, containerLenS, containerLenS) + copy(s, *v) + } else if containerLenS > len(s) { + s = s[:containerLenS] + } + for j := 0; j < containerLenS; j++ { + // d.decode(&s[j]) + d.d.initReadNext() + s[j] = d.d.decodeString() + } + *v = s +} + +func (d *Decoder) decMapIntfIntf(v *map[interface{}]interface{}) { + containerLen := d.d.readMapLen() + m := *v + if m == nil { + m = make(map[interface{}]interface{}, containerLen) + *v = m + } + for j := 0; j < containerLen; j++ { + var mk interface{} + d.decode(&mk) + // special case if a byte array. + if bv, bok := mk.([]byte); bok { + mk = string(bv) + } + mv := m[mk] + d.decode(&mv) + m[mk] = mv + } +} + +func (d *Decoder) decMapInt64Intf(v *map[int64]interface{}) { + containerLen := d.d.readMapLen() + m := *v + if m == nil { + m = make(map[int64]interface{}, containerLen) + *v = m + } + for j := 0; j < containerLen; j++ { + d.d.initReadNext() + mk := d.d.decodeInt(intBitsize) + mv := m[mk] + d.decode(&mv) + m[mk] = mv + } +} + +func (d *Decoder) decMapUint64Intf(v *map[uint64]interface{}) { + containerLen := d.d.readMapLen() + m := *v + if m == nil { + m = make(map[uint64]interface{}, containerLen) + *v = m + } + for j := 0; j < containerLen; j++ { + d.d.initReadNext() + mk := d.d.decodeUint(intBitsize) + mv := m[mk] + d.decode(&mv) + m[mk] = mv + } +} + +func (d *Decoder) decMapStrIntf(v *map[string]interface{}) { + containerLen := d.d.readMapLen() + m := *v + if m == nil { + m = make(map[string]interface{}, containerLen) + *v = m + } + for j := 0; j < containerLen; j++ { + d.d.initReadNext() + mk := d.d.decodeString() + mv := m[mk] + d.decode(&mv) + m[mk] = mv + } +} + +// ---------------------------------------- + +func decContLens(dd decDriver, currEncodedType valueType) (containerLen, containerLenS int) { + if currEncodedType == valueTypeInvalid { + currEncodedType = dd.currentEncodedType() + } + switch currEncodedType { + case valueTypeArray: + containerLen = dd.readArrayLen() + containerLenS = containerLen + case valueTypeMap: + containerLen = dd.readMapLen() + containerLenS = containerLen * 2 + default: + decErr("Only encoded map or array can be decoded into a slice. (valueType: %0x)", + currEncodedType) + } + return +} + +func decErr(format string, params ...interface{}) { + doPanic(msgTagDec, format, params...) +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/encode.go b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/encode.go new file mode 100644 index 0000000000000..4914be0c748bf --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/encode.go @@ -0,0 +1,1001 @@ +// Copyright (c) 2012, 2013 Ugorji Nwoke. All rights reserved. +// Use of this source code is governed by a BSD-style license found in the LICENSE file. + +package codec + +import ( + "io" + "reflect" +) + +const ( + // Some tagging information for error messages. + msgTagEnc = "codec.encoder" + defEncByteBufSize = 1 << 6 // 4:16, 6:64, 8:256, 10:1024 + // maxTimeSecs32 = math.MaxInt32 / 60 / 24 / 366 +) + +// AsSymbolFlag defines what should be encoded as symbols. +type AsSymbolFlag uint8 + +const ( + // AsSymbolDefault is default. + // Currently, this means only encode struct field names as symbols. + // The default is subject to change. + AsSymbolDefault AsSymbolFlag = iota + + // AsSymbolAll means encode anything which could be a symbol as a symbol. + AsSymbolAll = 0xfe + + // AsSymbolNone means do not encode anything as a symbol. + AsSymbolNone = 1 << iota + + // AsSymbolMapStringKeys means encode keys in map[string]XXX as symbols. + AsSymbolMapStringKeysFlag + + // AsSymbolStructFieldName means encode struct field names as symbols. + AsSymbolStructFieldNameFlag +) + +// encWriter abstracting writing to a byte array or to an io.Writer. +type encWriter interface { + writeUint16(uint16) + writeUint32(uint32) + writeUint64(uint64) + writeb([]byte) + writestr(string) + writen1(byte) + writen2(byte, byte) + atEndOfEncode() +} + +// encDriver abstracts the actual codec (binc vs msgpack, etc) +type encDriver interface { + isBuiltinType(rt uintptr) bool + encodeBuiltin(rt uintptr, v interface{}) + encodeNil() + encodeInt(i int64) + encodeUint(i uint64) + encodeBool(b bool) + encodeFloat32(f float32) + encodeFloat64(f float64) + encodeExtPreamble(xtag byte, length int) + encodeArrayPreamble(length int) + encodeMapPreamble(length int) + encodeString(c charEncoding, v string) + encodeSymbol(v string) + encodeStringBytes(c charEncoding, v []byte) + //TODO + //encBignum(f *big.Int) + //encStringRunes(c charEncoding, v []rune) +} + +type ioEncWriterWriter interface { + WriteByte(c byte) error + WriteString(s string) (n int, err error) + Write(p []byte) (n int, err error) +} + +type ioEncStringWriter interface { + WriteString(s string) (n int, err error) +} + +type EncodeOptions struct { + // Encode a struct as an array, and not as a map. + StructToArray bool + + // AsSymbols defines what should be encoded as symbols. + // + // Encoding as symbols can reduce the encoded size significantly. + // + // However, during decoding, each string to be encoded as a symbol must + // be checked to see if it has been seen before. Consequently, encoding time + // will increase if using symbols, because string comparisons has a clear cost. + // + // Sample values: + // AsSymbolNone + // AsSymbolAll + // AsSymbolMapStringKeys + // AsSymbolMapStringKeysFlag | AsSymbolStructFieldNameFlag + AsSymbols AsSymbolFlag +} + +// --------------------------------------------- + +type simpleIoEncWriterWriter struct { + w io.Writer + bw io.ByteWriter + sw ioEncStringWriter +} + +func (o *simpleIoEncWriterWriter) WriteByte(c byte) (err error) { + if o.bw != nil { + return o.bw.WriteByte(c) + } + _, err = o.w.Write([]byte{c}) + return +} + +func (o *simpleIoEncWriterWriter) WriteString(s string) (n int, err error) { + if o.sw != nil { + return o.sw.WriteString(s) + } + return o.w.Write([]byte(s)) +} + +func (o *simpleIoEncWriterWriter) Write(p []byte) (n int, err error) { + return o.w.Write(p) +} + +// ---------------------------------------- + +// ioEncWriter implements encWriter and can write to an io.Writer implementation +type ioEncWriter struct { + w ioEncWriterWriter + x [8]byte // temp byte array re-used internally for efficiency +} + +func (z *ioEncWriter) writeUint16(v uint16) { + bigen.PutUint16(z.x[:2], v) + z.writeb(z.x[:2]) +} + +func (z *ioEncWriter) writeUint32(v uint32) { + bigen.PutUint32(z.x[:4], v) + z.writeb(z.x[:4]) +} + +func (z *ioEncWriter) writeUint64(v uint64) { + bigen.PutUint64(z.x[:8], v) + z.writeb(z.x[:8]) +} + +func (z *ioEncWriter) writeb(bs []byte) { + if len(bs) == 0 { + return + } + n, err := z.w.Write(bs) + if err != nil { + panic(err) + } + if n != len(bs) { + encErr("write: Incorrect num bytes written. Expecting: %v, Wrote: %v", len(bs), n) + } +} + +func (z *ioEncWriter) writestr(s string) { + n, err := z.w.WriteString(s) + if err != nil { + panic(err) + } + if n != len(s) { + encErr("write: Incorrect num bytes written. Expecting: %v, Wrote: %v", len(s), n) + } +} + +func (z *ioEncWriter) writen1(b byte) { + if err := z.w.WriteByte(b); err != nil { + panic(err) + } +} + +func (z *ioEncWriter) writen2(b1 byte, b2 byte) { + z.writen1(b1) + z.writen1(b2) +} + +func (z *ioEncWriter) atEndOfEncode() {} + +// ---------------------------------------- + +// bytesEncWriter implements encWriter and can write to an byte slice. +// It is used by Marshal function. +type bytesEncWriter struct { + b []byte + c int // cursor + out *[]byte // write out on atEndOfEncode +} + +func (z *bytesEncWriter) writeUint16(v uint16) { + c := z.grow(2) + z.b[c] = byte(v >> 8) + z.b[c+1] = byte(v) +} + +func (z *bytesEncWriter) writeUint32(v uint32) { + c := z.grow(4) + z.b[c] = byte(v >> 24) + z.b[c+1] = byte(v >> 16) + z.b[c+2] = byte(v >> 8) + z.b[c+3] = byte(v) +} + +func (z *bytesEncWriter) writeUint64(v uint64) { + c := z.grow(8) + z.b[c] = byte(v >> 56) + z.b[c+1] = byte(v >> 48) + z.b[c+2] = byte(v >> 40) + z.b[c+3] = byte(v >> 32) + z.b[c+4] = byte(v >> 24) + z.b[c+5] = byte(v >> 16) + z.b[c+6] = byte(v >> 8) + z.b[c+7] = byte(v) +} + +func (z *bytesEncWriter) writeb(s []byte) { + if len(s) == 0 { + return + } + c := z.grow(len(s)) + copy(z.b[c:], s) +} + +func (z *bytesEncWriter) writestr(s string) { + c := z.grow(len(s)) + copy(z.b[c:], s) +} + +func (z *bytesEncWriter) writen1(b1 byte) { + c := z.grow(1) + z.b[c] = b1 +} + +func (z *bytesEncWriter) writen2(b1 byte, b2 byte) { + c := z.grow(2) + z.b[c] = b1 + z.b[c+1] = b2 +} + +func (z *bytesEncWriter) atEndOfEncode() { + *(z.out) = z.b[:z.c] +} + +func (z *bytesEncWriter) grow(n int) (oldcursor int) { + oldcursor = z.c + z.c = oldcursor + n + if z.c > cap(z.b) { + // Tried using appendslice logic: (if cap < 1024, *2, else *1.25). + // However, it was too expensive, causing too many iterations of copy. + // Using bytes.Buffer model was much better (2*cap + n) + bs := make([]byte, 2*cap(z.b)+n) + copy(bs, z.b[:oldcursor]) + z.b = bs + } else if z.c > len(z.b) { + z.b = z.b[:cap(z.b)] + } + return +} + +// --------------------------------------------- + +type encFnInfo struct { + ti *typeInfo + e *Encoder + ee encDriver + xfFn func(reflect.Value) ([]byte, error) + xfTag byte +} + +func (f *encFnInfo) builtin(rv reflect.Value) { + f.ee.encodeBuiltin(f.ti.rtid, rv.Interface()) +} + +func (f *encFnInfo) rawExt(rv reflect.Value) { + f.e.encRawExt(rv.Interface().(RawExt)) +} + +func (f *encFnInfo) ext(rv reflect.Value) { + bs, fnerr := f.xfFn(rv) + if fnerr != nil { + panic(fnerr) + } + if bs == nil { + f.ee.encodeNil() + return + } + if f.e.hh.writeExt() { + f.ee.encodeExtPreamble(f.xfTag, len(bs)) + f.e.w.writeb(bs) + } else { + f.ee.encodeStringBytes(c_RAW, bs) + } + +} + +func (f *encFnInfo) binaryMarshal(rv reflect.Value) { + var bm binaryMarshaler + if f.ti.mIndir == 0 { + bm = rv.Interface().(binaryMarshaler) + } else if f.ti.mIndir == -1 { + bm = rv.Addr().Interface().(binaryMarshaler) + } else { + for j, k := int8(0), f.ti.mIndir; j < k; j++ { + if rv.IsNil() { + f.ee.encodeNil() + return + } + rv = rv.Elem() + } + bm = rv.Interface().(binaryMarshaler) + } + // debugf(">>>> binaryMarshaler: %T", rv.Interface()) + bs, fnerr := bm.MarshalBinary() + if fnerr != nil { + panic(fnerr) + } + if bs == nil { + f.ee.encodeNil() + } else { + f.ee.encodeStringBytes(c_RAW, bs) + } +} + +func (f *encFnInfo) kBool(rv reflect.Value) { + f.ee.encodeBool(rv.Bool()) +} + +func (f *encFnInfo) kString(rv reflect.Value) { + f.ee.encodeString(c_UTF8, rv.String()) +} + +func (f *encFnInfo) kFloat64(rv reflect.Value) { + f.ee.encodeFloat64(rv.Float()) +} + +func (f *encFnInfo) kFloat32(rv reflect.Value) { + f.ee.encodeFloat32(float32(rv.Float())) +} + +func (f *encFnInfo) kInt(rv reflect.Value) { + f.ee.encodeInt(rv.Int()) +} + +func (f *encFnInfo) kUint(rv reflect.Value) { + f.ee.encodeUint(rv.Uint()) +} + +func (f *encFnInfo) kInvalid(rv reflect.Value) { + f.ee.encodeNil() +} + +func (f *encFnInfo) kErr(rv reflect.Value) { + encErr("Unsupported kind: %s, for: %#v", rv.Kind(), rv) +} + +func (f *encFnInfo) kSlice(rv reflect.Value) { + if rv.IsNil() { + f.ee.encodeNil() + return + } + + if shortCircuitReflectToFastPath { + switch f.ti.rtid { + case intfSliceTypId: + f.e.encSliceIntf(rv.Interface().([]interface{})) + return + case strSliceTypId: + f.e.encSliceStr(rv.Interface().([]string)) + return + case uint64SliceTypId: + f.e.encSliceUint64(rv.Interface().([]uint64)) + return + case int64SliceTypId: + f.e.encSliceInt64(rv.Interface().([]int64)) + return + } + } + + // If in this method, then there was no extension function defined. + // So it's okay to treat as []byte. + if f.ti.rtid == uint8SliceTypId || f.ti.rt.Elem().Kind() == reflect.Uint8 { + f.ee.encodeStringBytes(c_RAW, rv.Bytes()) + return + } + + l := rv.Len() + if f.ti.mbs { + if l%2 == 1 { + encErr("mapBySlice: invalid length (must be divisible by 2): %v", l) + } + f.ee.encodeMapPreamble(l / 2) + } else { + f.ee.encodeArrayPreamble(l) + } + if l == 0 { + return + } + for j := 0; j < l; j++ { + // TODO: Consider perf implication of encoding odd index values as symbols if type is string + f.e.encodeValue(rv.Index(j)) + } +} + +func (f *encFnInfo) kArray(rv reflect.Value) { + // We cannot share kSlice method, because the array may be non-addressable. + // E.g. type struct S{B [2]byte}; Encode(S{}) will bomb on "panic: slice of unaddressable array". + // So we have to duplicate the functionality here. + // f.e.encodeValue(rv.Slice(0, rv.Len())) + // f.kSlice(rv.Slice(0, rv.Len())) + + l := rv.Len() + // Handle an array of bytes specially (in line with what is done for slices) + if f.ti.rt.Elem().Kind() == reflect.Uint8 { + if l == 0 { + f.ee.encodeStringBytes(c_RAW, nil) + return + } + var bs []byte + if rv.CanAddr() { + bs = rv.Slice(0, l).Bytes() + } else { + bs = make([]byte, l) + for i := 0; i < l; i++ { + bs[i] = byte(rv.Index(i).Uint()) + } + } + f.ee.encodeStringBytes(c_RAW, bs) + return + } + + if f.ti.mbs { + if l%2 == 1 { + encErr("mapBySlice: invalid length (must be divisible by 2): %v", l) + } + f.ee.encodeMapPreamble(l / 2) + } else { + f.ee.encodeArrayPreamble(l) + } + if l == 0 { + return + } + for j := 0; j < l; j++ { + // TODO: Consider perf implication of encoding odd index values as symbols if type is string + f.e.encodeValue(rv.Index(j)) + } +} + +func (f *encFnInfo) kStruct(rv reflect.Value) { + fti := f.ti + newlen := len(fti.sfi) + rvals := make([]reflect.Value, newlen) + var encnames []string + e := f.e + tisfi := fti.sfip + toMap := !(fti.toArray || e.h.StructToArray) + // if toMap, use the sorted array. If toArray, use unsorted array (to match sequence in struct) + if toMap { + tisfi = fti.sfi + encnames = make([]string, newlen) + } + newlen = 0 + for _, si := range tisfi { + if si.i != -1 { + rvals[newlen] = rv.Field(int(si.i)) + } else { + rvals[newlen] = rv.FieldByIndex(si.is) + } + if toMap { + if si.omitEmpty && isEmptyValue(rvals[newlen]) { + continue + } + encnames[newlen] = si.encName + } else { + if si.omitEmpty && isEmptyValue(rvals[newlen]) { + rvals[newlen] = reflect.Value{} //encode as nil + } + } + newlen++ + } + + // debugf(">>>> kStruct: newlen: %v", newlen) + if toMap { + ee := f.ee //don't dereference everytime + ee.encodeMapPreamble(newlen) + // asSymbols := e.h.AsSymbols&AsSymbolStructFieldNameFlag != 0 + asSymbols := e.h.AsSymbols == AsSymbolDefault || e.h.AsSymbols&AsSymbolStructFieldNameFlag != 0 + for j := 0; j < newlen; j++ { + if asSymbols { + ee.encodeSymbol(encnames[j]) + } else { + ee.encodeString(c_UTF8, encnames[j]) + } + e.encodeValue(rvals[j]) + } + } else { + f.ee.encodeArrayPreamble(newlen) + for j := 0; j < newlen; j++ { + e.encodeValue(rvals[j]) + } + } +} + +// func (f *encFnInfo) kPtr(rv reflect.Value) { +// debugf(">>>>>>> ??? encode kPtr called - shouldn't get called") +// if rv.IsNil() { +// f.ee.encodeNil() +// return +// } +// f.e.encodeValue(rv.Elem()) +// } + +func (f *encFnInfo) kInterface(rv reflect.Value) { + if rv.IsNil() { + f.ee.encodeNil() + return + } + f.e.encodeValue(rv.Elem()) +} + +func (f *encFnInfo) kMap(rv reflect.Value) { + if rv.IsNil() { + f.ee.encodeNil() + return + } + + if shortCircuitReflectToFastPath { + switch f.ti.rtid { + case mapIntfIntfTypId: + f.e.encMapIntfIntf(rv.Interface().(map[interface{}]interface{})) + return + case mapStrIntfTypId: + f.e.encMapStrIntf(rv.Interface().(map[string]interface{})) + return + case mapStrStrTypId: + f.e.encMapStrStr(rv.Interface().(map[string]string)) + return + case mapInt64IntfTypId: + f.e.encMapInt64Intf(rv.Interface().(map[int64]interface{})) + return + case mapUint64IntfTypId: + f.e.encMapUint64Intf(rv.Interface().(map[uint64]interface{})) + return + } + } + + l := rv.Len() + f.ee.encodeMapPreamble(l) + if l == 0 { + return + } + // keyTypeIsString := f.ti.rt.Key().Kind() == reflect.String + keyTypeIsString := f.ti.rt.Key() == stringTyp + var asSymbols bool + if keyTypeIsString { + asSymbols = f.e.h.AsSymbols&AsSymbolMapStringKeysFlag != 0 + } + mks := rv.MapKeys() + // for j, lmks := 0, len(mks); j < lmks; j++ { + for j := range mks { + if keyTypeIsString { + if asSymbols { + f.ee.encodeSymbol(mks[j].String()) + } else { + f.ee.encodeString(c_UTF8, mks[j].String()) + } + } else { + f.e.encodeValue(mks[j]) + } + f.e.encodeValue(rv.MapIndex(mks[j])) + } + +} + +// -------------------------------------------------- + +// encFn encapsulates the captured variables and the encode function. +// This way, we only do some calculations one times, and pass to the +// code block that should be called (encapsulated in a function) +// instead of executing the checks every time. +type encFn struct { + i *encFnInfo + f func(*encFnInfo, reflect.Value) +} + +// -------------------------------------------------- + +// An Encoder writes an object to an output stream in the codec format. +type Encoder struct { + w encWriter + e encDriver + h *BasicHandle + hh Handle + f map[uintptr]encFn + x []uintptr + s []encFn +} + +// NewEncoder returns an Encoder for encoding into an io.Writer. +// +// For efficiency, Users are encouraged to pass in a memory buffered writer +// (eg bufio.Writer, bytes.Buffer). +func NewEncoder(w io.Writer, h Handle) *Encoder { + ww, ok := w.(ioEncWriterWriter) + if !ok { + sww := simpleIoEncWriterWriter{w: w} + sww.bw, _ = w.(io.ByteWriter) + sww.sw, _ = w.(ioEncStringWriter) + ww = &sww + //ww = bufio.NewWriterSize(w, defEncByteBufSize) + } + z := ioEncWriter{ + w: ww, + } + return &Encoder{w: &z, hh: h, h: h.getBasicHandle(), e: h.newEncDriver(&z)} +} + +// NewEncoderBytes returns an encoder for encoding directly and efficiently +// into a byte slice, using zero-copying to temporary slices. +// +// It will potentially replace the output byte slice pointed to. +// After encoding, the out parameter contains the encoded contents. +func NewEncoderBytes(out *[]byte, h Handle) *Encoder { + in := *out + if in == nil { + in = make([]byte, defEncByteBufSize) + } + z := bytesEncWriter{ + b: in, + out: out, + } + return &Encoder{w: &z, hh: h, h: h.getBasicHandle(), e: h.newEncDriver(&z)} +} + +// Encode writes an object into a stream in the codec format. +// +// Encoding can be configured via the "codec" struct tag for the fields. +// +// The "codec" key in struct field's tag value is the key name, +// followed by an optional comma and options. +// +// To set an option on all fields (e.g. omitempty on all fields), you +// can create a field called _struct, and set flags on it. +// +// Struct values "usually" encode as maps. Each exported struct field is encoded unless: +// - the field's codec tag is "-", OR +// - the field is empty and its codec tag specifies the "omitempty" option. +// +// When encoding as a map, the first string in the tag (before the comma) +// is the map key string to use when encoding. +// +// However, struct values may encode as arrays. This happens when: +// - StructToArray Encode option is set, OR +// - the codec tag on the _struct field sets the "toarray" option +// +// Values with types that implement MapBySlice are encoded as stream maps. +// +// The empty values (for omitempty option) are false, 0, any nil pointer +// or interface value, and any array, slice, map, or string of length zero. +// +// Anonymous fields are encoded inline if no struct tag is present. +// Else they are encoded as regular fields. +// +// Examples: +// +// type MyStruct struct { +// _struct bool `codec:",omitempty"` //set omitempty for every field +// Field1 string `codec:"-"` //skip this field +// Field2 int `codec:"myName"` //Use key "myName" in encode stream +// Field3 int32 `codec:",omitempty"` //use key "Field3". Omit if empty. +// Field4 bool `codec:"f4,omitempty"` //use key "f4". Omit if empty. +// ... +// } +// +// type MyStruct struct { +// _struct bool `codec:",omitempty,toarray"` //set omitempty for every field +// //and encode struct as an array +// } +// +// The mode of encoding is based on the type of the value. When a value is seen: +// - If an extension is registered for it, call that extension function +// - If it implements BinaryMarshaler, call its MarshalBinary() (data []byte, err error) +// - Else encode it based on its reflect.Kind +// +// Note that struct field names and keys in map[string]XXX will be treated as symbols. +// Some formats support symbols (e.g. binc) and will properly encode the string +// only once in the stream, and use a tag to refer to it thereafter. +func (e *Encoder) Encode(v interface{}) (err error) { + defer panicToErr(&err) + e.encode(v) + e.w.atEndOfEncode() + return +} + +func (e *Encoder) encode(iv interface{}) { + switch v := iv.(type) { + case nil: + e.e.encodeNil() + + case reflect.Value: + e.encodeValue(v) + + case string: + e.e.encodeString(c_UTF8, v) + case bool: + e.e.encodeBool(v) + case int: + e.e.encodeInt(int64(v)) + case int8: + e.e.encodeInt(int64(v)) + case int16: + e.e.encodeInt(int64(v)) + case int32: + e.e.encodeInt(int64(v)) + case int64: + e.e.encodeInt(v) + case uint: + e.e.encodeUint(uint64(v)) + case uint8: + e.e.encodeUint(uint64(v)) + case uint16: + e.e.encodeUint(uint64(v)) + case uint32: + e.e.encodeUint(uint64(v)) + case uint64: + e.e.encodeUint(v) + case float32: + e.e.encodeFloat32(v) + case float64: + e.e.encodeFloat64(v) + + case []interface{}: + e.encSliceIntf(v) + case []string: + e.encSliceStr(v) + case []int64: + e.encSliceInt64(v) + case []uint64: + e.encSliceUint64(v) + case []uint8: + e.e.encodeStringBytes(c_RAW, v) + + case map[interface{}]interface{}: + e.encMapIntfIntf(v) + case map[string]interface{}: + e.encMapStrIntf(v) + case map[string]string: + e.encMapStrStr(v) + case map[int64]interface{}: + e.encMapInt64Intf(v) + case map[uint64]interface{}: + e.encMapUint64Intf(v) + + case *string: + e.e.encodeString(c_UTF8, *v) + case *bool: + e.e.encodeBool(*v) + case *int: + e.e.encodeInt(int64(*v)) + case *int8: + e.e.encodeInt(int64(*v)) + case *int16: + e.e.encodeInt(int64(*v)) + case *int32: + e.e.encodeInt(int64(*v)) + case *int64: + e.e.encodeInt(*v) + case *uint: + e.e.encodeUint(uint64(*v)) + case *uint8: + e.e.encodeUint(uint64(*v)) + case *uint16: + e.e.encodeUint(uint64(*v)) + case *uint32: + e.e.encodeUint(uint64(*v)) + case *uint64: + e.e.encodeUint(*v) + case *float32: + e.e.encodeFloat32(*v) + case *float64: + e.e.encodeFloat64(*v) + + case *[]interface{}: + e.encSliceIntf(*v) + case *[]string: + e.encSliceStr(*v) + case *[]int64: + e.encSliceInt64(*v) + case *[]uint64: + e.encSliceUint64(*v) + case *[]uint8: + e.e.encodeStringBytes(c_RAW, *v) + + case *map[interface{}]interface{}: + e.encMapIntfIntf(*v) + case *map[string]interface{}: + e.encMapStrIntf(*v) + case *map[string]string: + e.encMapStrStr(*v) + case *map[int64]interface{}: + e.encMapInt64Intf(*v) + case *map[uint64]interface{}: + e.encMapUint64Intf(*v) + + default: + e.encodeValue(reflect.ValueOf(iv)) + } +} + +func (e *Encoder) encodeValue(rv reflect.Value) { + for rv.Kind() == reflect.Ptr { + if rv.IsNil() { + e.e.encodeNil() + return + } + rv = rv.Elem() + } + + rt := rv.Type() + rtid := reflect.ValueOf(rt).Pointer() + + // if e.f == nil && e.s == nil { debugf("---->Creating new enc f map for type: %v\n", rt) } + var fn encFn + var ok bool + if useMapForCodecCache { + fn, ok = e.f[rtid] + } else { + for i, v := range e.x { + if v == rtid { + fn, ok = e.s[i], true + break + } + } + } + if !ok { + // debugf("\tCreating new enc fn for type: %v\n", rt) + fi := encFnInfo{ti: getTypeInfo(rtid, rt), e: e, ee: e.e} + fn.i = &fi + if rtid == rawExtTypId { + fn.f = (*encFnInfo).rawExt + } else if e.e.isBuiltinType(rtid) { + fn.f = (*encFnInfo).builtin + } else if xfTag, xfFn := e.h.getEncodeExt(rtid); xfFn != nil { + fi.xfTag, fi.xfFn = xfTag, xfFn + fn.f = (*encFnInfo).ext + } else if supportBinaryMarshal && fi.ti.m { + fn.f = (*encFnInfo).binaryMarshal + } else { + switch rk := rt.Kind(); rk { + case reflect.Bool: + fn.f = (*encFnInfo).kBool + case reflect.String: + fn.f = (*encFnInfo).kString + case reflect.Float64: + fn.f = (*encFnInfo).kFloat64 + case reflect.Float32: + fn.f = (*encFnInfo).kFloat32 + case reflect.Int, reflect.Int8, reflect.Int64, reflect.Int32, reflect.Int16: + fn.f = (*encFnInfo).kInt + case reflect.Uint8, reflect.Uint64, reflect.Uint, reflect.Uint32, reflect.Uint16: + fn.f = (*encFnInfo).kUint + case reflect.Invalid: + fn.f = (*encFnInfo).kInvalid + case reflect.Slice: + fn.f = (*encFnInfo).kSlice + case reflect.Array: + fn.f = (*encFnInfo).kArray + case reflect.Struct: + fn.f = (*encFnInfo).kStruct + // case reflect.Ptr: + // fn.f = (*encFnInfo).kPtr + case reflect.Interface: + fn.f = (*encFnInfo).kInterface + case reflect.Map: + fn.f = (*encFnInfo).kMap + default: + fn.f = (*encFnInfo).kErr + } + } + if useMapForCodecCache { + if e.f == nil { + e.f = make(map[uintptr]encFn, 16) + } + e.f[rtid] = fn + } else { + e.s = append(e.s, fn) + e.x = append(e.x, rtid) + } + } + + fn.f(fn.i, rv) + +} + +func (e *Encoder) encRawExt(re RawExt) { + if re.Data == nil { + e.e.encodeNil() + return + } + if e.hh.writeExt() { + e.e.encodeExtPreamble(re.Tag, len(re.Data)) + e.w.writeb(re.Data) + } else { + e.e.encodeStringBytes(c_RAW, re.Data) + } +} + +// --------------------------------------------- +// short circuit functions for common maps and slices + +func (e *Encoder) encSliceIntf(v []interface{}) { + e.e.encodeArrayPreamble(len(v)) + for _, v2 := range v { + e.encode(v2) + } +} + +func (e *Encoder) encSliceStr(v []string) { + e.e.encodeArrayPreamble(len(v)) + for _, v2 := range v { + e.e.encodeString(c_UTF8, v2) + } +} + +func (e *Encoder) encSliceInt64(v []int64) { + e.e.encodeArrayPreamble(len(v)) + for _, v2 := range v { + e.e.encodeInt(v2) + } +} + +func (e *Encoder) encSliceUint64(v []uint64) { + e.e.encodeArrayPreamble(len(v)) + for _, v2 := range v { + e.e.encodeUint(v2) + } +} + +func (e *Encoder) encMapStrStr(v map[string]string) { + e.e.encodeMapPreamble(len(v)) + asSymbols := e.h.AsSymbols&AsSymbolMapStringKeysFlag != 0 + for k2, v2 := range v { + if asSymbols { + e.e.encodeSymbol(k2) + } else { + e.e.encodeString(c_UTF8, k2) + } + e.e.encodeString(c_UTF8, v2) + } +} + +func (e *Encoder) encMapStrIntf(v map[string]interface{}) { + e.e.encodeMapPreamble(len(v)) + asSymbols := e.h.AsSymbols&AsSymbolMapStringKeysFlag != 0 + for k2, v2 := range v { + if asSymbols { + e.e.encodeSymbol(k2) + } else { + e.e.encodeString(c_UTF8, k2) + } + e.encode(v2) + } +} + +func (e *Encoder) encMapInt64Intf(v map[int64]interface{}) { + e.e.encodeMapPreamble(len(v)) + for k2, v2 := range v { + e.e.encodeInt(k2) + e.encode(v2) + } +} + +func (e *Encoder) encMapUint64Intf(v map[uint64]interface{}) { + e.e.encodeMapPreamble(len(v)) + for k2, v2 := range v { + e.e.encodeUint(uint64(k2)) + e.encode(v2) + } +} + +func (e *Encoder) encMapIntfIntf(v map[interface{}]interface{}) { + e.e.encodeMapPreamble(len(v)) + for k2, v2 := range v { + e.encode(k2) + e.encode(v2) + } +} + +// ---------------------------------------- + +func encErr(format string, params ...interface{}) { + doPanic(msgTagEnc, format, params...) +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/ext_dep_test.go b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/ext_dep_test.go new file mode 100644 index 0000000000000..bdf448d5210aa --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/ext_dep_test.go @@ -0,0 +1,75 @@ +// //+build ignore + +// Copyright (c) 2012, 2013 Ugorji Nwoke. All rights reserved. +// Use of this source code is governed by a BSD-style license found in the LICENSE file. + +package codec + +// This file includes benchmarks which have dependencies on 3rdparty +// packages (bson and vmihailenco/msgpack) which must be installed locally. +// +// To run the benchmarks including these 3rdparty packages, first +// - Uncomment first line in this file (put // // in front of it) +// - Get those packages: +// go get github.com/vmihailenco/msgpack +// go get labix.org/v2/mgo/bson +// - Run: +// go test -bi -bench=. + +import ( + "testing" + + vmsgpack "gopkg.in/vmihailenco/msgpack.v2" + "labix.org/v2/mgo/bson" +) + +func init() { + benchCheckers = append(benchCheckers, + benchChecker{"v-msgpack", fnVMsgpackEncodeFn, fnVMsgpackDecodeFn}, + benchChecker{"bson", fnBsonEncodeFn, fnBsonDecodeFn}, + ) +} + +func fnVMsgpackEncodeFn(ts interface{}) ([]byte, error) { + return vmsgpack.Marshal(ts) +} + +func fnVMsgpackDecodeFn(buf []byte, ts interface{}) error { + return vmsgpack.Unmarshal(buf, ts) +} + +func fnBsonEncodeFn(ts interface{}) ([]byte, error) { + return bson.Marshal(ts) +} + +func fnBsonDecodeFn(buf []byte, ts interface{}) error { + return bson.Unmarshal(buf, ts) +} + +func Benchmark__Bson_______Encode(b *testing.B) { + fnBenchmarkEncode(b, "bson", benchTs, fnBsonEncodeFn) +} + +func Benchmark__Bson_______Decode(b *testing.B) { + fnBenchmarkDecode(b, "bson", benchTs, fnBsonEncodeFn, fnBsonDecodeFn, fnBenchNewTs) +} + +func Benchmark__VMsgpack___Encode(b *testing.B) { + fnBenchmarkEncode(b, "v-msgpack", benchTs, fnVMsgpackEncodeFn) +} + +func Benchmark__VMsgpack___Decode(b *testing.B) { + fnBenchmarkDecode(b, "v-msgpack", benchTs, fnVMsgpackEncodeFn, fnVMsgpackDecodeFn, fnBenchNewTs) +} + +func TestMsgpackPythonGenStreams(t *testing.T) { + doTestMsgpackPythonGenStreams(t) +} + +func TestMsgpackRpcSpecGoClientToPythonSvc(t *testing.T) { + doTestMsgpackRpcSpecGoClientToPythonSvc(t) +} + +func TestMsgpackRpcSpecPythonClientToGoSvc(t *testing.T) { + doTestMsgpackRpcSpecPythonClientToGoSvc(t) +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/helper.go b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/helper.go new file mode 100644 index 0000000000000..e6dc0563f090c --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/helper.go @@ -0,0 +1,589 @@ +// Copyright (c) 2012, 2013 Ugorji Nwoke. All rights reserved. +// Use of this source code is governed by a BSD-style license found in the LICENSE file. + +package codec + +// Contains code shared by both encode and decode. + +import ( + "encoding/binary" + "fmt" + "math" + "reflect" + "sort" + "strings" + "sync" + "time" + "unicode" + "unicode/utf8" +) + +const ( + structTagName = "codec" + + // Support + // encoding.BinaryMarshaler: MarshalBinary() (data []byte, err error) + // encoding.BinaryUnmarshaler: UnmarshalBinary(data []byte) error + // This constant flag will enable or disable it. + supportBinaryMarshal = true + + // Each Encoder or Decoder uses a cache of functions based on conditionals, + // so that the conditionals are not run every time. + // + // Either a map or a slice is used to keep track of the functions. + // The map is more natural, but has a higher cost than a slice/array. + // This flag (useMapForCodecCache) controls which is used. + useMapForCodecCache = false + + // For some common container types, we can short-circuit an elaborate + // reflection dance and call encode/decode directly. + // The currently supported types are: + // - slices of strings, or id's (int64,uint64) or interfaces. + // - maps of str->str, str->intf, id(int64,uint64)->intf, intf->intf + shortCircuitReflectToFastPath = true + + // for debugging, set this to false, to catch panic traces. + // Note that this will always cause rpc tests to fail, since they need io.EOF sent via panic. + recoverPanicToErr = true +) + +type charEncoding uint8 + +const ( + c_RAW charEncoding = iota + c_UTF8 + c_UTF16LE + c_UTF16BE + c_UTF32LE + c_UTF32BE +) + +// valueType is the stream type +type valueType uint8 + +const ( + valueTypeUnset valueType = iota + valueTypeNil + valueTypeInt + valueTypeUint + valueTypeFloat + valueTypeBool + valueTypeString + valueTypeSymbol + valueTypeBytes + valueTypeMap + valueTypeArray + valueTypeTimestamp + valueTypeExt + + valueTypeInvalid = 0xff +) + +var ( + bigen = binary.BigEndian + structInfoFieldName = "_struct" + + cachedTypeInfo = make(map[uintptr]*typeInfo, 4) + cachedTypeInfoMutex sync.RWMutex + + intfSliceTyp = reflect.TypeOf([]interface{}(nil)) + intfTyp = intfSliceTyp.Elem() + + strSliceTyp = reflect.TypeOf([]string(nil)) + boolSliceTyp = reflect.TypeOf([]bool(nil)) + uintSliceTyp = reflect.TypeOf([]uint(nil)) + uint8SliceTyp = reflect.TypeOf([]uint8(nil)) + uint16SliceTyp = reflect.TypeOf([]uint16(nil)) + uint32SliceTyp = reflect.TypeOf([]uint32(nil)) + uint64SliceTyp = reflect.TypeOf([]uint64(nil)) + intSliceTyp = reflect.TypeOf([]int(nil)) + int8SliceTyp = reflect.TypeOf([]int8(nil)) + int16SliceTyp = reflect.TypeOf([]int16(nil)) + int32SliceTyp = reflect.TypeOf([]int32(nil)) + int64SliceTyp = reflect.TypeOf([]int64(nil)) + float32SliceTyp = reflect.TypeOf([]float32(nil)) + float64SliceTyp = reflect.TypeOf([]float64(nil)) + + mapIntfIntfTyp = reflect.TypeOf(map[interface{}]interface{}(nil)) + mapStrIntfTyp = reflect.TypeOf(map[string]interface{}(nil)) + mapStrStrTyp = reflect.TypeOf(map[string]string(nil)) + + mapIntIntfTyp = reflect.TypeOf(map[int]interface{}(nil)) + mapInt64IntfTyp = reflect.TypeOf(map[int64]interface{}(nil)) + mapUintIntfTyp = reflect.TypeOf(map[uint]interface{}(nil)) + mapUint64IntfTyp = reflect.TypeOf(map[uint64]interface{}(nil)) + + stringTyp = reflect.TypeOf("") + timeTyp = reflect.TypeOf(time.Time{}) + rawExtTyp = reflect.TypeOf(RawExt{}) + + mapBySliceTyp = reflect.TypeOf((*MapBySlice)(nil)).Elem() + binaryMarshalerTyp = reflect.TypeOf((*binaryMarshaler)(nil)).Elem() + binaryUnmarshalerTyp = reflect.TypeOf((*binaryUnmarshaler)(nil)).Elem() + + rawExtTypId = reflect.ValueOf(rawExtTyp).Pointer() + intfTypId = reflect.ValueOf(intfTyp).Pointer() + timeTypId = reflect.ValueOf(timeTyp).Pointer() + + intfSliceTypId = reflect.ValueOf(intfSliceTyp).Pointer() + strSliceTypId = reflect.ValueOf(strSliceTyp).Pointer() + + boolSliceTypId = reflect.ValueOf(boolSliceTyp).Pointer() + uintSliceTypId = reflect.ValueOf(uintSliceTyp).Pointer() + uint8SliceTypId = reflect.ValueOf(uint8SliceTyp).Pointer() + uint16SliceTypId = reflect.ValueOf(uint16SliceTyp).Pointer() + uint32SliceTypId = reflect.ValueOf(uint32SliceTyp).Pointer() + uint64SliceTypId = reflect.ValueOf(uint64SliceTyp).Pointer() + intSliceTypId = reflect.ValueOf(intSliceTyp).Pointer() + int8SliceTypId = reflect.ValueOf(int8SliceTyp).Pointer() + int16SliceTypId = reflect.ValueOf(int16SliceTyp).Pointer() + int32SliceTypId = reflect.ValueOf(int32SliceTyp).Pointer() + int64SliceTypId = reflect.ValueOf(int64SliceTyp).Pointer() + float32SliceTypId = reflect.ValueOf(float32SliceTyp).Pointer() + float64SliceTypId = reflect.ValueOf(float64SliceTyp).Pointer() + + mapStrStrTypId = reflect.ValueOf(mapStrStrTyp).Pointer() + mapIntfIntfTypId = reflect.ValueOf(mapIntfIntfTyp).Pointer() + mapStrIntfTypId = reflect.ValueOf(mapStrIntfTyp).Pointer() + mapIntIntfTypId = reflect.ValueOf(mapIntIntfTyp).Pointer() + mapInt64IntfTypId = reflect.ValueOf(mapInt64IntfTyp).Pointer() + mapUintIntfTypId = reflect.ValueOf(mapUintIntfTyp).Pointer() + mapUint64IntfTypId = reflect.ValueOf(mapUint64IntfTyp).Pointer() + // Id = reflect.ValueOf().Pointer() + // mapBySliceTypId = reflect.ValueOf(mapBySliceTyp).Pointer() + + binaryMarshalerTypId = reflect.ValueOf(binaryMarshalerTyp).Pointer() + binaryUnmarshalerTypId = reflect.ValueOf(binaryUnmarshalerTyp).Pointer() + + intBitsize uint8 = uint8(reflect.TypeOf(int(0)).Bits()) + uintBitsize uint8 = uint8(reflect.TypeOf(uint(0)).Bits()) + + bsAll0x00 = []byte{0, 0, 0, 0, 0, 0, 0, 0} + bsAll0xff = []byte{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff} +) + +type binaryUnmarshaler interface { + UnmarshalBinary(data []byte) error +} + +type binaryMarshaler interface { + MarshalBinary() (data []byte, err error) +} + +// MapBySlice represents a slice which should be encoded as a map in the stream. +// The slice contains a sequence of key-value pairs. +type MapBySlice interface { + MapBySlice() +} + +// WARNING: DO NOT USE DIRECTLY. EXPORTED FOR GODOC BENEFIT. WILL BE REMOVED. +// +// BasicHandle encapsulates the common options and extension functions. +type BasicHandle struct { + extHandle + EncodeOptions + DecodeOptions +} + +// Handle is the interface for a specific encoding format. +// +// Typically, a Handle is pre-configured before first time use, +// and not modified while in use. Such a pre-configured Handle +// is safe for concurrent access. +type Handle interface { + writeExt() bool + getBasicHandle() *BasicHandle + newEncDriver(w encWriter) encDriver + newDecDriver(r decReader) decDriver +} + +// RawExt represents raw unprocessed extension data. +type RawExt struct { + Tag byte + Data []byte +} + +type extTypeTagFn struct { + rtid uintptr + rt reflect.Type + tag byte + encFn func(reflect.Value) ([]byte, error) + decFn func(reflect.Value, []byte) error +} + +type extHandle []*extTypeTagFn + +// AddExt registers an encode and decode function for a reflect.Type. +// Note that the type must be a named type, and specifically not +// a pointer or Interface. An error is returned if that is not honored. +// +// To Deregister an ext, call AddExt with 0 tag, nil encfn and nil decfn. +func (o *extHandle) AddExt( + rt reflect.Type, + tag byte, + encfn func(reflect.Value) ([]byte, error), + decfn func(reflect.Value, []byte) error, +) (err error) { + // o is a pointer, because we may need to initialize it + if rt.PkgPath() == "" || rt.Kind() == reflect.Interface { + err = fmt.Errorf("codec.Handle.AddExt: Takes named type, especially not a pointer or interface: %T", + reflect.Zero(rt).Interface()) + return + } + + // o cannot be nil, since it is always embedded in a Handle. + // if nil, let it panic. + // if o == nil { + // err = errors.New("codec.Handle.AddExt: extHandle cannot be a nil pointer.") + // return + // } + + rtid := reflect.ValueOf(rt).Pointer() + for _, v := range *o { + if v.rtid == rtid { + v.tag, v.encFn, v.decFn = tag, encfn, decfn + return + } + } + + *o = append(*o, &extTypeTagFn{rtid, rt, tag, encfn, decfn}) + return +} + +func (o extHandle) getExt(rtid uintptr) *extTypeTagFn { + for _, v := range o { + if v.rtid == rtid { + return v + } + } + return nil +} + +func (o extHandle) getExtForTag(tag byte) *extTypeTagFn { + for _, v := range o { + if v.tag == tag { + return v + } + } + return nil +} + +func (o extHandle) getDecodeExtForTag(tag byte) ( + rv reflect.Value, fn func(reflect.Value, []byte) error) { + if x := o.getExtForTag(tag); x != nil { + // ext is only registered for base + rv = reflect.New(x.rt).Elem() + fn = x.decFn + } + return +} + +func (o extHandle) getDecodeExt(rtid uintptr) (tag byte, fn func(reflect.Value, []byte) error) { + if x := o.getExt(rtid); x != nil { + tag = x.tag + fn = x.decFn + } + return +} + +func (o extHandle) getEncodeExt(rtid uintptr) (tag byte, fn func(reflect.Value) ([]byte, error)) { + if x := o.getExt(rtid); x != nil { + tag = x.tag + fn = x.encFn + } + return +} + +type structFieldInfo struct { + encName string // encode name + + // only one of 'i' or 'is' can be set. If 'i' is -1, then 'is' has been set. + + is []int // (recursive/embedded) field index in struct + i int16 // field index in struct + omitEmpty bool + toArray bool // if field is _struct, is the toArray set? + + // tag string // tag + // name string // field name + // encNameBs []byte // encoded name as byte stream + // ikind int // kind of the field as an int i.e. int(reflect.Kind) +} + +func parseStructFieldInfo(fname string, stag string) *structFieldInfo { + if fname == "" { + panic("parseStructFieldInfo: No Field Name") + } + si := structFieldInfo{ + // name: fname, + encName: fname, + // tag: stag, + } + + if stag != "" { + for i, s := range strings.Split(stag, ",") { + if i == 0 { + if s != "" { + si.encName = s + } + } else { + switch s { + case "omitempty": + si.omitEmpty = true + case "toarray": + si.toArray = true + } + } + } + } + // si.encNameBs = []byte(si.encName) + return &si +} + +type sfiSortedByEncName []*structFieldInfo + +func (p sfiSortedByEncName) Len() int { + return len(p) +} + +func (p sfiSortedByEncName) Less(i, j int) bool { + return p[i].encName < p[j].encName +} + +func (p sfiSortedByEncName) Swap(i, j int) { + p[i], p[j] = p[j], p[i] +} + +// typeInfo keeps information about each type referenced in the encode/decode sequence. +// +// During an encode/decode sequence, we work as below: +// - If base is a built in type, en/decode base value +// - If base is registered as an extension, en/decode base value +// - If type is binary(M/Unm)arshaler, call Binary(M/Unm)arshal method +// - Else decode appropriately based on the reflect.Kind +type typeInfo struct { + sfi []*structFieldInfo // sorted. Used when enc/dec struct to map. + sfip []*structFieldInfo // unsorted. Used when enc/dec struct to array. + + rt reflect.Type + rtid uintptr + + // baseId gives pointer to the base reflect.Type, after deferencing + // the pointers. E.g. base type of ***time.Time is time.Time. + base reflect.Type + baseId uintptr + baseIndir int8 // number of indirections to get to base + + mbs bool // base type (T or *T) is a MapBySlice + + m bool // base type (T or *T) is a binaryMarshaler + unm bool // base type (T or *T) is a binaryUnmarshaler + mIndir int8 // number of indirections to get to binaryMarshaler type + unmIndir int8 // number of indirections to get to binaryUnmarshaler type + toArray bool // whether this (struct) type should be encoded as an array +} + +func (ti *typeInfo) indexForEncName(name string) int { + //tisfi := ti.sfi + const binarySearchThreshold = 16 + if sfilen := len(ti.sfi); sfilen < binarySearchThreshold { + // linear search. faster than binary search in my testing up to 16-field structs. + for i, si := range ti.sfi { + if si.encName == name { + return i + } + } + } else { + // binary search. adapted from sort/search.go. + h, i, j := 0, 0, sfilen + for i < j { + h = i + (j-i)/2 + if ti.sfi[h].encName < name { + i = h + 1 + } else { + j = h + } + } + if i < sfilen && ti.sfi[i].encName == name { + return i + } + } + return -1 +} + +func getTypeInfo(rtid uintptr, rt reflect.Type) (pti *typeInfo) { + var ok bool + cachedTypeInfoMutex.RLock() + pti, ok = cachedTypeInfo[rtid] + cachedTypeInfoMutex.RUnlock() + if ok { + return + } + + cachedTypeInfoMutex.Lock() + defer cachedTypeInfoMutex.Unlock() + if pti, ok = cachedTypeInfo[rtid]; ok { + return + } + + ti := typeInfo{rt: rt, rtid: rtid} + pti = &ti + + var indir int8 + if ok, indir = implementsIntf(rt, binaryMarshalerTyp); ok { + ti.m, ti.mIndir = true, indir + } + if ok, indir = implementsIntf(rt, binaryUnmarshalerTyp); ok { + ti.unm, ti.unmIndir = true, indir + } + if ok, _ = implementsIntf(rt, mapBySliceTyp); ok { + ti.mbs = true + } + + pt := rt + var ptIndir int8 + // for ; pt.Kind() == reflect.Ptr; pt, ptIndir = pt.Elem(), ptIndir+1 { } + for pt.Kind() == reflect.Ptr { + pt = pt.Elem() + ptIndir++ + } + if ptIndir == 0 { + ti.base = rt + ti.baseId = rtid + } else { + ti.base = pt + ti.baseId = reflect.ValueOf(pt).Pointer() + ti.baseIndir = ptIndir + } + + if rt.Kind() == reflect.Struct { + var siInfo *structFieldInfo + if f, ok := rt.FieldByName(structInfoFieldName); ok { + siInfo = parseStructFieldInfo(structInfoFieldName, f.Tag.Get(structTagName)) + ti.toArray = siInfo.toArray + } + sfip := make([]*structFieldInfo, 0, rt.NumField()) + rgetTypeInfo(rt, nil, make(map[string]bool), &sfip, siInfo) + + // // try to put all si close together + // const tryToPutAllStructFieldInfoTogether = true + // if tryToPutAllStructFieldInfoTogether { + // sfip2 := make([]structFieldInfo, len(sfip)) + // for i, si := range sfip { + // sfip2[i] = *si + // } + // for i := range sfip { + // sfip[i] = &sfip2[i] + // } + // } + + ti.sfip = make([]*structFieldInfo, len(sfip)) + ti.sfi = make([]*structFieldInfo, len(sfip)) + copy(ti.sfip, sfip) + sort.Sort(sfiSortedByEncName(sfip)) + copy(ti.sfi, sfip) + } + // sfi = sfip + cachedTypeInfo[rtid] = pti + return +} + +func rgetTypeInfo(rt reflect.Type, indexstack []int, fnameToHastag map[string]bool, + sfi *[]*structFieldInfo, siInfo *structFieldInfo, +) { + // for rt.Kind() == reflect.Ptr { + // // indexstack = append(indexstack, 0) + // rt = rt.Elem() + // } + for j := 0; j < rt.NumField(); j++ { + f := rt.Field(j) + stag := f.Tag.Get(structTagName) + if stag == "-" { + continue + } + if r1, _ := utf8.DecodeRuneInString(f.Name); r1 == utf8.RuneError || !unicode.IsUpper(r1) { + continue + } + // if anonymous and there is no struct tag and its a struct (or pointer to struct), inline it. + if f.Anonymous && stag == "" { + ft := f.Type + for ft.Kind() == reflect.Ptr { + ft = ft.Elem() + } + if ft.Kind() == reflect.Struct { + indexstack2 := append(append(make([]int, 0, len(indexstack)+4), indexstack...), j) + rgetTypeInfo(ft, indexstack2, fnameToHastag, sfi, siInfo) + continue + } + } + // do not let fields with same name in embedded structs override field at higher level. + // this must be done after anonymous check, to allow anonymous field + // still include their child fields + if _, ok := fnameToHastag[f.Name]; ok { + continue + } + si := parseStructFieldInfo(f.Name, stag) + // si.ikind = int(f.Type.Kind()) + if len(indexstack) == 0 { + si.i = int16(j) + } else { + si.i = -1 + si.is = append(append(make([]int, 0, len(indexstack)+4), indexstack...), j) + } + + if siInfo != nil { + if siInfo.omitEmpty { + si.omitEmpty = true + } + } + *sfi = append(*sfi, si) + fnameToHastag[f.Name] = stag != "" + } +} + +func panicToErr(err *error) { + if recoverPanicToErr { + if x := recover(); x != nil { + //debug.PrintStack() + panicValToErr(x, err) + } + } +} + +func doPanic(tag string, format string, params ...interface{}) { + params2 := make([]interface{}, len(params)+1) + params2[0] = tag + copy(params2[1:], params) + panic(fmt.Errorf("%s: "+format, params2...)) +} + +func checkOverflowFloat32(f float64, doCheck bool) { + if !doCheck { + return + } + // check overflow (logic adapted from std pkg reflect/value.go OverflowFloat() + f2 := f + if f2 < 0 { + f2 = -f + } + if math.MaxFloat32 < f2 && f2 <= math.MaxFloat64 { + decErr("Overflow float32 value: %v", f2) + } +} + +func checkOverflow(ui uint64, i int64, bitsize uint8) { + // check overflow (logic adapted from std pkg reflect/value.go OverflowUint() + if bitsize == 0 { + return + } + if i != 0 { + if trunc := (i << (64 - bitsize)) >> (64 - bitsize); i != trunc { + decErr("Overflow int value: %v", i) + } + } + if ui != 0 { + if trunc := (ui << (64 - bitsize)) >> (64 - bitsize); ui != trunc { + decErr("Overflow uint value: %v", ui) + } + } +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/helper_internal.go b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/helper_internal.go new file mode 100644 index 0000000000000..58417da958ffd --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/helper_internal.go @@ -0,0 +1,127 @@ +// Copyright (c) 2012, 2013 Ugorji Nwoke. All rights reserved. +// Use of this source code is governed by a BSD-style license found in the LICENSE file. + +package codec + +// All non-std package dependencies live in this file, +// so porting to different environment is easy (just update functions). + +import ( + "errors" + "fmt" + "math" + "reflect" +) + +var ( + raisePanicAfterRecover = false + debugging = true +) + +func panicValToErr(panicVal interface{}, err *error) { + switch xerr := panicVal.(type) { + case error: + *err = xerr + case string: + *err = errors.New(xerr) + default: + *err = fmt.Errorf("%v", panicVal) + } + if raisePanicAfterRecover { + panic(panicVal) + } + return +} + +func isEmptyValueDeref(v reflect.Value, deref bool) bool { + switch v.Kind() { + case reflect.Array, reflect.Map, reflect.Slice, reflect.String: + return v.Len() == 0 + case reflect.Bool: + return !v.Bool() + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return v.Int() == 0 + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + return v.Uint() == 0 + case reflect.Float32, reflect.Float64: + return v.Float() == 0 + case reflect.Interface, reflect.Ptr: + if deref { + if v.IsNil() { + return true + } + return isEmptyValueDeref(v.Elem(), deref) + } else { + return v.IsNil() + } + case reflect.Struct: + // return true if all fields are empty. else return false. + + // we cannot use equality check, because some fields may be maps/slices/etc + // and consequently the structs are not comparable. + // return v.Interface() == reflect.Zero(v.Type()).Interface() + for i, n := 0, v.NumField(); i < n; i++ { + if !isEmptyValueDeref(v.Field(i), deref) { + return false + } + } + return true + } + return false +} + +func isEmptyValue(v reflect.Value) bool { + return isEmptyValueDeref(v, true) +} + +func debugf(format string, args ...interface{}) { + if debugging { + if len(format) == 0 || format[len(format)-1] != '\n' { + format = format + "\n" + } + fmt.Printf(format, args...) + } +} + +func pruneSignExt(v []byte, pos bool) (n int) { + if len(v) < 2 { + } else if pos && v[0] == 0 { + for ; v[n] == 0 && n+1 < len(v) && (v[n+1]&(1<<7) == 0); n++ { + } + } else if !pos && v[0] == 0xff { + for ; v[n] == 0xff && n+1 < len(v) && (v[n+1]&(1<<7) != 0); n++ { + } + } + return +} + +func implementsIntf(typ, iTyp reflect.Type) (success bool, indir int8) { + if typ == nil { + return + } + rt := typ + // The type might be a pointer and we need to keep + // dereferencing to the base type until we find an implementation. + for { + if rt.Implements(iTyp) { + return true, indir + } + if p := rt; p.Kind() == reflect.Ptr { + indir++ + if indir >= math.MaxInt8 { // insane number of indirections + return false, 0 + } + rt = p.Elem() + continue + } + break + } + // No luck yet, but if this is a base type (non-pointer), the pointer might satisfy. + if typ.Kind() != reflect.Ptr { + // Not a pointer, but does the pointer work? + if reflect.PtrTo(typ).Implements(iTyp) { + return true, -1 + } + } + return false, 0 +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/msgpack.go b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/msgpack.go new file mode 100644 index 0000000000000..da0500d19223b --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/msgpack.go @@ -0,0 +1,816 @@ +// Copyright (c) 2012, 2013 Ugorji Nwoke. All rights reserved. +// Use of this source code is governed by a BSD-style license found in the LICENSE file. + +/* +MSGPACK + +Msgpack-c implementation powers the c, c++, python, ruby, etc libraries. +We need to maintain compatibility with it and how it encodes integer values +without caring about the type. + +For compatibility with behaviour of msgpack-c reference implementation: + - Go intX (>0) and uintX + IS ENCODED AS + msgpack +ve fixnum, unsigned + - Go intX (<0) + IS ENCODED AS + msgpack -ve fixnum, signed + +*/ +package codec + +import ( + "fmt" + "io" + "math" + "net/rpc" +) + +const ( + mpPosFixNumMin byte = 0x00 + mpPosFixNumMax = 0x7f + mpFixMapMin = 0x80 + mpFixMapMax = 0x8f + mpFixArrayMin = 0x90 + mpFixArrayMax = 0x9f + mpFixStrMin = 0xa0 + mpFixStrMax = 0xbf + mpNil = 0xc0 + _ = 0xc1 + mpFalse = 0xc2 + mpTrue = 0xc3 + mpFloat = 0xca + mpDouble = 0xcb + mpUint8 = 0xcc + mpUint16 = 0xcd + mpUint32 = 0xce + mpUint64 = 0xcf + mpInt8 = 0xd0 + mpInt16 = 0xd1 + mpInt32 = 0xd2 + mpInt64 = 0xd3 + + // extensions below + mpBin8 = 0xc4 + mpBin16 = 0xc5 + mpBin32 = 0xc6 + mpExt8 = 0xc7 + mpExt16 = 0xc8 + mpExt32 = 0xc9 + mpFixExt1 = 0xd4 + mpFixExt2 = 0xd5 + mpFixExt4 = 0xd6 + mpFixExt8 = 0xd7 + mpFixExt16 = 0xd8 + + mpStr8 = 0xd9 // new + mpStr16 = 0xda + mpStr32 = 0xdb + + mpArray16 = 0xdc + mpArray32 = 0xdd + + mpMap16 = 0xde + mpMap32 = 0xdf + + mpNegFixNumMin = 0xe0 + mpNegFixNumMax = 0xff +) + +// MsgpackSpecRpcMultiArgs is a special type which signifies to the MsgpackSpecRpcCodec +// that the backend RPC service takes multiple arguments, which have been arranged +// in sequence in the slice. +// +// The Codec then passes it AS-IS to the rpc service (without wrapping it in an +// array of 1 element). +type MsgpackSpecRpcMultiArgs []interface{} + +// A MsgpackContainer type specifies the different types of msgpackContainers. +type msgpackContainerType struct { + fixCutoff int + bFixMin, b8, b16, b32 byte + hasFixMin, has8, has8Always bool +} + +var ( + msgpackContainerStr = msgpackContainerType{32, mpFixStrMin, mpStr8, mpStr16, mpStr32, true, true, false} + msgpackContainerBin = msgpackContainerType{0, 0, mpBin8, mpBin16, mpBin32, false, true, true} + msgpackContainerList = msgpackContainerType{16, mpFixArrayMin, 0, mpArray16, mpArray32, true, false, false} + msgpackContainerMap = msgpackContainerType{16, mpFixMapMin, 0, mpMap16, mpMap32, true, false, false} +) + +//--------------------------------------------- + +type msgpackEncDriver struct { + w encWriter + h *MsgpackHandle +} + +func (e *msgpackEncDriver) isBuiltinType(rt uintptr) bool { + //no builtin types. All encodings are based on kinds. Types supported as extensions. + return false +} + +func (e *msgpackEncDriver) encodeBuiltin(rt uintptr, v interface{}) {} + +func (e *msgpackEncDriver) encodeNil() { + e.w.writen1(mpNil) +} + +func (e *msgpackEncDriver) encodeInt(i int64) { + + switch { + case i >= 0: + e.encodeUint(uint64(i)) + case i >= -32: + e.w.writen1(byte(i)) + case i >= math.MinInt8: + e.w.writen2(mpInt8, byte(i)) + case i >= math.MinInt16: + e.w.writen1(mpInt16) + e.w.writeUint16(uint16(i)) + case i >= math.MinInt32: + e.w.writen1(mpInt32) + e.w.writeUint32(uint32(i)) + default: + e.w.writen1(mpInt64) + e.w.writeUint64(uint64(i)) + } +} + +func (e *msgpackEncDriver) encodeUint(i uint64) { + switch { + case i <= math.MaxInt8: + e.w.writen1(byte(i)) + case i <= math.MaxUint8: + e.w.writen2(mpUint8, byte(i)) + case i <= math.MaxUint16: + e.w.writen1(mpUint16) + e.w.writeUint16(uint16(i)) + case i <= math.MaxUint32: + e.w.writen1(mpUint32) + e.w.writeUint32(uint32(i)) + default: + e.w.writen1(mpUint64) + e.w.writeUint64(uint64(i)) + } +} + +func (e *msgpackEncDriver) encodeBool(b bool) { + if b { + e.w.writen1(mpTrue) + } else { + e.w.writen1(mpFalse) + } +} + +func (e *msgpackEncDriver) encodeFloat32(f float32) { + e.w.writen1(mpFloat) + e.w.writeUint32(math.Float32bits(f)) +} + +func (e *msgpackEncDriver) encodeFloat64(f float64) { + e.w.writen1(mpDouble) + e.w.writeUint64(math.Float64bits(f)) +} + +func (e *msgpackEncDriver) encodeExtPreamble(xtag byte, l int) { + switch { + case l == 1: + e.w.writen2(mpFixExt1, xtag) + case l == 2: + e.w.writen2(mpFixExt2, xtag) + case l == 4: + e.w.writen2(mpFixExt4, xtag) + case l == 8: + e.w.writen2(mpFixExt8, xtag) + case l == 16: + e.w.writen2(mpFixExt16, xtag) + case l < 256: + e.w.writen2(mpExt8, byte(l)) + e.w.writen1(xtag) + case l < 65536: + e.w.writen1(mpExt16) + e.w.writeUint16(uint16(l)) + e.w.writen1(xtag) + default: + e.w.writen1(mpExt32) + e.w.writeUint32(uint32(l)) + e.w.writen1(xtag) + } +} + +func (e *msgpackEncDriver) encodeArrayPreamble(length int) { + e.writeContainerLen(msgpackContainerList, length) +} + +func (e *msgpackEncDriver) encodeMapPreamble(length int) { + e.writeContainerLen(msgpackContainerMap, length) +} + +func (e *msgpackEncDriver) encodeString(c charEncoding, s string) { + if c == c_RAW && e.h.WriteExt { + e.writeContainerLen(msgpackContainerBin, len(s)) + } else { + e.writeContainerLen(msgpackContainerStr, len(s)) + } + if len(s) > 0 { + e.w.writestr(s) + } +} + +func (e *msgpackEncDriver) encodeSymbol(v string) { + e.encodeString(c_UTF8, v) +} + +func (e *msgpackEncDriver) encodeStringBytes(c charEncoding, bs []byte) { + if c == c_RAW && e.h.WriteExt { + e.writeContainerLen(msgpackContainerBin, len(bs)) + } else { + e.writeContainerLen(msgpackContainerStr, len(bs)) + } + if len(bs) > 0 { + e.w.writeb(bs) + } +} + +func (e *msgpackEncDriver) writeContainerLen(ct msgpackContainerType, l int) { + switch { + case ct.hasFixMin && l < ct.fixCutoff: + e.w.writen1(ct.bFixMin | byte(l)) + case ct.has8 && l < 256 && (ct.has8Always || e.h.WriteExt): + e.w.writen2(ct.b8, uint8(l)) + case l < 65536: + e.w.writen1(ct.b16) + e.w.writeUint16(uint16(l)) + default: + e.w.writen1(ct.b32) + e.w.writeUint32(uint32(l)) + } +} + +//--------------------------------------------- + +type msgpackDecDriver struct { + r decReader + h *MsgpackHandle + bd byte + bdRead bool + bdType valueType +} + +func (d *msgpackDecDriver) isBuiltinType(rt uintptr) bool { + //no builtin types. All encodings are based on kinds. Types supported as extensions. + return false +} + +func (d *msgpackDecDriver) decodeBuiltin(rt uintptr, v interface{}) {} + +// Note: This returns either a primitive (int, bool, etc) for non-containers, +// or a containerType, or a specific type denoting nil or extension. +// It is called when a nil interface{} is passed, leaving it up to the DecDriver +// to introspect the stream and decide how best to decode. +// It deciphers the value by looking at the stream first. +func (d *msgpackDecDriver) decodeNaked() (v interface{}, vt valueType, decodeFurther bool) { + d.initReadNext() + bd := d.bd + + switch bd { + case mpNil: + vt = valueTypeNil + d.bdRead = false + case mpFalse: + vt = valueTypeBool + v = false + case mpTrue: + vt = valueTypeBool + v = true + + case mpFloat: + vt = valueTypeFloat + v = float64(math.Float32frombits(d.r.readUint32())) + case mpDouble: + vt = valueTypeFloat + v = math.Float64frombits(d.r.readUint64()) + + case mpUint8: + vt = valueTypeUint + v = uint64(d.r.readn1()) + case mpUint16: + vt = valueTypeUint + v = uint64(d.r.readUint16()) + case mpUint32: + vt = valueTypeUint + v = uint64(d.r.readUint32()) + case mpUint64: + vt = valueTypeUint + v = uint64(d.r.readUint64()) + + case mpInt8: + vt = valueTypeInt + v = int64(int8(d.r.readn1())) + case mpInt16: + vt = valueTypeInt + v = int64(int16(d.r.readUint16())) + case mpInt32: + vt = valueTypeInt + v = int64(int32(d.r.readUint32())) + case mpInt64: + vt = valueTypeInt + v = int64(int64(d.r.readUint64())) + + default: + switch { + case bd >= mpPosFixNumMin && bd <= mpPosFixNumMax: + // positive fixnum (always signed) + vt = valueTypeInt + v = int64(int8(bd)) + case bd >= mpNegFixNumMin && bd <= mpNegFixNumMax: + // negative fixnum + vt = valueTypeInt + v = int64(int8(bd)) + case bd == mpStr8, bd == mpStr16, bd == mpStr32, bd >= mpFixStrMin && bd <= mpFixStrMax: + if d.h.RawToString { + var rvm string + vt = valueTypeString + v = &rvm + } else { + var rvm = []byte{} + vt = valueTypeBytes + v = &rvm + } + decodeFurther = true + case bd == mpBin8, bd == mpBin16, bd == mpBin32: + var rvm = []byte{} + vt = valueTypeBytes + v = &rvm + decodeFurther = true + case bd == mpArray16, bd == mpArray32, bd >= mpFixArrayMin && bd <= mpFixArrayMax: + vt = valueTypeArray + decodeFurther = true + case bd == mpMap16, bd == mpMap32, bd >= mpFixMapMin && bd <= mpFixMapMax: + vt = valueTypeMap + decodeFurther = true + case bd >= mpFixExt1 && bd <= mpFixExt16, bd >= mpExt8 && bd <= mpExt32: + clen := d.readExtLen() + var re RawExt + re.Tag = d.r.readn1() + re.Data = d.r.readn(clen) + v = &re + vt = valueTypeExt + default: + decErr("Nil-Deciphered DecodeValue: %s: hex: %x, dec: %d", msgBadDesc, bd, bd) + } + } + if !decodeFurther { + d.bdRead = false + } + return +} + +// int can be decoded from msgpack type: intXXX or uintXXX +func (d *msgpackDecDriver) decodeInt(bitsize uint8) (i int64) { + switch d.bd { + case mpUint8: + i = int64(uint64(d.r.readn1())) + case mpUint16: + i = int64(uint64(d.r.readUint16())) + case mpUint32: + i = int64(uint64(d.r.readUint32())) + case mpUint64: + i = int64(d.r.readUint64()) + case mpInt8: + i = int64(int8(d.r.readn1())) + case mpInt16: + i = int64(int16(d.r.readUint16())) + case mpInt32: + i = int64(int32(d.r.readUint32())) + case mpInt64: + i = int64(d.r.readUint64()) + default: + switch { + case d.bd >= mpPosFixNumMin && d.bd <= mpPosFixNumMax: + i = int64(int8(d.bd)) + case d.bd >= mpNegFixNumMin && d.bd <= mpNegFixNumMax: + i = int64(int8(d.bd)) + default: + decErr("Unhandled single-byte unsigned integer value: %s: %x", msgBadDesc, d.bd) + } + } + // check overflow (logic adapted from std pkg reflect/value.go OverflowUint() + if bitsize > 0 { + if trunc := (i << (64 - bitsize)) >> (64 - bitsize); i != trunc { + decErr("Overflow int value: %v", i) + } + } + d.bdRead = false + return +} + +// uint can be decoded from msgpack type: intXXX or uintXXX +func (d *msgpackDecDriver) decodeUint(bitsize uint8) (ui uint64) { + switch d.bd { + case mpUint8: + ui = uint64(d.r.readn1()) + case mpUint16: + ui = uint64(d.r.readUint16()) + case mpUint32: + ui = uint64(d.r.readUint32()) + case mpUint64: + ui = d.r.readUint64() + case mpInt8: + if i := int64(int8(d.r.readn1())); i >= 0 { + ui = uint64(i) + } else { + decErr("Assigning negative signed value: %v, to unsigned type", i) + } + case mpInt16: + if i := int64(int16(d.r.readUint16())); i >= 0 { + ui = uint64(i) + } else { + decErr("Assigning negative signed value: %v, to unsigned type", i) + } + case mpInt32: + if i := int64(int32(d.r.readUint32())); i >= 0 { + ui = uint64(i) + } else { + decErr("Assigning negative signed value: %v, to unsigned type", i) + } + case mpInt64: + if i := int64(d.r.readUint64()); i >= 0 { + ui = uint64(i) + } else { + decErr("Assigning negative signed value: %v, to unsigned type", i) + } + default: + switch { + case d.bd >= mpPosFixNumMin && d.bd <= mpPosFixNumMax: + ui = uint64(d.bd) + case d.bd >= mpNegFixNumMin && d.bd <= mpNegFixNumMax: + decErr("Assigning negative signed value: %v, to unsigned type", int(d.bd)) + default: + decErr("Unhandled single-byte unsigned integer value: %s: %x", msgBadDesc, d.bd) + } + } + // check overflow (logic adapted from std pkg reflect/value.go OverflowUint() + if bitsize > 0 { + if trunc := (ui << (64 - bitsize)) >> (64 - bitsize); ui != trunc { + decErr("Overflow uint value: %v", ui) + } + } + d.bdRead = false + return +} + +// float can either be decoded from msgpack type: float, double or intX +func (d *msgpackDecDriver) decodeFloat(chkOverflow32 bool) (f float64) { + switch d.bd { + case mpFloat: + f = float64(math.Float32frombits(d.r.readUint32())) + case mpDouble: + f = math.Float64frombits(d.r.readUint64()) + default: + f = float64(d.decodeInt(0)) + } + checkOverflowFloat32(f, chkOverflow32) + d.bdRead = false + return +} + +// bool can be decoded from bool, fixnum 0 or 1. +func (d *msgpackDecDriver) decodeBool() (b bool) { + switch d.bd { + case mpFalse, 0: + // b = false + case mpTrue, 1: + b = true + default: + decErr("Invalid single-byte value for bool: %s: %x", msgBadDesc, d.bd) + } + d.bdRead = false + return +} + +func (d *msgpackDecDriver) decodeString() (s string) { + clen := d.readContainerLen(msgpackContainerStr) + if clen > 0 { + s = string(d.r.readn(clen)) + } + d.bdRead = false + return +} + +// Callers must check if changed=true (to decide whether to replace the one they have) +func (d *msgpackDecDriver) decodeBytes(bs []byte) (bsOut []byte, changed bool) { + // bytes can be decoded from msgpackContainerStr or msgpackContainerBin + var clen int + switch d.bd { + case mpBin8, mpBin16, mpBin32: + clen = d.readContainerLen(msgpackContainerBin) + default: + clen = d.readContainerLen(msgpackContainerStr) + } + // if clen < 0 { + // changed = true + // panic("length cannot be zero. this cannot be nil.") + // } + if clen > 0 { + // if no contents in stream, don't update the passed byteslice + if len(bs) != clen { + // Return changed=true if length of passed slice diff from length of bytes in stream + if len(bs) > clen { + bs = bs[:clen] + } else { + bs = make([]byte, clen) + } + bsOut = bs + changed = true + } + d.r.readb(bs) + } + d.bdRead = false + return +} + +// Every top-level decode funcs (i.e. decodeValue, decode) must call this first. +func (d *msgpackDecDriver) initReadNext() { + if d.bdRead { + return + } + d.bd = d.r.readn1() + d.bdRead = true + d.bdType = valueTypeUnset +} + +func (d *msgpackDecDriver) currentEncodedType() valueType { + if d.bdType == valueTypeUnset { + bd := d.bd + switch bd { + case mpNil: + d.bdType = valueTypeNil + case mpFalse, mpTrue: + d.bdType = valueTypeBool + case mpFloat, mpDouble: + d.bdType = valueTypeFloat + case mpUint8, mpUint16, mpUint32, mpUint64: + d.bdType = valueTypeUint + case mpInt8, mpInt16, mpInt32, mpInt64: + d.bdType = valueTypeInt + default: + switch { + case bd >= mpPosFixNumMin && bd <= mpPosFixNumMax: + d.bdType = valueTypeInt + case bd >= mpNegFixNumMin && bd <= mpNegFixNumMax: + d.bdType = valueTypeInt + case bd == mpStr8, bd == mpStr16, bd == mpStr32, bd >= mpFixStrMin && bd <= mpFixStrMax: + if d.h.RawToString { + d.bdType = valueTypeString + } else { + d.bdType = valueTypeBytes + } + case bd == mpBin8, bd == mpBin16, bd == mpBin32: + d.bdType = valueTypeBytes + case bd == mpArray16, bd == mpArray32, bd >= mpFixArrayMin && bd <= mpFixArrayMax: + d.bdType = valueTypeArray + case bd == mpMap16, bd == mpMap32, bd >= mpFixMapMin && bd <= mpFixMapMax: + d.bdType = valueTypeMap + case bd >= mpFixExt1 && bd <= mpFixExt16, bd >= mpExt8 && bd <= mpExt32: + d.bdType = valueTypeExt + default: + decErr("currentEncodedType: Undeciphered descriptor: %s: hex: %x, dec: %d", msgBadDesc, bd, bd) + } + } + } + return d.bdType +} + +func (d *msgpackDecDriver) tryDecodeAsNil() bool { + if d.bd == mpNil { + d.bdRead = false + return true + } + return false +} + +func (d *msgpackDecDriver) readContainerLen(ct msgpackContainerType) (clen int) { + bd := d.bd + switch { + case bd == mpNil: + clen = -1 // to represent nil + case bd == ct.b8: + clen = int(d.r.readn1()) + case bd == ct.b16: + clen = int(d.r.readUint16()) + case bd == ct.b32: + clen = int(d.r.readUint32()) + case (ct.bFixMin & bd) == ct.bFixMin: + clen = int(ct.bFixMin ^ bd) + default: + decErr("readContainerLen: %s: hex: %x, dec: %d", msgBadDesc, bd, bd) + } + d.bdRead = false + return +} + +func (d *msgpackDecDriver) readMapLen() int { + return d.readContainerLen(msgpackContainerMap) +} + +func (d *msgpackDecDriver) readArrayLen() int { + return d.readContainerLen(msgpackContainerList) +} + +func (d *msgpackDecDriver) readExtLen() (clen int) { + switch d.bd { + case mpNil: + clen = -1 // to represent nil + case mpFixExt1: + clen = 1 + case mpFixExt2: + clen = 2 + case mpFixExt4: + clen = 4 + case mpFixExt8: + clen = 8 + case mpFixExt16: + clen = 16 + case mpExt8: + clen = int(d.r.readn1()) + case mpExt16: + clen = int(d.r.readUint16()) + case mpExt32: + clen = int(d.r.readUint32()) + default: + decErr("decoding ext bytes: found unexpected byte: %x", d.bd) + } + return +} + +func (d *msgpackDecDriver) decodeExt(verifyTag bool, tag byte) (xtag byte, xbs []byte) { + xbd := d.bd + switch { + case xbd == mpBin8, xbd == mpBin16, xbd == mpBin32: + xbs, _ = d.decodeBytes(nil) + case xbd == mpStr8, xbd == mpStr16, xbd == mpStr32, + xbd >= mpFixStrMin && xbd <= mpFixStrMax: + xbs = []byte(d.decodeString()) + default: + clen := d.readExtLen() + xtag = d.r.readn1() + if verifyTag && xtag != tag { + decErr("Wrong extension tag. Got %b. Expecting: %v", xtag, tag) + } + xbs = d.r.readn(clen) + } + d.bdRead = false + return +} + +//-------------------------------------------------- + +//MsgpackHandle is a Handle for the Msgpack Schema-Free Encoding Format. +type MsgpackHandle struct { + BasicHandle + + // RawToString controls how raw bytes are decoded into a nil interface{}. + RawToString bool + // WriteExt flag supports encoding configured extensions with extension tags. + // It also controls whether other elements of the new spec are encoded (ie Str8). + // + // With WriteExt=false, configured extensions are serialized as raw bytes + // and Str8 is not encoded. + // + // A stream can still be decoded into a typed value, provided an appropriate value + // is provided, but the type cannot be inferred from the stream. If no appropriate + // type is provided (e.g. decoding into a nil interface{}), you get back + // a []byte or string based on the setting of RawToString. + WriteExt bool +} + +func (h *MsgpackHandle) newEncDriver(w encWriter) encDriver { + return &msgpackEncDriver{w: w, h: h} +} + +func (h *MsgpackHandle) newDecDriver(r decReader) decDriver { + return &msgpackDecDriver{r: r, h: h} +} + +func (h *MsgpackHandle) writeExt() bool { + return h.WriteExt +} + +func (h *MsgpackHandle) getBasicHandle() *BasicHandle { + return &h.BasicHandle +} + +//-------------------------------------------------- + +type msgpackSpecRpcCodec struct { + rpcCodec +} + +// /////////////// Spec RPC Codec /////////////////// +func (c *msgpackSpecRpcCodec) WriteRequest(r *rpc.Request, body interface{}) error { + // WriteRequest can write to both a Go service, and other services that do + // not abide by the 1 argument rule of a Go service. + // We discriminate based on if the body is a MsgpackSpecRpcMultiArgs + var bodyArr []interface{} + if m, ok := body.(MsgpackSpecRpcMultiArgs); ok { + bodyArr = ([]interface{})(m) + } else { + bodyArr = []interface{}{body} + } + r2 := []interface{}{0, uint32(r.Seq), r.ServiceMethod, bodyArr} + return c.write(r2, nil, false, true) +} + +func (c *msgpackSpecRpcCodec) WriteResponse(r *rpc.Response, body interface{}) error { + var moe interface{} + if r.Error != "" { + moe = r.Error + } + if moe != nil && body != nil { + body = nil + } + r2 := []interface{}{1, uint32(r.Seq), moe, body} + return c.write(r2, nil, false, true) +} + +func (c *msgpackSpecRpcCodec) ReadResponseHeader(r *rpc.Response) error { + return c.parseCustomHeader(1, &r.Seq, &r.Error) +} + +func (c *msgpackSpecRpcCodec) ReadRequestHeader(r *rpc.Request) error { + return c.parseCustomHeader(0, &r.Seq, &r.ServiceMethod) +} + +func (c *msgpackSpecRpcCodec) ReadRequestBody(body interface{}) error { + if body == nil { // read and discard + return c.read(nil) + } + bodyArr := []interface{}{body} + return c.read(&bodyArr) +} + +func (c *msgpackSpecRpcCodec) parseCustomHeader(expectTypeByte byte, msgid *uint64, methodOrError *string) (err error) { + + if c.cls { + return io.EOF + } + + // We read the response header by hand + // so that the body can be decoded on its own from the stream at a later time. + + const fia byte = 0x94 //four item array descriptor value + // Not sure why the panic of EOF is swallowed above. + // if bs1 := c.dec.r.readn1(); bs1 != fia { + // err = fmt.Errorf("Unexpected value for array descriptor: Expecting %v. Received %v", fia, bs1) + // return + // } + var b byte + b, err = c.br.ReadByte() + if err != nil { + return + } + if b != fia { + err = fmt.Errorf("Unexpected value for array descriptor: Expecting %v. Received %v", fia, b) + return + } + + if err = c.read(&b); err != nil { + return + } + if b != expectTypeByte { + err = fmt.Errorf("Unexpected byte descriptor in header. Expecting %v. Received %v", expectTypeByte, b) + return + } + if err = c.read(msgid); err != nil { + return + } + if err = c.read(methodOrError); err != nil { + return + } + return +} + +//-------------------------------------------------- + +// msgpackSpecRpc is the implementation of Rpc that uses custom communication protocol +// as defined in the msgpack spec at https://github.com/msgpack-rpc/msgpack-rpc/blob/master/spec.md +type msgpackSpecRpc struct{} + +// MsgpackSpecRpc implements Rpc using the communication protocol defined in +// the msgpack spec at https://github.com/msgpack-rpc/msgpack-rpc/blob/master/spec.md . +// Its methods (ServerCodec and ClientCodec) return values that implement RpcCodecBuffered. +var MsgpackSpecRpc msgpackSpecRpc + +func (x msgpackSpecRpc) ServerCodec(conn io.ReadWriteCloser, h Handle) rpc.ServerCodec { + return &msgpackSpecRpcCodec{newRPCCodec(conn, h)} +} + +func (x msgpackSpecRpc) ClientCodec(conn io.ReadWriteCloser, h Handle) rpc.ClientCodec { + return &msgpackSpecRpcCodec{newRPCCodec(conn, h)} +} + +var _ decDriver = (*msgpackDecDriver)(nil) +var _ encDriver = (*msgpackEncDriver)(nil) diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/msgpack_test.py b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/msgpack_test.py new file mode 100644 index 0000000000000..e933838c56a89 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/msgpack_test.py @@ -0,0 +1,110 @@ +#!/usr/bin/env python + +# This will create golden files in a directory passed to it. +# A Test calls this internally to create the golden files +# So it can process them (so we don't have to checkin the files). + +import msgpack, msgpackrpc, sys, os, threading + +def get_test_data_list(): + # get list with all primitive types, and a combo type + l0 = [ + -8, + -1616, + -32323232, + -6464646464646464, + 192, + 1616, + 32323232, + 6464646464646464, + 192, + -3232.0, + -6464646464.0, + 3232.0, + 6464646464.0, + False, + True, + None, + "someday", + "", + "bytestring", + 1328176922000002000, + -2206187877999998000, + 0, + -6795364578871345152 + ] + l1 = [ + { "true": True, + "false": False }, + { "true": "True", + "false": False, + "uint16(1616)": 1616 }, + { "list": [1616, 32323232, True, -3232.0, {"TRUE":True, "FALSE":False}, [True, False] ], + "int32":32323232, "bool": True, + "LONG STRING": "123456789012345678901234567890123456789012345678901234567890", + "SHORT STRING": "1234567890" }, + { True: "true", 8: False, "false": 0 } + ] + + l = [] + l.extend(l0) + l.append(l0) + l.extend(l1) + return l + +def build_test_data(destdir): + l = get_test_data_list() + for i in range(len(l)): + packer = msgpack.Packer() + serialized = packer.pack(l[i]) + f = open(os.path.join(destdir, str(i) + '.golden'), 'wb') + f.write(serialized) + f.close() + +def doRpcServer(port, stopTimeSec): + class EchoHandler(object): + def Echo123(self, msg1, msg2, msg3): + return ("1:%s 2:%s 3:%s" % (msg1, msg2, msg3)) + def EchoStruct(self, msg): + return ("%s" % msg) + + addr = msgpackrpc.Address('localhost', port) + server = msgpackrpc.Server(EchoHandler()) + server.listen(addr) + # run thread to stop it after stopTimeSec seconds if > 0 + if stopTimeSec > 0: + def myStopRpcServer(): + server.stop() + t = threading.Timer(stopTimeSec, myStopRpcServer) + t.start() + server.start() + +def doRpcClientToPythonSvc(port): + address = msgpackrpc.Address('localhost', port) + client = msgpackrpc.Client(address, unpack_encoding='utf-8') + print client.call("Echo123", "A1", "B2", "C3") + print client.call("EchoStruct", {"A" :"Aa", "B":"Bb", "C":"Cc"}) + +def doRpcClientToGoSvc(port): + # print ">>>> port: ", port, " <<<<<" + address = msgpackrpc.Address('localhost', port) + client = msgpackrpc.Client(address, unpack_encoding='utf-8') + print client.call("TestRpcInt.Echo123", ["A1", "B2", "C3"]) + print client.call("TestRpcInt.EchoStruct", {"A" :"Aa", "B":"Bb", "C":"Cc"}) + +def doMain(args): + if len(args) == 2 and args[0] == "testdata": + build_test_data(args[1]) + elif len(args) == 3 and args[0] == "rpc-server": + doRpcServer(int(args[1]), int(args[2])) + elif len(args) == 2 and args[0] == "rpc-client-python-service": + doRpcClientToPythonSvc(int(args[1])) + elif len(args) == 2 and args[0] == "rpc-client-go-service": + doRpcClientToGoSvc(int(args[1])) + else: + print("Usage: msgpack_test.py " + + "[testdata|rpc-server|rpc-client-python-service|rpc-client-go-service] ...") + +if __name__ == "__main__": + doMain(sys.argv[1:]) + diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/rpc.go b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/rpc.go new file mode 100644 index 0000000000000..d014dbdcc7d0a --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/rpc.go @@ -0,0 +1,152 @@ +// Copyright (c) 2012, 2013 Ugorji Nwoke. All rights reserved. +// Use of this source code is governed by a BSD-style license found in the LICENSE file. + +package codec + +import ( + "bufio" + "io" + "net/rpc" + "sync" +) + +// Rpc provides a rpc Server or Client Codec for rpc communication. +type Rpc interface { + ServerCodec(conn io.ReadWriteCloser, h Handle) rpc.ServerCodec + ClientCodec(conn io.ReadWriteCloser, h Handle) rpc.ClientCodec +} + +// RpcCodecBuffered allows access to the underlying bufio.Reader/Writer +// used by the rpc connection. It accomodates use-cases where the connection +// should be used by rpc and non-rpc functions, e.g. streaming a file after +// sending an rpc response. +type RpcCodecBuffered interface { + BufferedReader() *bufio.Reader + BufferedWriter() *bufio.Writer +} + +// ------------------------------------- + +// rpcCodec defines the struct members and common methods. +type rpcCodec struct { + rwc io.ReadWriteCloser + dec *Decoder + enc *Encoder + bw *bufio.Writer + br *bufio.Reader + mu sync.Mutex + cls bool +} + +func newRPCCodec(conn io.ReadWriteCloser, h Handle) rpcCodec { + bw := bufio.NewWriter(conn) + br := bufio.NewReader(conn) + return rpcCodec{ + rwc: conn, + bw: bw, + br: br, + enc: NewEncoder(bw, h), + dec: NewDecoder(br, h), + } +} + +func (c *rpcCodec) BufferedReader() *bufio.Reader { + return c.br +} + +func (c *rpcCodec) BufferedWriter() *bufio.Writer { + return c.bw +} + +func (c *rpcCodec) write(obj1, obj2 interface{}, writeObj2, doFlush bool) (err error) { + if c.cls { + return io.EOF + } + if err = c.enc.Encode(obj1); err != nil { + return + } + if writeObj2 { + if err = c.enc.Encode(obj2); err != nil { + return + } + } + if doFlush && c.bw != nil { + return c.bw.Flush() + } + return +} + +func (c *rpcCodec) read(obj interface{}) (err error) { + if c.cls { + return io.EOF + } + //If nil is passed in, we should still attempt to read content to nowhere. + if obj == nil { + var obj2 interface{} + return c.dec.Decode(&obj2) + } + return c.dec.Decode(obj) +} + +func (c *rpcCodec) Close() error { + if c.cls { + return io.EOF + } + c.cls = true + return c.rwc.Close() +} + +func (c *rpcCodec) ReadResponseBody(body interface{}) error { + return c.read(body) +} + +// ------------------------------------- + +type goRpcCodec struct { + rpcCodec +} + +func (c *goRpcCodec) WriteRequest(r *rpc.Request, body interface{}) error { + // Must protect for concurrent access as per API + c.mu.Lock() + defer c.mu.Unlock() + return c.write(r, body, true, true) +} + +func (c *goRpcCodec) WriteResponse(r *rpc.Response, body interface{}) error { + c.mu.Lock() + defer c.mu.Unlock() + return c.write(r, body, true, true) +} + +func (c *goRpcCodec) ReadResponseHeader(r *rpc.Response) error { + return c.read(r) +} + +func (c *goRpcCodec) ReadRequestHeader(r *rpc.Request) error { + return c.read(r) +} + +func (c *goRpcCodec) ReadRequestBody(body interface{}) error { + return c.read(body) +} + +// ------------------------------------- + +// goRpc is the implementation of Rpc that uses the communication protocol +// as defined in net/rpc package. +type goRpc struct{} + +// GoRpc implements Rpc using the communication protocol defined in net/rpc package. +// Its methods (ServerCodec and ClientCodec) return values that implement RpcCodecBuffered. +var GoRpc goRpc + +func (x goRpc) ServerCodec(conn io.ReadWriteCloser, h Handle) rpc.ServerCodec { + return &goRpcCodec{newRPCCodec(conn, h)} +} + +func (x goRpc) ClientCodec(conn io.ReadWriteCloser, h Handle) rpc.ClientCodec { + return &goRpcCodec{newRPCCodec(conn, h)} +} + +var _ RpcCodecBuffered = (*rpcCodec)(nil) // ensure *rpcCodec implements RpcCodecBuffered diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/simple.go b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/simple.go new file mode 100644 index 0000000000000..9e4d148a2a179 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/simple.go @@ -0,0 +1,461 @@ +// Copyright (c) 2012, 2013 Ugorji Nwoke. All rights reserved. +// Use of this source code is governed by a BSD-style license found in the LICENSE file. + +package codec + +import "math" + +const ( + _ uint8 = iota + simpleVdNil = 1 + simpleVdFalse = 2 + simpleVdTrue = 3 + simpleVdFloat32 = 4 + simpleVdFloat64 = 5 + + // each lasts for 4 (ie n, n+1, n+2, n+3) + simpleVdPosInt = 8 + simpleVdNegInt = 12 + + // containers: each lasts for 4 (ie n, n+1, n+2, ... n+7) + simpleVdString = 216 + simpleVdByteArray = 224 + simpleVdArray = 232 + simpleVdMap = 240 + simpleVdExt = 248 +) + +type simpleEncDriver struct { + h *SimpleHandle + w encWriter + //b [8]byte +} + +func (e *simpleEncDriver) isBuiltinType(rt uintptr) bool { + return false +} + +func (e *simpleEncDriver) encodeBuiltin(rt uintptr, v interface{}) { +} + +func (e *simpleEncDriver) encodeNil() { + e.w.writen1(simpleVdNil) +} + +func (e *simpleEncDriver) encodeBool(b bool) { + if b { + e.w.writen1(simpleVdTrue) + } else { + e.w.writen1(simpleVdFalse) + } +} + +func (e *simpleEncDriver) encodeFloat32(f float32) { + e.w.writen1(simpleVdFloat32) + e.w.writeUint32(math.Float32bits(f)) +} + +func (e *simpleEncDriver) encodeFloat64(f float64) { + e.w.writen1(simpleVdFloat64) + e.w.writeUint64(math.Float64bits(f)) +} + +func (e *simpleEncDriver) encodeInt(v int64) { + if v < 0 { + e.encUint(uint64(-v), simpleVdNegInt) + } else { + e.encUint(uint64(v), simpleVdPosInt) + } +} + +func (e *simpleEncDriver) encodeUint(v uint64) { + e.encUint(v, simpleVdPosInt) +} + +func (e *simpleEncDriver) encUint(v uint64, bd uint8) { + switch { + case v <= math.MaxUint8: + e.w.writen2(bd, uint8(v)) + case v <= math.MaxUint16: + e.w.writen1(bd + 1) + e.w.writeUint16(uint16(v)) + case v <= math.MaxUint32: + e.w.writen1(bd + 2) + e.w.writeUint32(uint32(v)) + case v <= math.MaxUint64: + e.w.writen1(bd + 3) + e.w.writeUint64(v) + } +} + +func (e *simpleEncDriver) encLen(bd byte, length int) { + switch { + case length == 0: + e.w.writen1(bd) + case length <= math.MaxUint8: + e.w.writen1(bd + 1) + e.w.writen1(uint8(length)) + case length <= math.MaxUint16: + e.w.writen1(bd + 2) + e.w.writeUint16(uint16(length)) + case int64(length) <= math.MaxUint32: + e.w.writen1(bd + 3) + e.w.writeUint32(uint32(length)) + default: + e.w.writen1(bd + 4) + e.w.writeUint64(uint64(length)) + } +} + +func (e *simpleEncDriver) encodeExtPreamble(xtag byte, length int) { + e.encLen(simpleVdExt, length) + e.w.writen1(xtag) +} + +func (e *simpleEncDriver) encodeArrayPreamble(length int) { + e.encLen(simpleVdArray, length) +} + +func (e *simpleEncDriver) encodeMapPreamble(length int) { + e.encLen(simpleVdMap, length) +} + +func (e *simpleEncDriver) encodeString(c charEncoding, v string) { + e.encLen(simpleVdString, len(v)) + e.w.writestr(v) +} + +func (e *simpleEncDriver) encodeSymbol(v string) { + e.encodeString(c_UTF8, v) +} + +func (e *simpleEncDriver) encodeStringBytes(c charEncoding, v []byte) { + e.encLen(simpleVdByteArray, len(v)) + e.w.writeb(v) +} + +//------------------------------------ + +type simpleDecDriver struct { + h *SimpleHandle + r decReader + bdRead bool + bdType valueType + bd byte + //b [8]byte +} + +func (d *simpleDecDriver) initReadNext() { + if d.bdRead { + return + } + d.bd = d.r.readn1() + d.bdRead = true + d.bdType = valueTypeUnset +} + +func (d *simpleDecDriver) currentEncodedType() valueType { + if d.bdType == valueTypeUnset { + switch d.bd { + case simpleVdNil: + d.bdType = valueTypeNil + case simpleVdTrue, simpleVdFalse: + d.bdType = valueTypeBool + case simpleVdPosInt, simpleVdPosInt + 1, simpleVdPosInt + 2, simpleVdPosInt + 3: + d.bdType = valueTypeUint + case simpleVdNegInt, simpleVdNegInt + 1, simpleVdNegInt + 2, simpleVdNegInt + 3: + d.bdType = valueTypeInt + case simpleVdFloat32, simpleVdFloat64: + d.bdType = valueTypeFloat + case simpleVdString, simpleVdString + 1, simpleVdString + 2, simpleVdString + 3, simpleVdString + 4: + d.bdType = valueTypeString + case simpleVdByteArray, simpleVdByteArray + 1, simpleVdByteArray + 2, simpleVdByteArray + 3, simpleVdByteArray + 4: + d.bdType = valueTypeBytes + case simpleVdExt, simpleVdExt + 1, simpleVdExt + 2, simpleVdExt + 3, simpleVdExt + 4: + d.bdType = valueTypeExt + case simpleVdArray, simpleVdArray + 1, simpleVdArray + 2, simpleVdArray + 3, simpleVdArray + 4: + d.bdType = valueTypeArray + case simpleVdMap, simpleVdMap + 1, simpleVdMap + 2, simpleVdMap + 3, simpleVdMap + 4: + d.bdType = valueTypeMap + default: + decErr("currentEncodedType: Unrecognized d.vd: 0x%x", d.bd) + } + } + return d.bdType +} + +func (d *simpleDecDriver) tryDecodeAsNil() bool { + if d.bd == simpleVdNil { + d.bdRead = false + return true + } + return false +} + +func (d *simpleDecDriver) isBuiltinType(rt uintptr) bool { + return false +} + +func (d *simpleDecDriver) decodeBuiltin(rt uintptr, v interface{}) { +} + +func (d *simpleDecDriver) decIntAny() (ui uint64, i int64, neg bool) { + switch d.bd { + case simpleVdPosInt: + ui = uint64(d.r.readn1()) + i = int64(ui) + case simpleVdPosInt + 1: + ui = uint64(d.r.readUint16()) + i = int64(ui) + case simpleVdPosInt + 2: + ui = uint64(d.r.readUint32()) + i = int64(ui) + case simpleVdPosInt + 3: + ui = uint64(d.r.readUint64()) + i = int64(ui) + case simpleVdNegInt: + ui = uint64(d.r.readn1()) + i = -(int64(ui)) + neg = true + case simpleVdNegInt + 1: + ui = uint64(d.r.readUint16()) + i = -(int64(ui)) + neg = true + case simpleVdNegInt + 2: + ui = uint64(d.r.readUint32()) + i = -(int64(ui)) + neg = true + case simpleVdNegInt + 3: + ui = uint64(d.r.readUint64()) + i = -(int64(ui)) + neg = true + default: + decErr("decIntAny: Integer only valid from pos/neg integer1..8. Invalid descriptor: %v", d.bd) + } + // don't do this check, because callers may only want the unsigned value. + // if ui > math.MaxInt64 { + // decErr("decIntAny: Integer out of range for signed int64: %v", ui) + // } + return +} + +func (d *simpleDecDriver) decodeInt(bitsize uint8) (i int64) { + _, i, _ = d.decIntAny() + checkOverflow(0, i, bitsize) + d.bdRead = false + return +} + +func (d *simpleDecDriver) decodeUint(bitsize uint8) (ui uint64) { + ui, i, neg := d.decIntAny() + if neg { + decErr("Assigning negative signed value: %v, to unsigned type", i) + } + checkOverflow(ui, 0, bitsize) + d.bdRead = false + return +} + +func (d *simpleDecDriver) decodeFloat(chkOverflow32 bool) (f float64) { + switch d.bd { + case simpleVdFloat32: + f = float64(math.Float32frombits(d.r.readUint32())) + case simpleVdFloat64: + f = math.Float64frombits(d.r.readUint64()) + default: + if d.bd >= simpleVdPosInt && d.bd <= simpleVdNegInt+3 { + _, i, _ := d.decIntAny() + f = float64(i) + } else { + decErr("Float only valid from float32/64: Invalid descriptor: %v", d.bd) + } + } + checkOverflowFloat32(f, chkOverflow32) + d.bdRead = false + return +} + +// bool can be decoded from bool only (single byte). +func (d *simpleDecDriver) decodeBool() (b bool) { + switch d.bd { + case simpleVdTrue: + b = true + case simpleVdFalse: + default: + decErr("Invalid single-byte value for bool: %s: %x", msgBadDesc, d.bd) + } + d.bdRead = false + return +} + +func (d *simpleDecDriver) readMapLen() (length int) { + d.bdRead = false + return d.decLen() +} + +func (d *simpleDecDriver) readArrayLen() (length int) { + d.bdRead = false + return d.decLen() +} + +func (d *simpleDecDriver) decLen() int { + switch d.bd % 8 { + case 0: + return 0 + case 1: + return int(d.r.readn1()) + case 2: + return int(d.r.readUint16()) + case 3: + ui := uint64(d.r.readUint32()) + checkOverflow(ui, 0, intBitsize) + return int(ui) + case 4: + ui := d.r.readUint64() + checkOverflow(ui, 0, intBitsize) + return int(ui) + } + decErr("decLen: Cannot read length: bd%8 must be in range 0..4. Got: %d", d.bd%8) + return -1 +} + +func (d *simpleDecDriver) decodeString() (s string) { + s = string(d.r.readn(d.decLen())) + d.bdRead = false + return +} + +func (d *simpleDecDriver) decodeBytes(bs []byte) (bsOut []byte, changed bool) { + if clen := d.decLen(); clen > 0 { + // if no contents in stream, don't update the passed byteslice + if len(bs) != clen { + if len(bs) > clen { + bs = bs[:clen] + } else { + bs = make([]byte, clen) + } + bsOut = bs + changed = true + } + d.r.readb(bs) + } + d.bdRead = false + return +} + +func (d *simpleDecDriver) decodeExt(verifyTag bool, tag byte) (xtag byte, xbs []byte) { + switch d.bd { + case simpleVdExt, simpleVdExt + 1, simpleVdExt + 2, simpleVdExt + 3, simpleVdExt + 4: + l := d.decLen() + xtag = d.r.readn1() + if verifyTag && xtag != tag { + decErr("Wrong extension tag. Got %b. Expecting: %v", xtag, tag) + } + xbs = d.r.readn(l) + case simpleVdByteArray, simpleVdByteArray + 1, simpleVdByteArray + 2, simpleVdByteArray + 3, simpleVdByteArray + 4: + xbs, _ = d.decodeBytes(nil) + default: + decErr("Invalid d.vd for extensions (Expecting extensions or byte array). Got: 0x%x", d.bd) + } + d.bdRead = false + return +} + +func (d *simpleDecDriver) decodeNaked() (v interface{}, vt valueType, decodeFurther bool) { + d.initReadNext() + + switch d.bd { + case simpleVdNil: + vt = valueTypeNil + case simpleVdFalse: + vt = valueTypeBool + v = false + case simpleVdTrue: + vt = valueTypeBool + v = true + case simpleVdPosInt, simpleVdPosInt + 1, simpleVdPosInt + 2, simpleVdPosInt + 3: + vt = valueTypeUint + ui, _, _ := d.decIntAny() + v = ui + case simpleVdNegInt, simpleVdNegInt + 1, simpleVdNegInt + 2, simpleVdNegInt + 3: + vt = valueTypeInt + _, i, _ := d.decIntAny() + v = i + case simpleVdFloat32: + vt = valueTypeFloat + v = d.decodeFloat(true) + case simpleVdFloat64: + vt = valueTypeFloat + v = d.decodeFloat(false) + case simpleVdString, simpleVdString + 1, simpleVdString + 2, simpleVdString + 3, simpleVdString + 4: + vt = valueTypeString + v = d.decodeString() + case simpleVdByteArray, simpleVdByteArray + 1, simpleVdByteArray + 2, simpleVdByteArray + 3, simpleVdByteArray + 4: + vt = valueTypeBytes + v, _ = d.decodeBytes(nil) + case simpleVdExt, simpleVdExt + 1, simpleVdExt + 2, simpleVdExt + 3, simpleVdExt + 4: + vt = valueTypeExt + l := d.decLen() + var re RawExt + re.Tag = d.r.readn1() + re.Data = d.r.readn(l) + v = &re + vt = valueTypeExt + case simpleVdArray, simpleVdArray + 1, simpleVdArray + 2, simpleVdArray + 3, simpleVdArray + 4: + vt = valueTypeArray + decodeFurther = true + case simpleVdMap, simpleVdMap + 1, simpleVdMap + 2, simpleVdMap + 3, simpleVdMap + 4: + vt = valueTypeMap + decodeFurther = true + default: + decErr("decodeNaked: Unrecognized d.vd: 0x%x", d.bd) + } + + if !decodeFurther { + d.bdRead = false + } + return +} + +//------------------------------------ + +// SimpleHandle is a Handle for a very simple encoding format. +// +// simple is a simplistic codec similar to binc, but not as compact. +// - Encoding of a value is always preceeded by the descriptor byte (bd) +// - True, false, nil are encoded fully in 1 byte (the descriptor) +// - Integers (intXXX, uintXXX) are encoded in 1, 2, 4 or 8 bytes (plus a descriptor byte). +// There are positive (uintXXX and intXXX >= 0) and negative (intXXX < 0) integers. +// - Floats are encoded in 4 or 8 bytes (plus a descriptor byte) +// - Lenght of containers (strings, bytes, array, map, extensions) +// are encoded in 0, 1, 2, 4 or 8 bytes. +// Zero-length containers have no length encoded. +// For others, the number of bytes is given by pow(2, bd%3) +// - maps are encoded as [bd] [length] [[key][value]]... +// - arrays are encoded as [bd] [length] [value]... +// - extensions are encoded as [bd] [length] [tag] [byte]... +// - strings/bytearrays are encoded as [bd] [length] [byte]... +// +// The full spec will be published soon. +type SimpleHandle struct { + BasicHandle +} + +func (h *SimpleHandle) newEncDriver(w encWriter) encDriver { + return &simpleEncDriver{w: w, h: h} +} + +func (h *SimpleHandle) newDecDriver(r decReader) decDriver { + return &simpleDecDriver{r: r, h: h} +} + +func (_ *SimpleHandle) writeExt() bool { + return true +} + +func (h *SimpleHandle) getBasicHandle() *BasicHandle { + return &h.BasicHandle +} + +var _ decDriver = (*simpleDecDriver)(nil) +var _ encDriver = (*simpleEncDriver)(nil) diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/time.go b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/time.go new file mode 100644 index 0000000000000..c86d65328d76a --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/time.go @@ -0,0 +1,193 @@ +// Copyright (c) 2012, 2013 Ugorji Nwoke. All rights reserved. +// Use of this source code is governed by a BSD-style license found in the LICENSE file. + +package codec + +import ( + "time" +) + +var ( + timeDigits = [...]byte{'0', '1', '2', '3', '4', '5', '6', '7', '8', '9'} +) + +// EncodeTime encodes a time.Time as a []byte, including +// information on the instant in time and UTC offset. +// +// Format Description +// +// A timestamp is composed of 3 components: +// +// - secs: signed integer representing seconds since unix epoch +// - nsces: unsigned integer representing fractional seconds as a +// nanosecond offset within secs, in the range 0 <= nsecs < 1e9 +// - tz: signed integer representing timezone offset in minutes east of UTC, +// and a dst (daylight savings time) flag +// +// When encoding a timestamp, the first byte is the descriptor, which +// defines which components are encoded and how many bytes are used to +// encode secs and nsecs components. *If secs/nsecs is 0 or tz is UTC, it +// is not encoded in the byte array explicitly*. +// +// Descriptor 8 bits are of the form `A B C DDD EE`: +// A: Is secs component encoded? 1 = true +// B: Is nsecs component encoded? 1 = true +// C: Is tz component encoded? 1 = true +// DDD: Number of extra bytes for secs (range 0-7). +// If A = 1, secs encoded in DDD+1 bytes. +// If A = 0, secs is not encoded, and is assumed to be 0. +// If A = 1, then we need at least 1 byte to encode secs. +// DDD says the number of extra bytes beyond that 1. +// E.g. if DDD=0, then secs is represented in 1 byte. +// if DDD=2, then secs is represented in 3 bytes. +// EE: Number of extra bytes for nsecs (range 0-3). +// If B = 1, nsecs encoded in EE+1 bytes (similar to secs/DDD above) +// +// Following the descriptor bytes, subsequent bytes are: +// +// secs component encoded in `DDD + 1` bytes (if A == 1) +// nsecs component encoded in `EE + 1` bytes (if B == 1) +// tz component encoded in 2 bytes (if C == 1) +// +// secs and nsecs components are integers encoded in a BigEndian +// 2-complement encoding format. +// +// tz component is encoded as 2 bytes (16 bits). Most significant bit 15 to +// Least significant bit 0 are described below: +// +// Timezone offset has a range of -12:00 to +14:00 (ie -720 to +840 minutes). +// Bit 15 = have\_dst: set to 1 if we set the dst flag. +// Bit 14 = dst\_on: set to 1 if dst is in effect at the time, or 0 if not. +// Bits 13..0 = timezone offset in minutes. It is a signed integer in Big Endian format. +// +func encodeTime(t time.Time) []byte { + //t := rv.Interface().(time.Time) + tsecs, tnsecs := t.Unix(), t.Nanosecond() + var ( + bd byte + btmp [8]byte + bs [16]byte + i int = 1 + ) + l := t.Location() + if l == time.UTC { + l = nil + } + if tsecs != 0 { + bd = bd | 0x80 + bigen.PutUint64(btmp[:], uint64(tsecs)) + f := pruneSignExt(btmp[:], tsecs >= 0) + bd = bd | (byte(7-f) << 2) + copy(bs[i:], btmp[f:]) + i = i + (8 - f) + } + if tnsecs != 0 { + bd = bd | 0x40 + bigen.PutUint32(btmp[:4], uint32(tnsecs)) + f := pruneSignExt(btmp[:4], true) + bd = bd | byte(3-f) + copy(bs[i:], btmp[f:4]) + i = i + (4 - f) + } + if l != nil { + bd = bd | 0x20 + // Note that Go Libs do not give access to dst flag. + _, zoneOffset := t.Zone() + //zoneName, zoneOffset := t.Zone() + zoneOffset /= 60 + z := uint16(zoneOffset) + bigen.PutUint16(btmp[:2], z) + // clear dst flags + bs[i] = btmp[0] & 0x3f + bs[i+1] = btmp[1] + i = i + 2 + } + bs[0] = bd + return bs[0:i] +} + +// DecodeTime decodes a []byte into a time.Time. +func decodeTime(bs []byte) (tt time.Time, err error) { + bd := bs[0] + var ( + tsec int64 + tnsec uint32 + tz uint16 + i byte = 1 + i2 byte + n byte + ) + if bd&(1<<7) != 0 { + var btmp [8]byte + n = ((bd >> 2) & 0x7) + 1 + i2 = i + n + copy(btmp[8-n:], bs[i:i2]) + //if first bit of bs[i] is set, then fill btmp[0..8-n] with 0xff (ie sign extend it) + if bs[i]&(1<<7) != 0 { + copy(btmp[0:8-n], bsAll0xff) + //for j,k := byte(0), 8-n; j < k; j++ { btmp[j] = 0xff } + } + i = i2 + tsec = int64(bigen.Uint64(btmp[:])) + } + if bd&(1<<6) != 0 { + var btmp [4]byte + n = (bd & 0x3) + 1 + i2 = i + n + copy(btmp[4-n:], bs[i:i2]) + i = i2 + tnsec = bigen.Uint32(btmp[:]) + } + if bd&(1<<5) == 0 { + tt = time.Unix(tsec, int64(tnsec)).UTC() + return + } + // In stdlib time.Parse, when a date is parsed without a zone name, it uses "" as zone name. + // However, we need name here, so it can be shown when time is printed. + // Zone name is in form: UTC-08:00. + // Note that Go Libs do not give access to dst flag, so we ignore dst bits + + i2 = i + 2 + tz = bigen.Uint16(bs[i:i2]) + i = i2 + // sign extend sign bit into top 2 MSB (which were dst bits): + if tz&(1<<13) == 0 { // positive + tz = tz & 0x3fff //clear 2 MSBs: dst bits + } else { // negative + tz = tz | 0xc000 //set 2 MSBs: dst bits + //tzname[3] = '-' (TODO: verify. this works here) + } + tzint := int16(tz) + if tzint == 0 { + tt = time.Unix(tsec, int64(tnsec)).UTC() + } else { + // For Go Time, do not use a descriptive timezone. + // It's unnecessary, and makes it harder to do a reflect.DeepEqual. + // The Offset already tells what the offset should be, if not on UTC and unknown zone name. + // var zoneName = timeLocUTCName(tzint) + tt = time.Unix(tsec, int64(tnsec)).In(time.FixedZone("", int(tzint)*60)) + } + return +} + +func timeLocUTCName(tzint int16) string { + if tzint == 0 { + return "UTC" + } + var tzname = []byte("UTC+00:00") + //tzname := fmt.Sprintf("UTC%s%02d:%02d", tzsign, tz/60, tz%60) //perf issue using Sprintf. inline below. + //tzhr, tzmin := tz/60, tz%60 //faster if u convert to int first + var tzhr, tzmin int16 + if tzint < 0 { + tzname[3] = '-' // (TODO: verify. this works here) + tzhr, tzmin = -tzint/60, (-tzint)%60 + } else { + tzhr, tzmin = tzint/60, tzint%60 + } + tzname[4] = timeDigits[tzhr/10] + tzname[5] = timeDigits[tzhr%10] + tzname[7] = timeDigits[tzmin/10] + tzname[8] = timeDigits[tzmin%10] + return string(tzname) + //return time.FixedZone(string(tzname), int(tzint)*60) +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/z_helper_test.go b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/z_helper_test.go new file mode 100644 index 0000000000000..2e9b3a0f05d03 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/go-msgpack/codec/z_helper_test.go @@ -0,0 +1,103 @@ +// Copyright (c) 2012, 2013 Ugorji Nwoke. All rights reserved. +// Use of this source code is governed by a BSD-style license found in the LICENSE file. + +package codec + +// All non-std package dependencies related to testing live in this file, +// so porting to different environment is easy (just update functions). +// +// Also, this file is called z_helper_test, to give a "hint" to compiler +// that its init() function should be called last. (not guaranteed by spec) + +import ( + "errors" + "reflect" + "flag" + "testing" +) + +var ( + testLogToT = true + failNowOnFail = true +) + +func init() { + testInitFlags() + benchInitFlags() + flag.Parse() + testInit() + benchInit() +} + +func checkErrT(t *testing.T, err error) { + if err != nil { + logT(t, err.Error()) + failT(t) + } +} + +func checkEqualT(t *testing.T, v1 interface{}, v2 interface{}, desc string) (err error) { + if err = deepEqual(v1, v2); err != nil { + logT(t, "Not Equal: %s: %v. v1: %v, v2: %v", desc, err, v1, v2) + failT(t) + } + return +} + +func logT(x interface{}, format string, args ...interface{}) { + if t, ok := x.(*testing.T); ok && t != nil && testLogToT { + t.Logf(format, args...) + } else if b, ok := x.(*testing.B); ok && b != nil && testLogToT { + b.Logf(format, args...) + } else { + debugf(format, args...) + } +} + +func failT(t *testing.T) { + if failNowOnFail { + t.FailNow() + } else { + t.Fail() + } +} + +func deepEqual(v1, v2 interface{}) (err error) { + if !reflect.DeepEqual(v1, v2) { + err = errors.New("Not Match") + } + return +} + +func approxDataSize(rv reflect.Value) (sum int) { + switch rk := rv.Kind(); rk { + case reflect.Invalid: + case reflect.Ptr, reflect.Interface: + sum += int(rv.Type().Size()) + sum += approxDataSize(rv.Elem()) + case reflect.Slice: + sum += int(rv.Type().Size()) + for j := 0; j < rv.Len(); j++ { + sum += approxDataSize(rv.Index(j)) + } + case reflect.String: + sum += int(rv.Type().Size()) + sum += rv.Len() + case reflect.Map: + sum += int(rv.Type().Size()) + for _, mk := range rv.MapKeys() { + sum += approxDataSize(mk) + sum += approxDataSize(rv.MapIndex(mk)) + } + case reflect.Struct: + //struct size already includes the full data size. + //sum += int(rv.Type().Size()) + for j := 0; j < rv.NumField(); j++ { + sum += approxDataSize(rv.Field(j)) + } + default: + //pure value types + sum += int(rv.Type().Size()) + } + return +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/LICENSE b/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/LICENSE new file mode 100644 index 0000000000000..f0e5c79e18115 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/LICENSE @@ -0,0 +1,362 @@ +Mozilla Public License, version 2.0 + +1. Definitions + +1.1. "Contributor" + + means each individual or legal entity that creates, contributes to the + creation of, or owns Covered Software. + +1.2. "Contributor Version" + + means the combination of the Contributions of others (if any) used by a + Contributor and that particular Contributor's Contribution. + +1.3. "Contribution" + + means Covered Software of a particular Contributor. + +1.4. "Covered Software" + + means Source Code Form to which the initial Contributor has attached the + notice in Exhibit A, the Executable Form of such Source Code Form, and + Modifications of such Source Code Form, in each case including portions + thereof. + +1.5. "Incompatible With Secondary Licenses" + means + + a. that the initial Contributor has attached the notice described in + Exhibit B to the Covered Software; or + + b. that the Covered Software was made available under the terms of + version 1.1 or earlier of the License, but not also under the terms of + a Secondary License. + +1.6. "Executable Form" + + means any form of the work other than Source Code Form. + +1.7. "Larger Work" + + means a work that combines Covered Software with other material, in a + separate file or files, that is not Covered Software. + +1.8. "License" + + means this document. + +1.9. "Licensable" + + means having the right to grant, to the maximum extent possible, whether + at the time of the initial grant or subsequently, any and all of the + rights conveyed by this License. + +1.10. "Modifications" + + means any of the following: + + a. any file in Source Code Form that results from an addition to, + deletion from, or modification of the contents of Covered Software; or + + b. any new file in Source Code Form that contains any Covered Software. + +1.11. "Patent Claims" of a Contributor + + means any patent claim(s), including without limitation, method, + process, and apparatus claims, in any patent Licensable by such + Contributor that would be infringed, but for the grant of the License, + by the making, using, selling, offering for sale, having made, import, + or transfer of either its Contributions or its Contributor Version. + +1.12. "Secondary License" + + means either the GNU General Public License, Version 2.0, the GNU Lesser + General Public License, Version 2.1, the GNU Affero General Public + License, Version 3.0, or any later versions of those licenses. + +1.13. "Source Code Form" + + means the form of the work preferred for making modifications. + +1.14. "You" (or "Your") + + means an individual or a legal entity exercising rights under this + License. For legal entities, "You" includes any entity that controls, is + controlled by, or is under common control with You. For purposes of this + definition, "control" means (a) the power, direct or indirect, to cause + the direction or management of such entity, whether by contract or + otherwise, or (b) ownership of more than fifty percent (50%) of the + outstanding shares or beneficial ownership of such entity. + + +2. License Grants and Conditions + +2.1. Grants + + Each Contributor hereby grants You a world-wide, royalty-free, + non-exclusive license: + + a. under intellectual property rights (other than patent or trademark) + Licensable by such Contributor to use, reproduce, make available, + modify, display, perform, distribute, and otherwise exploit its + Contributions, either on an unmodified basis, with Modifications, or + as part of a Larger Work; and + + b. under Patent Claims of such Contributor to make, use, sell, offer for + sale, have made, import, and otherwise transfer either its + Contributions or its Contributor Version. + +2.2. Effective Date + + The licenses granted in Section 2.1 with respect to any Contribution + become effective for each Contribution on the date the Contributor first + distributes such Contribution. + +2.3. Limitations on Grant Scope + + The licenses granted in this Section 2 are the only rights granted under + this License. No additional rights or licenses will be implied from the + distribution or licensing of Covered Software under this License. + Notwithstanding Section 2.1(b) above, no patent license is granted by a + Contributor: + + a. for any code that a Contributor has removed from Covered Software; or + + b. for infringements caused by: (i) Your and any other third party's + modifications of Covered Software, or (ii) the combination of its + Contributions with other software (except as part of its Contributor + Version); or + + c. under Patent Claims infringed by Covered Software in the absence of + its Contributions. + + This License does not grant any rights in the trademarks, service marks, + or logos of any Contributor (except as may be necessary to comply with + the notice requirements in Section 3.4). + +2.4. Subsequent Licenses + + No Contributor makes additional grants as a result of Your choice to + distribute the Covered Software under a subsequent version of this + License (see Section 10.2) or under the terms of a Secondary License (if + permitted under the terms of Section 3.3). + +2.5. Representation + + Each Contributor represents that the Contributor believes its + Contributions are its original creation(s) or it has sufficient rights to + grant the rights to its Contributions conveyed by this License. + +2.6. Fair Use + + This License is not intended to limit any rights You have under + applicable copyright doctrines of fair use, fair dealing, or other + equivalents. + +2.7. Conditions + + Sections 3.1, 3.2, 3.3, and 3.4 are conditions of the licenses granted in + Section 2.1. + + +3. Responsibilities + +3.1. Distribution of Source Form + + All distribution of Covered Software in Source Code Form, including any + Modifications that You create or to which You contribute, must be under + the terms of this License. You must inform recipients that the Source + Code Form of the Covered Software is governed by the terms of this + License, and how they can obtain a copy of this License. You may not + attempt to alter or restrict the recipients' rights in the Source Code + Form. + +3.2. Distribution of Executable Form + + If You distribute Covered Software in Executable Form then: + + a. such Covered Software must also be made available in Source Code Form, + as described in Section 3.1, and You must inform recipients of the + Executable Form how they can obtain a copy of such Source Code Form by + reasonable means in a timely manner, at a charge no more than the cost + of distribution to the recipient; and + + b. You may distribute such Executable Form under the terms of this + License, or sublicense it under different terms, provided that the + license for the Executable Form does not attempt to limit or alter the + recipients' rights in the Source Code Form under this License. + +3.3. Distribution of a Larger Work + + You may create and distribute a Larger Work under terms of Your choice, + provided that You also comply with the requirements of this License for + the Covered Software. If the Larger Work is a combination of Covered + Software with a work governed by one or more Secondary Licenses, and the + Covered Software is not Incompatible With Secondary Licenses, this + License permits You to additionally distribute such Covered Software + under the terms of such Secondary License(s), so that the recipient of + the Larger Work may, at their option, further distribute the Covered + Software under the terms of either this License or such Secondary + License(s). + +3.4. Notices + + You may not remove or alter the substance of any license notices + (including copyright notices, patent notices, disclaimers of warranty, or + limitations of liability) contained within the Source Code Form of the + Covered Software, except that You may alter any license notices to the + extent required to remedy known factual inaccuracies. + +3.5. Application of Additional Terms + + You may choose to offer, and to charge a fee for, warranty, support, + indemnity or liability obligations to one or more recipients of Covered + Software. However, You may do so only on Your own behalf, and not on + behalf of any Contributor. You must make it absolutely clear that any + such warranty, support, indemnity, or liability obligation is offered by + You alone, and You hereby agree to indemnify every Contributor for any + liability incurred by such Contributor as a result of warranty, support, + indemnity or liability terms You offer. You may include additional + disclaimers of warranty and limitations of liability specific to any + jurisdiction. + +4. Inability to Comply Due to Statute or Regulation + + If it is impossible for You to comply with any of the terms of this License + with respect to some or all of the Covered Software due to statute, + judicial order, or regulation then You must: (a) comply with the terms of + this License to the maximum extent possible; and (b) describe the + limitations and the code they affect. Such description must be placed in a + text file included with all distributions of the Covered Software under + this License. Except to the extent prohibited by statute or regulation, + such description must be sufficiently detailed for a recipient of ordinary + skill to be able to understand it. + +5. Termination + +5.1. The rights granted under this License will terminate automatically if You + fail to comply with any of its terms. However, if You become compliant, + then the rights granted under this License from a particular Contributor + are reinstated (a) provisionally, unless and until such Contributor + explicitly and finally terminates Your grants, and (b) on an ongoing + basis, if such Contributor fails to notify You of the non-compliance by + some reasonable means prior to 60 days after You have come back into + compliance. Moreover, Your grants from a particular Contributor are + reinstated on an ongoing basis if such Contributor notifies You of the + non-compliance by some reasonable means, this is the first time You have + received notice of non-compliance with this License from such + Contributor, and You become compliant prior to 30 days after Your receipt + of the notice. + +5.2. If You initiate litigation against any entity by asserting a patent + infringement claim (excluding declaratory judgment actions, + counter-claims, and cross-claims) alleging that a Contributor Version + directly or indirectly infringes any patent, then the rights granted to + You by any and all Contributors for the Covered Software under Section + 2.1 of this License shall terminate. + +5.3. In the event of termination under Sections 5.1 or 5.2 above, all end user + license agreements (excluding distributors and resellers) which have been + validly granted by You or Your distributors under this License prior to + termination shall survive termination. + +6. Disclaimer of Warranty + + Covered Software is provided under this License on an "as is" basis, + without warranty of any kind, either expressed, implied, or statutory, + including, without limitation, warranties that the Covered Software is free + of defects, merchantable, fit for a particular purpose or non-infringing. + The entire risk as to the quality and performance of the Covered Software + is with You. Should any Covered Software prove defective in any respect, + You (not any Contributor) assume the cost of any necessary servicing, + repair, or correction. This disclaimer of warranty constitutes an essential + part of this License. No use of any Covered Software is authorized under + this License except under this disclaimer. + +7. Limitation of Liability + + Under no circumstances and under no legal theory, whether tort (including + negligence), contract, or otherwise, shall any Contributor, or anyone who + distributes Covered Software as permitted above, be liable to You for any + direct, indirect, special, incidental, or consequential damages of any + character including, without limitation, damages for lost profits, loss of + goodwill, work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses, even if such party shall have been + informed of the possibility of such damages. This limitation of liability + shall not apply to liability for death or personal injury resulting from + such party's negligence to the extent applicable law prohibits such + limitation. Some jurisdictions do not allow the exclusion or limitation of + incidental or consequential damages, so this exclusion and limitation may + not apply to You. + +8. Litigation + + Any litigation relating to this License may be brought only in the courts + of a jurisdiction where the defendant maintains its principal place of + business and such litigation shall be governed by laws of that + jurisdiction, without reference to its conflict-of-law provisions. Nothing + in this Section shall prevent a party's ability to bring cross-claims or + counter-claims. + +9. Miscellaneous + + This License represents the complete agreement concerning the subject + matter hereof. If any provision of this License is held to be + unenforceable, such provision shall be reformed only to the extent + necessary to make it enforceable. Any law or regulation which provides that + the language of a contract shall be construed against the drafter shall not + be used to construe this License against a Contributor. + + +10. Versions of the License + +10.1. New Versions + + Mozilla Foundation is the license steward. Except as provided in Section + 10.3, no one other than the license steward has the right to modify or + publish new versions of this License. Each version will be given a + distinguishing version number. + +10.2. Effect of New Versions + + You may distribute the Covered Software under the terms of the version + of the License under which You originally received the Covered Software, + or under the terms of any subsequent version published by the license + steward. + +10.3. Modified Versions + + If you create software not governed by this License, and you want to + create a new license for such software, you may create and use a + modified version of this License if you rename the license and remove + any references to the name of the license steward (except to note that + such modified license differs from this License). + +10.4. Distributing Source Code Form that is Incompatible With Secondary + Licenses If You choose to distribute Source Code Form that is + Incompatible With Secondary Licenses under the terms of this version of + the License, the notice described in Exhibit B of this License must be + attached. + +Exhibit A - Source Code Form License Notice + + This Source Code Form is subject to the + terms of the Mozilla Public License, v. + 2.0. If a copy of the MPL was not + distributed with this file, You can + obtain one at + http://mozilla.org/MPL/2.0/. + +If it is not possible or desirable to put the notice in a particular file, +then You may include the notice in a location (such as a LICENSE file in a +relevant directory) where a recipient would be likely to look for such a +notice. + +You may add additional accurate notices of copyright ownership. + +Exhibit B - "Incompatible With Secondary Licenses" Notice + + This Source Code Form is "Incompatible + With Secondary Licenses", as defined by + the Mozilla Public License, v. 2.0. \ No newline at end of file diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/README.md b/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/README.md new file mode 100644 index 0000000000000..5d7180ab9ec57 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/README.md @@ -0,0 +1,11 @@ +raft-boltdb +=========== + +This repository provides the `raftboltdb` package. The package exports the +`BoltStore` which is an implementation of both a `LogStore` and `StableStore`. + +It is meant to be used as a backend for the `raft` [package +here](https://github.com/hashicorp/raft). + +This implementation uses [BoltDB](https://github.com/boltdb/bolt). BoltDB is +a simple key/value store implemented in pure Go, and inspired by LMDB. diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/bench_test.go b/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/bench_test.go new file mode 100644 index 0000000000000..b860706fd4e17 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/bench_test.go @@ -0,0 +1,88 @@ +package raftboltdb + +import ( + "os" + "testing" + + "github.com/hashicorp/raft/bench" +) + +func BenchmarkBoltStore_FirstIndex(b *testing.B) { + store := testBoltStore(b) + defer store.Close() + defer os.Remove(store.path) + + raftbench.FirstIndex(b, store) +} + +func BenchmarkBoltStore_LastIndex(b *testing.B) { + store := testBoltStore(b) + defer store.Close() + defer os.Remove(store.path) + + raftbench.LastIndex(b, store) +} + +func BenchmarkBoltStore_GetLog(b *testing.B) { + store := testBoltStore(b) + defer store.Close() + defer os.Remove(store.path) + + raftbench.GetLog(b, store) +} + +func BenchmarkBoltStore_StoreLog(b *testing.B) { + store := testBoltStore(b) + defer store.Close() + defer os.Remove(store.path) + + raftbench.StoreLog(b, store) +} + +func BenchmarkBoltStore_StoreLogs(b *testing.B) { + store := testBoltStore(b) + defer store.Close() + defer os.Remove(store.path) + + raftbench.StoreLogs(b, store) +} + +func BenchmarkBoltStore_DeleteRange(b *testing.B) { + store := testBoltStore(b) + defer store.Close() + defer os.Remove(store.path) + + raftbench.DeleteRange(b, store) +} + +func BenchmarkBoltStore_Set(b *testing.B) { + store := testBoltStore(b) + defer store.Close() + defer os.Remove(store.path) + + raftbench.Set(b, store) +} + +func BenchmarkBoltStore_Get(b *testing.B) { + store := testBoltStore(b) + defer store.Close() + defer os.Remove(store.path) + + raftbench.Get(b, store) +} + +func BenchmarkBoltStore_SetUint64(b *testing.B) { + store := testBoltStore(b) + defer store.Close() + defer os.Remove(store.path) + + raftbench.SetUint64(b, store) +} + +func BenchmarkBoltStore_GetUint64(b *testing.B) { + store := testBoltStore(b) + defer store.Close() + defer os.Remove(store.path) + + raftbench.GetUint64(b, store) +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/bolt_store.go b/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/bolt_store.go new file mode 100644 index 0000000000000..ab6dd4803e610 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/bolt_store.go @@ -0,0 +1,231 @@ +package raftboltdb + +import ( + "errors" + + "github.com/boltdb/bolt" + "github.com/hashicorp/raft" +) + +const ( + // Permissions to use on the db file. This is only used if the + // database file does not exist and needs to be created. + dbFileMode = 0600 +) + +var ( + // Bucket names we perform transactions in + dbLogs = []byte("logs") + dbConf = []byte("conf") + + // An error indicating a given key does not exist + ErrKeyNotFound = errors.New("not found") +) + +// BoltStore provides access to BoltDB for Raft to store and retrieve +// log entries. It also provides key/value storage, and can be used as +// a LogStore and StableStore. +type BoltStore struct { + // conn is the underlying handle to the db. + conn *bolt.DB + + // The path to the Bolt database file + path string +} + +// NewBoltStore takes a file path and returns a connected Raft backend. +func NewBoltStore(path string) (*BoltStore, error) { + // Try to connect + handle, err := bolt.Open(path, dbFileMode, nil) + if err != nil { + return nil, err + } + + // Create the new store + store := &BoltStore{ + conn: handle, + path: path, + } + + // Set up our buckets + if err := store.initialize(); err != nil { + store.Close() + return nil, err + } + + return store, nil +} + +// initialize is used to set up all of the buckets. +func (b *BoltStore) initialize() error { + tx, err := b.conn.Begin(true) + if err != nil { + return err + } + defer tx.Rollback() + + // Create all the buckets + if _, err := tx.CreateBucketIfNotExists(dbLogs); err != nil { + return err + } + if _, err := tx.CreateBucketIfNotExists(dbConf); err != nil { + return err + } + + return tx.Commit() +} + +// Close is used to gracefully close the DB connection. +func (b *BoltStore) Close() error { + return b.conn.Close() +} + +// FirstIndex returns the first known index from the Raft log. +func (b *BoltStore) FirstIndex() (uint64, error) { + tx, err := b.conn.Begin(false) + if err != nil { + return 0, err + } + defer tx.Rollback() + + curs := tx.Bucket(dbLogs).Cursor() + if first, _ := curs.First(); first == nil { + return 0, nil + } else { + return bytesToUint64(first), nil + } +} + +// LastIndex returns the last known index from the Raft log. +func (b *BoltStore) LastIndex() (uint64, error) { + tx, err := b.conn.Begin(false) + if err != nil { + return 0, err + } + defer tx.Rollback() + + curs := tx.Bucket(dbLogs).Cursor() + if last, _ := curs.Last(); last == nil { + return 0, nil + } else { + return bytesToUint64(last), nil + } +} + +// GetLog is used to retrieve a log from BoltDB at a given index. +func (b *BoltStore) GetLog(idx uint64, log *raft.Log) error { + tx, err := b.conn.Begin(false) + if err != nil { + return err + } + defer tx.Rollback() + + bucket := tx.Bucket(dbLogs) + val := bucket.Get(uint64ToBytes(idx)) + + if val == nil { + return raft.ErrLogNotFound + } + return decodeMsgPack(val, log) +} + +// StoreLog is used to store a single raft log +func (b *BoltStore) StoreLog(log *raft.Log) error { + return b.StoreLogs([]*raft.Log{log}) +} + +// StoreLogs is used to store a set of raft logs +func (b *BoltStore) StoreLogs(logs []*raft.Log) error { + tx, err := b.conn.Begin(true) + if err != nil { + return err + } + defer tx.Rollback() + + for _, log := range logs { + key := uint64ToBytes(log.Index) + val, err := encodeMsgPack(log) + if err != nil { + return err + } + bucket := tx.Bucket(dbLogs) + if err := bucket.Put(key, val.Bytes()); err != nil { + return err + } + } + + return tx.Commit() +} + +// DeleteRange is used to delete logs within a given range inclusively. +func (b *BoltStore) DeleteRange(min, max uint64) error { + minKey := uint64ToBytes(min) + + tx, err := b.conn.Begin(true) + if err != nil { + return err + } + defer tx.Rollback() + + curs := tx.Bucket(dbLogs).Cursor() + for k, _ := curs.Seek(minKey); k != nil; k, _ = curs.Next() { + // Handle out-of-range log index + if bytesToUint64(k) > max { + break + } + + // Delete in-range log index + if err := curs.Delete(); err != nil { + return err + } + } + + return tx.Commit() +} + +// Set is used to set a key/value set outside of the raft log +func (b *BoltStore) Set(k, v []byte) error { + tx, err := b.conn.Begin(true) + if err != nil { + return err + } + defer tx.Rollback() + + bucket := tx.Bucket(dbConf) + if err := bucket.Put(k, v); err != nil { + return err + } + + return tx.Commit() +} + +// Get is used to retrieve a value from the k/v store by key +func (b *BoltStore) Get(k []byte) ([]byte, error) { + tx, err := b.conn.Begin(false) + if err != nil { + return nil, err + } + defer tx.Rollback() + + bucket := tx.Bucket(dbConf) + val := bucket.Get(k) + + if val == nil { + return nil, ErrKeyNotFound + } + return append([]byte{}, val...), nil +} + +// SetUint64 is like Set, but handles uint64 values +func (b *BoltStore) SetUint64(key []byte, val uint64) error { + return b.Set(key, uint64ToBytes(val)) +} + +// GetUint64 is like Get, but handles uint64 values +func (b *BoltStore) GetUint64(key []byte) (uint64, error) { + val, err := b.Get(key) + if err != nil { + return 0, err + } + return bytesToUint64(val), nil +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/bolt_store_test.go b/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/bolt_store_test.go new file mode 100644 index 0000000000000..ab2a1525c813a --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/bolt_store_test.go @@ -0,0 +1,332 @@ +package raftboltdb + +import ( + "bytes" + "io/ioutil" + "os" + "reflect" + "testing" + + "github.com/boltdb/bolt" + "github.com/hashicorp/raft" +) + +func testBoltStore(t testing.TB) *BoltStore { + fh, err := ioutil.TempFile("", "bolt") + if err != nil { + t.Fatalf("err: %s", err) + } + os.Remove(fh.Name()) + + // Successfully creates and returns a store + store, err := NewBoltStore(fh.Name()) + if err != nil { + t.Fatalf("err: %s", err) + } + + return store +} + +func testRaftLog(idx uint64, data string) *raft.Log { + return &raft.Log{ + Data: []byte(data), + Index: idx, + } +} + +func TestBoltStore_Implements(t *testing.T) { + var store interface{} = &BoltStore{} + if _, ok := store.(raft.StableStore); !ok { + t.Fatalf("BoltStore does not implement raft.StableStore") + } + if _, ok := store.(raft.LogStore); !ok { + t.Fatalf("BoltStore does not implement raft.LogStore") + } +} + +func TestNewBoltStore(t *testing.T) { + fh, err := ioutil.TempFile("", "bolt") + if err != nil { + t.Fatalf("err: %s", err) + } + os.Remove(fh.Name()) + defer os.Remove(fh.Name()) + + // Successfully creates and returns a store + store, err := NewBoltStore(fh.Name()) + if err != nil { + t.Fatalf("err: %s", err) + } + + // Ensure the file was created + if store.path != fh.Name() { + t.Fatalf("unexpected file path %q", store.path) + } + if _, err := os.Stat(fh.Name()); err != nil { + t.Fatalf("err: %s", err) + } + + // Close the store so we can open again + if err := store.Close(); err != nil { + t.Fatalf("err: %s", err) + } + + // Ensure our tables were created + db, err := bolt.Open(fh.Name(), dbFileMode, nil) + if err != nil { + t.Fatalf("err: %s", err) + } + tx, err := db.Begin(true) + if err != nil { + t.Fatalf("err: %s", err) + } + if _, err := tx.CreateBucket([]byte(dbLogs)); err != bolt.ErrBucketExists { + t.Fatalf("bad: %v", err) + } + if _, err := tx.CreateBucket([]byte(dbConf)); err != bolt.ErrBucketExists { + t.Fatalf("bad: %v", err) + } +} + +func TestBoltStore_FirstIndex(t *testing.T) { + store := testBoltStore(t) + defer store.Close() + defer os.Remove(store.path) + + // Should get 0 index on empty log + idx, err := store.FirstIndex() + if err != nil { + t.Fatalf("err: %s", err) + } + if idx != 0 { + t.Fatalf("bad: %v", idx) + } + + // Set a mock raft log + logs := []*raft.Log{ + testRaftLog(1, "log1"), + testRaftLog(2, "log2"), + testRaftLog(3, "log3"), + } + if err := store.StoreLogs(logs); err != nil { + t.Fatalf("bad: %s", err) + } + + // Fetch the first Raft index + idx, err = store.FirstIndex() + if err != nil { + t.Fatalf("err: %s", err) + } + if idx != 1 { + t.Fatalf("bad: %d", idx) + } +} + +func TestBoltStore_LastIndex(t *testing.T) { + store := testBoltStore(t) + defer store.Close() + defer os.Remove(store.path) + + // Should get 0 index on empty log + idx, err := store.LastIndex() + if err != nil { + t.Fatalf("err: %s", err) + } + if idx != 0 { + t.Fatalf("bad: %v", idx) + } + + // Set a mock raft log + logs := []*raft.Log{ + testRaftLog(1, "log1"), + testRaftLog(2, "log2"), + testRaftLog(3, "log3"), + } + if err := store.StoreLogs(logs); err != nil { + t.Fatalf("bad: %s", err) + } + + // Fetch the last Raft index + idx, err = store.LastIndex() + if err != nil { + t.Fatalf("err: %s", err) + } + if idx != 3 { + t.Fatalf("bad: %d", idx) + } +} + +func TestBoltStore_GetLog(t *testing.T) { + store := testBoltStore(t) + defer store.Close() + defer os.Remove(store.path) + + log := new(raft.Log) + + // Should return an error on non-existent log + if err := store.GetLog(1, log); err != raft.ErrLogNotFound { + t.Fatalf("expected raft log not found error, got: %v", err) + } + + // Set a mock raft log + logs := []*raft.Log{ + testRaftLog(1, "log1"), + testRaftLog(2, "log2"), + testRaftLog(3, "log3"), + } + if err := store.StoreLogs(logs); err != nil { + t.Fatalf("bad: %s", err) + } + + // Should return the proper log + if err := store.GetLog(2, log); err != nil { + t.Fatalf("err: %s", err) + } + if !reflect.DeepEqual(log, logs[1]) { + t.Fatalf("bad: %#v", log) + } +} + +func TestBoltStore_SetLog(t *testing.T) { + store := testBoltStore(t) + defer store.Close() + defer os.Remove(store.path) + + // Create the log + log := &raft.Log{ + Data: []byte("log1"), + Index: 1, + } + + // Attempt to store the log + if err := store.StoreLog(log); err != nil { + t.Fatalf("err: %s", err) + } + + // Retrieve the log again + result := new(raft.Log) + if err := store.GetLog(1, result); err != nil { + t.Fatalf("err: %s", err) + } + + // Ensure the log comes back the same + if !reflect.DeepEqual(log, result) { + t.Fatalf("bad: %v", result) + } +} + +func TestBoltStore_SetLogs(t *testing.T) { + store := testBoltStore(t) + defer store.Close() + defer os.Remove(store.path) + + // Create a set of logs + logs := []*raft.Log{ + testRaftLog(1, "log1"), + testRaftLog(2, "log2"), + } + + // Attempt to store the logs + if err := store.StoreLogs(logs); err != nil { + t.Fatalf("err: %s", err) + } + + // Ensure we stored them all + result1, result2 := new(raft.Log), new(raft.Log) + if err := store.GetLog(1, result1); err != nil { + t.Fatalf("err: %s", err) + } + if !reflect.DeepEqual(logs[0], result1) { + t.Fatalf("bad: %#v", result1) + } + if err := store.GetLog(2, result2); err != nil { + t.Fatalf("err: %s", err) + } + if !reflect.DeepEqual(logs[1], result2) { + t.Fatalf("bad: %#v", result2) + } +} + +func TestBoltStore_DeleteRange(t *testing.T) { + store := testBoltStore(t) + defer store.Close() + defer os.Remove(store.path) + + // Create a set of logs + log1 := testRaftLog(1, "log1") + log2 := testRaftLog(2, "log2") + log3 := testRaftLog(3, "log3") + logs := []*raft.Log{log1, log2, log3} + + // Attempt to store the logs + if err := store.StoreLogs(logs); err != nil { + t.Fatalf("err: %s", err) + } + + // Attempt to delete a range of logs + if err := store.DeleteRange(1, 2); err != nil { + t.Fatalf("err: %s", err) + } + + // Ensure the logs were deleted + if err := store.GetLog(1, new(raft.Log)); err != raft.ErrLogNotFound { + t.Fatalf("should have deleted log1") + } + if err := store.GetLog(2, new(raft.Log)); err != raft.ErrLogNotFound { + t.Fatalf("should have deleted log2") + } +} + +func TestBoltStore_Set_Get(t *testing.T) { + store := testBoltStore(t) + defer store.Close() + defer os.Remove(store.path) + + // Returns error on non-existent key + if _, err := store.Get([]byte("bad")); err != ErrKeyNotFound { + t.Fatalf("expected not found error, got: %q", err) + } + + k, v := []byte("hello"), []byte("world") + + // Try to set a k/v pair + if err := store.Set(k, v); err != nil { + t.Fatalf("err: %s", err) + } + + // Try to read it back + val, err := store.Get(k) + if err != nil { + t.Fatalf("err: %s", err) + } + if !bytes.Equal(val, v) { + t.Fatalf("bad: %v", val) + } +} + +func TestBoltStore_SetUint64_GetUint64(t *testing.T) { + store := testBoltStore(t) + defer store.Close() + defer os.Remove(store.path) + + // Returns error on non-existent key + if _, err := store.GetUint64([]byte("bad")); err != ErrKeyNotFound { + t.Fatalf("expected not found error, got: %q", err) + } + + k, v := []byte("abc"), uint64(123) + + // Attempt to set the k/v pair + if err := store.SetUint64(k, v); err != nil { + t.Fatalf("err: %s", err) + } + + // Read back the value + val, err := store.GetUint64(k) + if err != nil { + t.Fatalf("err: %s", err) + } + if val != v { + t.Fatalf("bad: %v", val) + } +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/util.go b/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/util.go new file mode 100644 index 0000000000000..68dd786b7adeb --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft-boltdb/util.go @@ -0,0 +1,37 @@ +package raftboltdb + +import ( + "bytes" + "encoding/binary" + + "github.com/hashicorp/go-msgpack/codec" +) + +// Decode reverses the encode operation on a byte slice input +func decodeMsgPack(buf []byte, out interface{}) error { + r := bytes.NewBuffer(buf) + hd := codec.MsgpackHandle{} + dec := codec.NewDecoder(r, &hd) + return dec.Decode(out) +} + +// Encode writes an encoded object to a new bytes buffer +func encodeMsgPack(in interface{}) (*bytes.Buffer, error) { + buf := bytes.NewBuffer(nil) + hd := codec.MsgpackHandle{} + enc := codec.NewEncoder(buf, &hd) + err := enc.Encode(in) + return buf, err +} + +// Converts bytes to an integer +func bytesToUint64(b []byte) uint64 { + return binary.BigEndian.Uint64(b) +} + +// Converts a uint to a byte slice +func uint64ToBytes(u uint64) []byte { + buf := make([]byte, 8) + binary.BigEndian.PutUint64(buf, u) + return buf +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/.gitignore b/Godeps/_workspace/src/github.com/hashicorp/raft/.gitignore new file mode 100644 index 0000000000000..836562412fe8a --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/.gitignore @@ -0,0 +1,23 @@ +# Compiled Object files, Static and Dynamic libs (Shared Objects) +*.o +*.a +*.so + +# Folders +_obj +_test + +# Architecture specific extensions/prefixes +*.[568vq] +[568vq].out + +*.cgo1.go +*.cgo2.c +_cgo_defun.c +_cgo_gotypes.go +_cgo_export.* + +_testmain.go + +*.exe +*.test diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/.travis.yml b/Godeps/_workspace/src/github.com/hashicorp/raft/.travis.yml new file mode 100644 index 0000000000000..5cf041d263a4d --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/.travis.yml @@ -0,0 +1,14 @@ +language: go + +go: + - 1.2 + - tip + +install: make deps +script: + - make integ + +notifications: + flowdock: + secure: fZrcf9rlh2IrQrlch1sHkn3YI7SKvjGnAl/zyV5D6NROe1Bbr6d3QRMuCXWWdhJHzjKmXk5rIzbqJhUc0PNF7YjxGNKSzqWMQ56KcvN1k8DzlqxpqkcA3Jbs6fXCWo2fssRtZ7hj/wOP1f5n6cc7kzHDt9dgaYJ6nO2fqNPJiTc= + diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/LICENSE b/Godeps/_workspace/src/github.com/hashicorp/raft/LICENSE new file mode 100644 index 0000000000000..c33dcc7c928c6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/LICENSE @@ -0,0 +1,354 @@ +Mozilla Public License, version 2.0 + +1. Definitions + +1.1. “Contributor” + + means each individual or legal entity that creates, contributes to the + creation of, or owns Covered Software. + +1.2. “Contributor Version” + + means the combination of the Contributions of others (if any) used by a + Contributor and that particular Contributor’s Contribution. + +1.3. “Contribution” + + means Covered Software of a particular Contributor. + +1.4. “Covered Software” + + means Source Code Form to which the initial Contributor has attached the + notice in Exhibit A, the Executable Form of such Source Code Form, and + Modifications of such Source Code Form, in each case including portions + thereof. + +1.5. “Incompatible With Secondary Licenses” + means + + a. that the initial Contributor has attached the notice described in + Exhibit B to the Covered Software; or + + b. that the Covered Software was made available under the terms of version + 1.1 or earlier of the License, but not also under the terms of a + Secondary License. + +1.6. “Executable Form” + + means any form of the work other than Source Code Form. + +1.7. “Larger Work” + + means a work that combines Covered Software with other material, in a separate + file or files, that is not Covered Software. + +1.8. “License” + + means this document. + +1.9. “Licensable” + + means having the right to grant, to the maximum extent possible, whether at the + time of the initial grant or subsequently, any and all of the rights conveyed by + this License. + +1.10. “Modifications” + + means any of the following: + + a. any file in Source Code Form that results from an addition to, deletion + from, or modification of the contents of Covered Software; or + + b. any new file in Source Code Form that contains any Covered Software. + +1.11. “Patent Claims” of a Contributor + + means any patent claim(s), including without limitation, method, process, + and apparatus claims, in any patent Licensable by such Contributor that + would be infringed, but for the grant of the License, by the making, + using, selling, offering for sale, having made, import, or transfer of + either its Contributions or its Contributor Version. + +1.12. “Secondary License” + + means either the GNU General Public License, Version 2.0, the GNU Lesser + General Public License, Version 2.1, the GNU Affero General Public + License, Version 3.0, or any later versions of those licenses. + +1.13. “Source Code Form” + + means the form of the work preferred for making modifications. + +1.14. “You” (or “Your”) + + means an individual or a legal entity exercising rights under this + License. For legal entities, “You” includes any entity that controls, is + controlled by, or is under common control with You. For purposes of this + definition, “control” means (a) the power, direct or indirect, to cause + the direction or management of such entity, whether by contract or + otherwise, or (b) ownership of more than fifty percent (50%) of the + outstanding shares or beneficial ownership of such entity. + + +2. License Grants and Conditions + +2.1. Grants + + Each Contributor hereby grants You a world-wide, royalty-free, + non-exclusive license: + + a. under intellectual property rights (other than patent or trademark) + Licensable by such Contributor to use, reproduce, make available, + modify, display, perform, distribute, and otherwise exploit its + Contributions, either on an unmodified basis, with Modifications, or as + part of a Larger Work; and + + b. under Patent Claims of such Contributor to make, use, sell, offer for + sale, have made, import, and otherwise transfer either its Contributions + or its Contributor Version. + +2.2. Effective Date + + The licenses granted in Section 2.1 with respect to any Contribution become + effective for each Contribution on the date the Contributor first distributes + such Contribution. + +2.3. Limitations on Grant Scope + + The licenses granted in this Section 2 are the only rights granted under this + License. No additional rights or licenses will be implied from the distribution + or licensing of Covered Software under this License. Notwithstanding Section + 2.1(b) above, no patent license is granted by a Contributor: + + a. for any code that a Contributor has removed from Covered Software; or + + b. for infringements caused by: (i) Your and any other third party’s + modifications of Covered Software, or (ii) the combination of its + Contributions with other software (except as part of its Contributor + Version); or + + c. under Patent Claims infringed by Covered Software in the absence of its + Contributions. + + This License does not grant any rights in the trademarks, service marks, or + logos of any Contributor (except as may be necessary to comply with the + notice requirements in Section 3.4). + +2.4. Subsequent Licenses + + No Contributor makes additional grants as a result of Your choice to + distribute the Covered Software under a subsequent version of this License + (see Section 10.2) or under the terms of a Secondary License (if permitted + under the terms of Section 3.3). + +2.5. Representation + + Each Contributor represents that the Contributor believes its Contributions + are its original creation(s) or it has sufficient rights to grant the + rights to its Contributions conveyed by this License. + +2.6. Fair Use + + This License is not intended to limit any rights You have under applicable + copyright doctrines of fair use, fair dealing, or other equivalents. + +2.7. Conditions + + Sections 3.1, 3.2, 3.3, and 3.4 are conditions of the licenses granted in + Section 2.1. + + +3. Responsibilities + +3.1. Distribution of Source Form + + All distribution of Covered Software in Source Code Form, including any + Modifications that You create or to which You contribute, must be under the + terms of this License. You must inform recipients that the Source Code Form + of the Covered Software is governed by the terms of this License, and how + they can obtain a copy of this License. You may not attempt to alter or + restrict the recipients’ rights in the Source Code Form. + +3.2. Distribution of Executable Form + + If You distribute Covered Software in Executable Form then: + + a. such Covered Software must also be made available in Source Code Form, + as described in Section 3.1, and You must inform recipients of the + Executable Form how they can obtain a copy of such Source Code Form by + reasonable means in a timely manner, at a charge no more than the cost + of distribution to the recipient; and + + b. You may distribute such Executable Form under the terms of this License, + or sublicense it under different terms, provided that the license for + the Executable Form does not attempt to limit or alter the recipients’ + rights in the Source Code Form under this License. + +3.3. Distribution of a Larger Work + + You may create and distribute a Larger Work under terms of Your choice, + provided that You also comply with the requirements of this License for the + Covered Software. If the Larger Work is a combination of Covered Software + with a work governed by one or more Secondary Licenses, and the Covered + Software is not Incompatible With Secondary Licenses, this License permits + You to additionally distribute such Covered Software under the terms of + such Secondary License(s), so that the recipient of the Larger Work may, at + their option, further distribute the Covered Software under the terms of + either this License or such Secondary License(s). + +3.4. Notices + + You may not remove or alter the substance of any license notices (including + copyright notices, patent notices, disclaimers of warranty, or limitations + of liability) contained within the Source Code Form of the Covered + Software, except that You may alter any license notices to the extent + required to remedy known factual inaccuracies. + +3.5. Application of Additional Terms + + You may choose to offer, and to charge a fee for, warranty, support, + indemnity or liability obligations to one or more recipients of Covered + Software. However, You may do so only on Your own behalf, and not on behalf + of any Contributor. You must make it absolutely clear that any such + warranty, support, indemnity, or liability obligation is offered by You + alone, and You hereby agree to indemnify every Contributor for any + liability incurred by such Contributor as a result of warranty, support, + indemnity or liability terms You offer. You may include additional + disclaimers of warranty and limitations of liability specific to any + jurisdiction. + +4. Inability to Comply Due to Statute or Regulation + + If it is impossible for You to comply with any of the terms of this License + with respect to some or all of the Covered Software due to statute, judicial + order, or regulation then You must: (a) comply with the terms of this License + to the maximum extent possible; and (b) describe the limitations and the code + they affect. Such description must be placed in a text file included with all + distributions of the Covered Software under this License. Except to the + extent prohibited by statute or regulation, such description must be + sufficiently detailed for a recipient of ordinary skill to be able to + understand it. + +5. Termination + +5.1. The rights granted under this License will terminate automatically if You + fail to comply with any of its terms. However, if You become compliant, + then the rights granted under this License from a particular Contributor + are reinstated (a) provisionally, unless and until such Contributor + explicitly and finally terminates Your grants, and (b) on an ongoing basis, + if such Contributor fails to notify You of the non-compliance by some + reasonable means prior to 60 days after You have come back into compliance. + Moreover, Your grants from a particular Contributor are reinstated on an + ongoing basis if such Contributor notifies You of the non-compliance by + some reasonable means, this is the first time You have received notice of + non-compliance with this License from such Contributor, and You become + compliant prior to 30 days after Your receipt of the notice. + +5.2. If You initiate litigation against any entity by asserting a patent + infringement claim (excluding declaratory judgment actions, counter-claims, + and cross-claims) alleging that a Contributor Version directly or + indirectly infringes any patent, then the rights granted to You by any and + all Contributors for the Covered Software under Section 2.1 of this License + shall terminate. + +5.3. In the event of termination under Sections 5.1 or 5.2 above, all end user + license agreements (excluding distributors and resellers) which have been + validly granted by You or Your distributors under this License prior to + termination shall survive termination. + +6. Disclaimer of Warranty + + Covered Software is provided under this License on an “as is” basis, without + warranty of any kind, either expressed, implied, or statutory, including, + without limitation, warranties that the Covered Software is free of defects, + merchantable, fit for a particular purpose or non-infringing. The entire + risk as to the quality and performance of the Covered Software is with You. + Should any Covered Software prove defective in any respect, You (not any + Contributor) assume the cost of any necessary servicing, repair, or + correction. This disclaimer of warranty constitutes an essential part of this + License. No use of any Covered Software is authorized under this License + except under this disclaimer. + +7. Limitation of Liability + + Under no circumstances and under no legal theory, whether tort (including + negligence), contract, or otherwise, shall any Contributor, or anyone who + distributes Covered Software as permitted above, be liable to You for any + direct, indirect, special, incidental, or consequential damages of any + character including, without limitation, damages for lost profits, loss of + goodwill, work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses, even if such party shall have been + informed of the possibility of such damages. This limitation of liability + shall not apply to liability for death or personal injury resulting from such + party’s negligence to the extent applicable law prohibits such limitation. + Some jurisdictions do not allow the exclusion or limitation of incidental or + consequential damages, so this exclusion and limitation may not apply to You. + +8. Litigation + + Any litigation relating to this License may be brought only in the courts of + a jurisdiction where the defendant maintains its principal place of business + and such litigation shall be governed by laws of that jurisdiction, without + reference to its conflict-of-law provisions. Nothing in this Section shall + prevent a party’s ability to bring cross-claims or counter-claims. + +9. Miscellaneous + + This License represents the complete agreement concerning the subject matter + hereof. If any provision of this License is held to be unenforceable, such + provision shall be reformed only to the extent necessary to make it + enforceable. Any law or regulation which provides that the language of a + contract shall be construed against the drafter shall not be used to construe + this License against a Contributor. + + +10. Versions of the License + +10.1. New Versions + + Mozilla Foundation is the license steward. Except as provided in Section + 10.3, no one other than the license steward has the right to modify or + publish new versions of this License. Each version will be given a + distinguishing version number. + +10.2. Effect of New Versions + + You may distribute the Covered Software under the terms of the version of + the License under which You originally received the Covered Software, or + under the terms of any subsequent version published by the license + steward. + +10.3. Modified Versions + + If you create software not governed by this License, and you want to + create a new license for such software, you may create and use a modified + version of this License if you rename the license and remove any + references to the name of the license steward (except to note that such + modified license differs from this License). + +10.4. Distributing Source Code Form that is Incompatible With Secondary Licenses + If You choose to distribute Source Code Form that is Incompatible With + Secondary Licenses under the terms of this version of the License, the + notice described in Exhibit B of this License must be attached. + +Exhibit A - Source Code Form License Notice + + This Source Code Form is subject to the + terms of the Mozilla Public License, v. + 2.0. If a copy of the MPL was not + distributed with this file, You can + obtain one at + http://mozilla.org/MPL/2.0/. + +If it is not possible or desirable to put the notice in a particular file, then +You may include the notice in a location (such as a LICENSE file in a relevant +directory) where a recipient would be likely to look for such a notice. + +You may add additional accurate notices of copyright ownership. + +Exhibit B - “Incompatible With Secondary Licenses” Notice + + This Source Code Form is “Incompatible + With Secondary Licenses”, as defined by + the Mozilla Public License, v. 2.0. + diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/Makefile b/Godeps/_workspace/src/github.com/hashicorp/raft/Makefile new file mode 100644 index 0000000000000..c61b34a8f6c7d --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/Makefile @@ -0,0 +1,17 @@ +DEPS = $(go list -f '{{range .TestImports}}{{.}} {{end}}' ./...) + +test: + go test -timeout=5s ./... + +integ: test + INTEG_TESTS=yes go test -timeout=3s -run=Integ ./... + +deps: + go get -d -v ./... + echo $(DEPS) | xargs -n1 go get -d + +cov: + INTEG_TESTS=yes gocov test github.com/hashicorp/raft | gocov-html > /tmp/coverage.html + open /tmp/coverage.html + +.PHONY: test cov integ deps diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/README.md b/Godeps/_workspace/src/github.com/hashicorp/raft/README.md new file mode 100644 index 0000000000000..ecb6c977eea13 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/README.md @@ -0,0 +1,89 @@ +raft [![Build Status](https://travis-ci.org/hashicorp/raft.png)](https://travis-ci.org/hashicorp/raft) +==== + +raft is a [Go](http://www.golang.org) library that manages a replicated +log and can be used with an FSM to manage replicated state machines. It +is library for providing [consensus](http://en.wikipedia.org/wiki/Consensus_(computer_science)). + +The use cases for such a library are far-reaching as replicated state +machines are a key component of many distributed systems. They enable +building Consistent, Partition Tolerant (CP) systems, with limited +fault tolerance as well. + +## Building + +If you wish to build raft you'll need Go version 1.2+ installed. + +Please check your installation with: + +``` +go version +``` + +## Documentation + +For complete documentation, see the associated [Godoc](http://godoc.org/github.com/hashicorp/raft). + +To prevent complications with cgo, the primary backend `MDBStore` is in a separate repositoy, +called [raft-mdb](http://github.com/hashicorp/raft-mdb). That is the recommended implementation +for the `LogStore` and `StableStore`. + +A pure Go backend using [BoltDB](https://github.com/boltdb/bolt) is also available called +[raft-boltdb](https://github.com/hashicorp/raft-boltdb). It can also be used as a `LogStore` +and `StableStore`. + +## Protocol + +raft is based on ["Raft: In Search of an Understandable Consensus Algorithm"](https://ramcloud.stanford.edu/wiki/download/attachments/11370504/raft.pdf) + +A high level overview of the Raft protocol is described below, but for details please read the full +[Raft paper](https://ramcloud.stanford.edu/wiki/download/attachments/11370504/raft.pdf) +followed by the raft source. Any questions about the raft protocol should be sent to the +[raft-dev mailing list](https://groups.google.com/forum/#!forum/raft-dev). + +### Protocol Description + +Raft nodes are always in one of three states: follower, candidate or leader. All +nodes initially start out as a follower. In this state, nodes can accept log entries +from a leader and cast votes. If no entries are received for some time, nodes +self-promote to the candidate state. In the candidate state nodes request votes from +their peers. If a candidate receives a quorum of votes, then it is promoted to a leader. +The leader must accept new log entries and replicate to all the other followers. +In addition, if stale reads are not acceptable, all queries must also be performed on +the leader. + +Once a cluster has a leader, it is able to accept new log entries. A client can +request that a leader append a new log entry, which is an opaque binary blob to +Raft. The leader then writes the entry to durable storage and attempts to replicate +to a quorum of followers. Once the log entry is considered *committed*, it can be +*applied* to a finite state machine. The finite state machine is application specific, +and is implemented using an interface. + +An obvious question relates to the unbounded nature of a replicated log. Raft provides +a mechanism by which the current state is snapshotted, and the log is compacted. Because +of the FSM abstraction, restoring the state of the FSM must result in the same state +as a replay of old logs. This allows Raft to capture the FSM state at a point in time, +and then remove all the logs that were used to reach that state. This is performed automatically +without user intervention, and prevents unbounded disk usage as well as minimizing +time spent replaying logs. + +Lastly, there is the issue of updating the peer set when new servers are joining +or existing servers are leaving. As long as a quorum of nodes is available, this +is not an issue as Raft provides mechanisms to dynamically update the peer set. +If a quorum of nodes is unavailable, then this becomes a very challenging issue. +For example, suppose there are only 2 peers, A and B. The quorum size is also +2, meaning both nodes must agree to commit a log entry. If either A or B fails, +it is now impossible to reach quorum. This means the cluster is unable to add, +or remove a node, or commit any additional log entries. This results in *unavailability*. +At this point, manual intervention would be required to remove either A or B, +and to restart the remaining node in bootstrap mode. + +A Raft cluster of 3 nodes can tolerate a single node failure, while a cluster +of 5 can tolerate 2 node failures. The recommended configuration is to either +run 3 or 5 raft servers. This maximizes availability without +greatly sacrificing performance. + +In terms of performance, Raft is comparable to Paxos. Assuming stable leadership, +committing a log entry requires a single round trip to half of the cluster. +Thus performance is bound by disk I/O and network latency. + diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/bench/bench.go b/Godeps/_workspace/src/github.com/hashicorp/raft/bench/bench.go new file mode 100644 index 0000000000000..d7a58f45f4486 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/bench/bench.go @@ -0,0 +1,171 @@ +package raftbench + +// raftbench provides common benchmarking functions which can be used by +// anything which implements the raft.LogStore and raft.StableStore interfaces. +// All functions accept these interfaces and perform benchmarking. This +// makes comparing backend performance easier by sharing the tests. + +import ( + "github.com/hashicorp/raft" + "testing" +) + +func FirstIndex(b *testing.B, store raft.LogStore) { + // Create some fake data + var logs []*raft.Log + for i := 1; i < 10; i++ { + logs = append(logs, &raft.Log{Index: uint64(i), Data: []byte("data")}) + } + if err := store.StoreLogs(logs); err != nil { + b.Fatalf("err: %s", err) + } + b.ResetTimer() + + // Run FirstIndex a number of times + for n := 0; n < b.N; n++ { + store.FirstIndex() + } +} + +func LastIndex(b *testing.B, store raft.LogStore) { + // Create some fake data + var logs []*raft.Log + for i := 1; i < 10; i++ { + logs = append(logs, &raft.Log{Index: uint64(i), Data: []byte("data")}) + } + if err := store.StoreLogs(logs); err != nil { + b.Fatalf("err: %s", err) + } + b.ResetTimer() + + // Run LastIndex a number of times + for n := 0; n < b.N; n++ { + store.LastIndex() + } +} + +func GetLog(b *testing.B, store raft.LogStore) { + // Create some fake data + var logs []*raft.Log + for i := 1; i < 10; i++ { + logs = append(logs, &raft.Log{Index: uint64(i), Data: []byte("data")}) + } + if err := store.StoreLogs(logs); err != nil { + b.Fatalf("err: %s", err) + } + b.ResetTimer() + + // Run GetLog a number of times + for n := 0; n < b.N; n++ { + if err := store.GetLog(5, new(raft.Log)); err != nil { + b.Fatalf("err: %s", err) + } + } +} + +func StoreLog(b *testing.B, store raft.LogStore) { + // Run StoreLog a number of times + for n := 0; n < b.N; n++ { + log := &raft.Log{Index: uint64(n), Data: []byte("data")} + if err := store.StoreLog(log); err != nil { + b.Fatalf("err: %s", err) + } + } +} + +func StoreLogs(b *testing.B, store raft.LogStore) { + // Run StoreLogs a number of times. We want to set multiple logs each + // run, so we create 3 logs with incrementing indexes for each iteration. + for n := 0; n < b.N; n++ { + b.StopTimer() + offset := 3 * (n + 1) + logs := []*raft.Log{ + &raft.Log{Index: uint64(offset - 2), Data: []byte("data")}, + &raft.Log{Index: uint64(offset - 1), Data: []byte("data")}, + &raft.Log{Index: uint64(offset), Data: []byte("data")}, + } + b.StartTimer() + + if err := store.StoreLogs(logs); err != nil { + b.Fatalf("err: %s", err) + } + } +} + +func DeleteRange(b *testing.B, store raft.LogStore) { + // Create some fake data. In this case, we create 3 new log entries for each + // test case, and separate them by index in multiples of 10. This allows + // some room so that we can test deleting ranges with "extra" logs to + // to ensure we stop going to the database once our max index is hit. + var logs []*raft.Log + for n := 0; n < b.N; n++ { + offset := 10 * n + for i := offset; i < offset+3; i++ { + logs = append(logs, &raft.Log{Index: uint64(i), Data: []byte("data")}) + } + } + if err := store.StoreLogs(logs); err != nil { + b.Fatalf("err: %s", err) + } + b.ResetTimer() + + // Delete a range of the data + for n := 0; n < b.N; n++ { + offset := 10 * n + if err := store.DeleteRange(uint64(offset), uint64(offset+9)); err != nil { + b.Fatalf("err: %s", err) + } + } +} + +func Set(b *testing.B, store raft.StableStore) { + // Run Set a number of times + for n := 0; n < b.N; n++ { + if err := store.Set([]byte{byte(n)}, []byte("val")); err != nil { + b.Fatalf("err: %s", err) + } + } +} + +func Get(b *testing.B, store raft.StableStore) { + // Create some fake data + for i := 1; i < 10; i++ { + if err := store.Set([]byte{byte(i)}, []byte("val")); err != nil { + b.Fatalf("err: %s", err) + } + } + b.ResetTimer() + + // Run Get a number of times + for n := 0; n < b.N; n++ { + if _, err := store.Get([]byte{0x05}); err != nil { + b.Fatalf("err: %s", err) + } + } +} + +func SetUint64(b *testing.B, store raft.StableStore) { + // Run SetUint64 a number of times + for n := 0; n < b.N; n++ { + if err := store.SetUint64([]byte{byte(n)}, uint64(n)); err != nil { + b.Fatalf("err: %s", err) + } + } +} + +func GetUint64(b *testing.B, store raft.StableStore) { + // Create some fake data + for i := 0; i < 10; i++ { + if err := store.SetUint64([]byte{byte(i)}, uint64(i)); err != nil { + b.Fatalf("err: %s", err) + } + } + b.ResetTimer() + + // Run GetUint64 a number of times + for n := 0; n < b.N; n++ { + if _, err := store.Get([]byte{0x05}); err != nil { + b.Fatalf("err: %s", err) + } + } +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/commands.go b/Godeps/_workspace/src/github.com/hashicorp/raft/commands.go new file mode 100644 index 0000000000000..fd019484101b6 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/commands.go @@ -0,0 +1,80 @@ +package raft + +// AppendEntriesRequest is the command used to append entries to the +// replicated log. +type AppendEntriesRequest struct { + // Provide the current term and leader + Term uint64 + Leader []byte + + // Provide the previous entries for integrity checking + PrevLogEntry uint64 + PrevLogTerm uint64 + + // New entries to commit + Entries []*Log + + // Commit index on the leader + LeaderCommitIndex uint64 +} + +// AppendEntriesResponse is the response returned from an +// AppendEntriesRequest. +type AppendEntriesResponse struct { + // Newer term if leader is out of date + Term uint64 + + // Last Log is a hint to help accelerate rebuilding slow nodes + LastLog uint64 + + // We may not succeed if we have a conflicting entry + Success bool +} + +// RequestVoteRequest is the command used by a candidate to ask a Raft peer +// for a vote in an election. +type RequestVoteRequest struct { + // Provide the term and our id + Term uint64 + Candidate []byte + + // Used to ensure safety + LastLogIndex uint64 + LastLogTerm uint64 +} + +// RequestVoteResponse is the response returned from a RequestVoteRequest. +type RequestVoteResponse struct { + // Newer term if leader is out of date + Term uint64 + + // Return the peers, so that a node can shutdown on removal + Peers []byte + + // Is the vote granted + Granted bool +} + +// InstallSnapshotRequest is the command sent to a Raft peer to bootstrap its +// log (and state machine) from a snapshot on another peer. +type InstallSnapshotRequest struct { + Term uint64 + Leader []byte + + // These are the last index/term included in the snapshot + LastLogIndex uint64 + LastLogTerm uint64 + + // Peer Set in the snapshot + Peers []byte + + // Size of the snapshot + Size int64 +} + +// InstallSnapshotResponse is the response returned from an +// InstallSnapshotRequest. +type InstallSnapshotResponse struct { + Term uint64 + Success bool +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/config.go b/Godeps/_workspace/src/github.com/hashicorp/raft/config.go new file mode 100644 index 0000000000000..047a88abe766d --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/config.go @@ -0,0 +1,125 @@ +package raft + +import ( + "fmt" + "io" + "log" + "time" +) + +// Config provides any necessary configuration to +// the Raft server +type Config struct { + // Time in follower state without a leader before we attempt an election. + HeartbeatTimeout time.Duration + + // Time in candidate state without a leader before we attempt an election. + ElectionTimeout time.Duration + + // Time without an Apply() operation before we heartbeat to ensure + // a timely commit. Due to random staggering, may be delayed as much as + // 2x this value. + CommitTimeout time.Duration + + // MaxAppendEntries controls the maximum number of append entries + // to send at once. We want to strike a balance between efficiency + // and avoiding waste if the follower is going to reject because of + // an inconsistent log. + MaxAppendEntries int + + // If we are a member of a cluster, and RemovePeer is invoked for the + // local node, then we forget all peers and transition into the follower state. + // If ShutdownOnRemove is is set, we additional shutdown Raft. Otherwise, + // we can become a leader of a cluster containing only this node. + ShutdownOnRemove bool + + // DisableBootstrapAfterElect is used to turn off EnableSingleNode + // after the node is elected. This is used to prevent self-election + // if the node is removed from the Raft cluster via RemovePeer. Setting + // it to false will keep the bootstrap mode, allowing the node to self-elect + // and potentially bootstrap a separate cluster. + DisableBootstrapAfterElect bool + + // TrailingLogs controls how many logs we leave after a snapshot. This is + // used so that we can quickly replay logs on a follower instead of being + // forced to send an entire snapshot. + TrailingLogs uint64 + + // SnapshotInterval controls how often we check if we should perform a snapshot. + // We randomly stagger between this value and 2x this value to avoid the entire + // cluster from performing a snapshot at once. + SnapshotInterval time.Duration + + // SnapshotThreshold controls how many outstanding logs there must be before + // we perform a snapshot. This is to prevent excessive snapshots when we can + // just replay a small set of logs. + SnapshotThreshold uint64 + + // EnableSingleNode allows for a single node mode of operation. This + // is false by default, which prevents a lone node from electing itself. + // leader. + EnableSingleNode bool + + // LeaderLeaseTimeout is used to control how long the "lease" lasts + // for being the leader without being able to contact a quorum + // of nodes. If we reach this interval without contact, we will + // step down as leader. + LeaderLeaseTimeout time.Duration + + // LogOutput is used as a sink for logs, unless Logger is specified. + // Defaults to os.Stderr. + LogOutput io.Writer + + // Logger is a user-provided logger. If nil, a logger writing to LogOutput + // is used. + Logger *log.Logger +} + +// DefaultConfig returns a Config with usable defaults. +func DefaultConfig() *Config { + return &Config{ + HeartbeatTimeout: 1000 * time.Millisecond, + ElectionTimeout: 1000 * time.Millisecond, + CommitTimeout: 50 * time.Millisecond, + MaxAppendEntries: 64, + ShutdownOnRemove: true, + DisableBootstrapAfterElect: true, + TrailingLogs: 10240, + SnapshotInterval: 120 * time.Second, + SnapshotThreshold: 8192, + EnableSingleNode: false, + LeaderLeaseTimeout: 500 * time.Millisecond, + } +} + +// ValidateConfig is used to validate a sane configuration +func ValidateConfig(config *Config) error { + if config.HeartbeatTimeout < 5*time.Millisecond { + return fmt.Errorf("Heartbeat timeout is too low") + } + if config.ElectionTimeout < 5*time.Millisecond { + return fmt.Errorf("Election timeout is too low") + } + if config.CommitTimeout < time.Millisecond { + return fmt.Errorf("Commit timeout is too low") + } + if config.MaxAppendEntries <= 0 { + return fmt.Errorf("MaxAppendEntries must be positive") + } + if config.MaxAppendEntries > 1024 { + return fmt.Errorf("MaxAppendEntries is too large") + } + if config.SnapshotInterval < 5*time.Millisecond { + return fmt.Errorf("Snapshot interval is too low") + } + if config.LeaderLeaseTimeout < 5*time.Millisecond { + return fmt.Errorf("Leader lease timeout is too low") + } + if config.LeaderLeaseTimeout > config.HeartbeatTimeout { + return fmt.Errorf("Leader lease timeout cannot be larger than heartbeat timeout") + } + if config.ElectionTimeout < config.HeartbeatTimeout { + return fmt.Errorf("Election timeout must be equal or greater than Heartbeat Timeout") + } + return nil +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/discard_snapshot.go b/Godeps/_workspace/src/github.com/hashicorp/raft/discard_snapshot.go new file mode 100644 index 0000000000000..1b4611d559f7c --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/discard_snapshot.go @@ -0,0 +1,48 @@ +package raft + +import ( + "fmt" + "io" +) + +// DiscardSnapshotStore is used to successfully snapshot while +// always discarding the snapshot. This is useful for when the +// log should be truncated but no snapshot should be retained. +// This should never be used for production use, and is only +// suitable for testing. +type DiscardSnapshotStore struct{} + +type DiscardSnapshotSink struct{} + +// NewDiscardSnapshotStore is used to create a new DiscardSnapshotStore. +func NewDiscardSnapshotStore() *DiscardSnapshotStore { + return &DiscardSnapshotStore{} +} + +func (d *DiscardSnapshotStore) Create(index, term uint64, peers []byte) (SnapshotSink, error) { + return &DiscardSnapshotSink{}, nil +} + +func (d *DiscardSnapshotStore) List() ([]*SnapshotMeta, error) { + return nil, nil +} + +func (d *DiscardSnapshotStore) Open(id string) (*SnapshotMeta, io.ReadCloser, error) { + return nil, nil, fmt.Errorf("open is not supported") +} + +func (d *DiscardSnapshotSink) Write(b []byte) (int, error) { + return len(b), nil +} + +func (d *DiscardSnapshotSink) Close() error { + return nil +} + +func (d *DiscardSnapshotSink) ID() string { + return "discard" +} + +func (d *DiscardSnapshotSink) Cancel() error { + return nil +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/discard_snapshot_test.go b/Godeps/_workspace/src/github.com/hashicorp/raft/discard_snapshot_test.go new file mode 100644 index 0000000000000..5abedfe2c6d59 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/discard_snapshot_test.go @@ -0,0 +1,17 @@ +package raft + +import "testing" + +func TestDiscardSnapshotStoreImpl(t *testing.T) { + var impl interface{} = &DiscardSnapshotStore{} + if _, ok := impl.(SnapshotStore); !ok { + t.Fatalf("DiscardSnapshotStore not a SnapshotStore") + } +} + +func TestDiscardSnapshotSinkImpl(t *testing.T) { + var impl interface{} = &DiscardSnapshotSink{} + if _, ok := impl.(SnapshotSink); !ok { + t.Fatalf("DiscardSnapshotSink not a SnapshotSink") + } +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/file_snapshot.go b/Godeps/_workspace/src/github.com/hashicorp/raft/file_snapshot.go new file mode 100644 index 0000000000000..bda3d6d8ecb20 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/file_snapshot.go @@ -0,0 +1,460 @@ +package raft + +import ( + "bufio" + "bytes" + "encoding/json" + "fmt" + "hash" + "hash/crc64" + "io" + "io/ioutil" + "log" + "os" + "path/filepath" + "sort" + "strings" + "time" +) + +const ( + testPath = "permTest" + snapPath = "snapshots" + metaFilePath = "meta.json" + stateFilePath = "state.bin" + tmpSuffix = ".tmp" +) + +// FileSnapshotStore implements the SnapshotStore interface and allows +// snapshots to be made on the local disk. +type FileSnapshotStore struct { + path string + retain int + logger *log.Logger +} + +type snapMetaSlice []*fileSnapshotMeta + +// FileSnapshotSink implements SnapshotSink with a file. +type FileSnapshotSink struct { + store *FileSnapshotStore + logger *log.Logger + dir string + meta fileSnapshotMeta + + stateFile *os.File + stateHash hash.Hash64 + buffered *bufio.Writer + + closed bool +} + +// fileSnapshotMeta is stored on disk. We also put a CRC +// on disk so that we can verify the snapshot. +type fileSnapshotMeta struct { + SnapshotMeta + CRC []byte +} + +// bufferedFile is returned when we open a snapshot. This way +// reads are buffered and the file still gets closed. +type bufferedFile struct { + bh *bufio.Reader + fh *os.File +} + +func (b *bufferedFile) Read(p []byte) (n int, err error) { + return b.bh.Read(p) +} + +func (b *bufferedFile) Close() error { + return b.fh.Close() +} + +// NewFileSnapshotStore creates a new FileSnapshotStore based +// on a base directory. The `retain` parameter controls how many +// snapshots are retained. Must be at least 1. +func NewFileSnapshotStore(base string, retain int, logOutput io.Writer) (*FileSnapshotStore, error) { + if retain < 1 { + return nil, fmt.Errorf("must retain at least one snapshot") + } + if logOutput == nil { + logOutput = os.Stderr + } + + // Ensure our path exists + path := filepath.Join(base, snapPath) + if err := os.MkdirAll(path, 0755); err != nil && !os.IsExist(err) { + return nil, fmt.Errorf("snapshot path not accessible: %v", err) + } + + // Setup the store + store := &FileSnapshotStore{ + path: path, + retain: retain, + logger: log.New(logOutput, "", log.LstdFlags), + } + + // Do a permissions test + if err := store.testPermissions(); err != nil { + return nil, fmt.Errorf("permissions test failed: %v", err) + } + return store, nil +} + +// testPermissions tries to touch a file in our path to see if it works. +func (f *FileSnapshotStore) testPermissions() error { + path := filepath.Join(f.path, testPath) + fh, err := os.Create(path) + if err != nil { + return err + } + fh.Close() + os.Remove(path) + return nil +} + +// snapshotName generates a name for the snapshot. +func snapshotName(term, index uint64) string { + now := time.Now() + msec := now.UnixNano() / int64(time.Millisecond) + return fmt.Sprintf("%d-%d-%d", term, index, msec) +} + +// Create is used to start a new snapshot +func (f *FileSnapshotStore) Create(index, term uint64, peers []byte) (SnapshotSink, error) { + // Create a new path + name := snapshotName(term, index) + path := filepath.Join(f.path, name+tmpSuffix) + f.logger.Printf("[INFO] snapshot: Creating new snapshot at %s", path) + + // Make the directory + if err := os.MkdirAll(path, 0755); err != nil { + f.logger.Printf("[ERR] snapshot: Failed to make snapshot directory: %v", err) + return nil, err + } + + // Create the sink + sink := &FileSnapshotSink{ + store: f, + logger: f.logger, + dir: path, + meta: fileSnapshotMeta{ + SnapshotMeta: SnapshotMeta{ + ID: name, + Index: index, + Term: term, + Peers: peers, + }, + CRC: nil, + }, + } + + // Write out the meta data + if err := sink.writeMeta(); err != nil { + f.logger.Printf("[ERR] snapshot: Failed to write metadata: %v", err) + return nil, err + } + + // Open the state file + statePath := filepath.Join(path, stateFilePath) + fh, err := os.Create(statePath) + if err != nil { + f.logger.Printf("[ERR] snapshot: Failed to create state file: %v", err) + return nil, err + } + sink.stateFile = fh + + // Create a CRC64 hash + sink.stateHash = crc64.New(crc64.MakeTable(crc64.ECMA)) + + // Wrap both the hash and file in a MultiWriter with buffering + multi := io.MultiWriter(sink.stateFile, sink.stateHash) + sink.buffered = bufio.NewWriter(multi) + + // Done + return sink, nil +} + +// List returns available snapshots in the store. +func (f *FileSnapshotStore) List() ([]*SnapshotMeta, error) { + // Get the eligible snapshots + snapshots, err := f.getSnapshots() + if err != nil { + f.logger.Printf("[ERR] snapshot: Failed to get snapshots: %v", err) + return nil, err + } + + var snapMeta []*SnapshotMeta + for _, meta := range snapshots { + snapMeta = append(snapMeta, &meta.SnapshotMeta) + if len(snapMeta) == f.retain { + break + } + } + return snapMeta, nil +} + +// getSnapshots returns all the known snapshots. +func (f *FileSnapshotStore) getSnapshots() ([]*fileSnapshotMeta, error) { + // Get the eligible snapshots + snapshots, err := ioutil.ReadDir(f.path) + if err != nil { + f.logger.Printf("[ERR] snapshot: Failed to scan snapshot dir: %v", err) + return nil, err + } + + // Populate the metadata + var snapMeta []*fileSnapshotMeta + for _, snap := range snapshots { + // Ignore any files + if !snap.IsDir() { + continue + } + + // Ignore any temporary snapshots + dirName := snap.Name() + if strings.HasSuffix(dirName, tmpSuffix) { + f.logger.Printf("[WARN] snapshot: Found temporary snapshot: %v", dirName) + continue + } + + // Try to read the meta data + meta, err := f.readMeta(dirName) + if err != nil { + f.logger.Printf("[WARN] snapshot: Failed to read metadata for %v: %v", dirName, err) + continue + } + + // Append, but only return up to the retain count + snapMeta = append(snapMeta, meta) + } + + // Sort the snapshot, reverse so we get new -> old + sort.Sort(sort.Reverse(snapMetaSlice(snapMeta))) + + return snapMeta, nil +} + +// readMeta is used to read the meta data for a given named backup +func (f *FileSnapshotStore) readMeta(name string) (*fileSnapshotMeta, error) { + // Open the meta file + metaPath := filepath.Join(f.path, name, metaFilePath) + fh, err := os.Open(metaPath) + if err != nil { + return nil, err + } + defer fh.Close() + + // Buffer the file IO + buffered := bufio.NewReader(fh) + + // Read in the JSON + meta := &fileSnapshotMeta{} + dec := json.NewDecoder(buffered) + if err := dec.Decode(meta); err != nil { + return nil, err + } + return meta, nil +} + +// Open takes a snapshot ID and returns a ReadCloser for that snapshot. +func (f *FileSnapshotStore) Open(id string) (*SnapshotMeta, io.ReadCloser, error) { + // Get the metadata + meta, err := f.readMeta(id) + if err != nil { + f.logger.Printf("[ERR] snapshot: Failed to get meta data to open snapshot: %v", err) + return nil, nil, err + } + + // Open the state file + statePath := filepath.Join(f.path, id, stateFilePath) + fh, err := os.Open(statePath) + if err != nil { + f.logger.Printf("[ERR] snapshot: Failed to open state file: %v", err) + return nil, nil, err + } + + // Create a CRC64 hash + stateHash := crc64.New(crc64.MakeTable(crc64.ECMA)) + + // Compute the hash + _, err = io.Copy(stateHash, fh) + if err != nil { + f.logger.Printf("[ERR] snapshot: Failed to read state file: %v", err) + fh.Close() + return nil, nil, err + } + + // Verify the hash + computed := stateHash.Sum(nil) + if bytes.Compare(meta.CRC, computed) != 0 { + f.logger.Printf("[ERR] snapshot: CRC checksum failed (stored: %v computed: %v)", + meta.CRC, computed) + fh.Close() + return nil, nil, fmt.Errorf("CRC mismatch") + } + + // Seek to the start + if _, err := fh.Seek(0, 0); err != nil { + f.logger.Printf("[ERR] snapshot: State file seek failed: %v", err) + fh.Close() + return nil, nil, err + } + + // Return a buffered file + buffered := &bufferedFile{ + bh: bufio.NewReader(fh), + fh: fh, + } + + return &meta.SnapshotMeta, buffered, nil +} + +// ReapSnapshots reaps any snapshots beyond the retain count. +func (f *FileSnapshotStore) ReapSnapshots() error { + snapshots, err := f.getSnapshots() + if err != nil { + f.logger.Printf("[ERR] snapshot: Failed to get snapshots: %v", err) + return err + } + + for i := f.retain; i < len(snapshots); i++ { + path := filepath.Join(f.path, snapshots[i].ID) + f.logger.Printf("[INFO] snapshot: reaping snapshot %v", path) + if err := os.RemoveAll(path); err != nil { + f.logger.Printf("[ERR] snapshot: Failed to reap snapshot %v: %v", path, err) + return err + } + } + return nil +} + +// ID returns the ID of the snapshot, can be used with Open() +// after the snapshot is finalized. +func (s *FileSnapshotSink) ID() string { + return s.meta.ID +} + +// Write is used to append to the state file. We write to the +// buffered IO object to reduce the amount of context switches. +func (s *FileSnapshotSink) Write(b []byte) (int, error) { + return s.buffered.Write(b) +} + +// Close is used to indicate a successful end. +func (s *FileSnapshotSink) Close() error { + // Make sure close is idempotent + if s.closed { + return nil + } + s.closed = true + + // Close the open handles + if err := s.finalize(); err != nil { + s.logger.Printf("[ERR] snapshot: Failed to finalize snapshot: %v", err) + return err + } + + // Write out the meta data + if err := s.writeMeta(); err != nil { + s.logger.Printf("[ERR] snapshot: Failed to write metadata: %v", err) + return err + } + + // Move the directory into place + newPath := strings.TrimSuffix(s.dir, tmpSuffix) + if err := os.Rename(s.dir, newPath); err != nil { + s.logger.Printf("[ERR] snapshot: Failed to move snapshot into place: %v", err) + return err + } + + // Reap any old snapshots + s.store.ReapSnapshots() + return nil +} + +// Cancel is used to indicate an unsuccessful end. +func (s *FileSnapshotSink) Cancel() error { + // Make sure close is idempotent + if s.closed { + return nil + } + s.closed = true + + // Close the open handles + if err := s.finalize(); err != nil { + s.logger.Printf("[ERR] snapshot: Failed to finalize snapshot: %v", err) + return err + } + + // Attempt to remove all artifacts + return os.RemoveAll(s.dir) +} + +// finalize is used to close all of our resources. +func (s *FileSnapshotSink) finalize() error { + // Flush any remaining data + if err := s.buffered.Flush(); err != nil { + return err + } + + // Get the file size + stat, statErr := s.stateFile.Stat() + + // Close the file + if err := s.stateFile.Close(); err != nil { + return err + } + + // Set the file size, check after we close + if statErr != nil { + return statErr + } + s.meta.Size = stat.Size() + + // Set the CRC + s.meta.CRC = s.stateHash.Sum(nil) + return nil +} + +// writeMeta is used to write out the metadata we have. +func (s *FileSnapshotSink) writeMeta() error { + // Open the meta file + metaPath := filepath.Join(s.dir, metaFilePath) + fh, err := os.Create(metaPath) + if err != nil { + return err + } + defer fh.Close() + + // Buffer the file IO + buffered := bufio.NewWriter(fh) + defer buffered.Flush() + + // Write out as JSON + enc := json.NewEncoder(buffered) + if err := enc.Encode(&s.meta); err != nil { + return err + } + return nil +} + +// Implement the sort interface for []*fileSnapshotMeta. +func (s snapMetaSlice) Len() int { + return len(s) +} + +func (s snapMetaSlice) Less(i, j int) bool { + if s[i].Term != s[j].Term { + return s[i].Term < s[j].Term + } + if s[i].Index != s[j].Index { + return s[i].Index < s[j].Index + } + return s[i].ID < s[j].ID +} + +func (s snapMetaSlice) Swap(i, j int) { + s[i], s[j] = s[j], s[i] +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/file_snapshot_test.go b/Godeps/_workspace/src/github.com/hashicorp/raft/file_snapshot_test.go new file mode 100644 index 0000000000000..7620c19384fcd --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/file_snapshot_test.go @@ -0,0 +1,343 @@ +package raft + +import ( + "bytes" + "io" + "io/ioutil" + "os" + "runtime" + "testing" +) + +func FileSnapTest(t *testing.T) (string, *FileSnapshotStore) { + // Create a test dir + dir, err := ioutil.TempDir("", "raft") + if err != nil { + t.Fatalf("err: %v ", err) + } + + snap, err := NewFileSnapshotStore(dir, 3, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + return dir, snap +} + +func TestFileSnapshotStoreImpl(t *testing.T) { + var impl interface{} = &FileSnapshotStore{} + if _, ok := impl.(SnapshotStore); !ok { + t.Fatalf("FileSnapshotStore not a SnapshotStore") + } +} + +func TestFileSnapshotSinkImpl(t *testing.T) { + var impl interface{} = &FileSnapshotSink{} + if _, ok := impl.(SnapshotSink); !ok { + t.Fatalf("FileSnapshotSink not a SnapshotSink") + } +} + +func TestFileSS_CreateSnapshotMissingParentDir(t *testing.T) { + parent, err := ioutil.TempDir("", "raft") + if err != nil { + t.Fatalf("err: %v ", err) + } + defer os.RemoveAll(parent) + + dir, err := ioutil.TempDir(parent, "raft") + if err != nil { + t.Fatalf("err: %v ", err) + } + + snap, err := NewFileSnapshotStore(dir, 3, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + + os.RemoveAll(parent) + peers := []byte("all my lovely friends") + _, err = snap.Create(10, 3, peers) + if err != nil { + t.Fatalf("should not fail when using non existing parent") + } + +} +func TestFileSS_CreateSnapshot(t *testing.T) { + // Create a test dir + dir, err := ioutil.TempDir("", "raft") + if err != nil { + t.Fatalf("err: %v ", err) + } + defer os.RemoveAll(dir) + + snap, err := NewFileSnapshotStore(dir, 3, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + + // Check no snapshots + snaps, err := snap.List() + if err != nil { + t.Fatalf("err: %v", err) + } + if len(snaps) != 0 { + t.Fatalf("did not expect any snapshots: %v", snaps) + } + + // Create a new sink + peers := []byte("all my lovely friends") + sink, err := snap.Create(10, 3, peers) + if err != nil { + t.Fatalf("err: %v", err) + } + + // The sink is not done, should not be in a list! + snaps, err = snap.List() + if err != nil { + t.Fatalf("err: %v", err) + } + if len(snaps) != 0 { + t.Fatalf("did not expect any snapshots: %v", snaps) + } + + // Write to the sink + _, err = sink.Write([]byte("first\n")) + if err != nil { + t.Fatalf("err: %v", err) + } + _, err = sink.Write([]byte("second\n")) + if err != nil { + t.Fatalf("err: %v", err) + } + + // Done! + err = sink.Close() + if err != nil { + t.Fatalf("err: %v", err) + } + + // Should have a snapshot! + snaps, err = snap.List() + if err != nil { + t.Fatalf("err: %v", err) + } + if len(snaps) != 1 { + t.Fatalf("expect a snapshots: %v", snaps) + } + + // Check the latest + latest := snaps[0] + if latest.Index != 10 { + t.Fatalf("bad snapshot: %v", *latest) + } + if latest.Term != 3 { + t.Fatalf("bad snapshot: %v", *latest) + } + if bytes.Compare(latest.Peers, peers) != 0 { + t.Fatalf("bad snapshot: %v", *latest) + } + if latest.Size != 13 { + t.Fatalf("bad snapshot: %v", *latest) + } + + // Read the snapshot + _, r, err := snap.Open(latest.ID) + if err != nil { + t.Fatalf("err: %v", err) + } + + // Read out everything + var buf bytes.Buffer + if _, err := io.Copy(&buf, r); err != nil { + t.Fatalf("err: %v", err) + } + if err := r.Close(); err != nil { + t.Fatalf("err: %v", err) + } + + // Ensure a match + if bytes.Compare(buf.Bytes(), []byte("first\nsecond\n")) != 0 { + t.Fatalf("content mismatch") + } +} + +func TestFileSS_CancelSnapshot(t *testing.T) { + // Create a test dir + dir, err := ioutil.TempDir("", "raft") + if err != nil { + t.Fatalf("err: %v ", err) + } + defer os.RemoveAll(dir) + + snap, err := NewFileSnapshotStore(dir, 3, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + + // Create a new sink + peers := []byte("all my lovely friends") + sink, err := snap.Create(10, 3, peers) + if err != nil { + t.Fatalf("err: %v", err) + } + + // Cancel the snapshot! Should delete + err = sink.Cancel() + if err != nil { + t.Fatalf("err: %v", err) + } + + // The sink is canceled, should not be in a list! + snaps, err := snap.List() + if err != nil { + t.Fatalf("err: %v", err) + } + if len(snaps) != 0 { + t.Fatalf("did not expect any snapshots: %v", snaps) + } +} + +func TestFileSS_Retention(t *testing.T) { + // Create a test dir + dir, err := ioutil.TempDir("", "raft") + if err != nil { + t.Fatalf("err: %v ", err) + } + defer os.RemoveAll(dir) + + snap, err := NewFileSnapshotStore(dir, 2, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + + // Create a new sink + peers := []byte("all my lovely friends") + + // Create a few snapshots + for i := 10; i < 15; i++ { + sink, err := snap.Create(uint64(i), 3, peers) + if err != nil { + t.Fatalf("err: %v", err) + } + err = sink.Close() + if err != nil { + t.Fatalf("err: %v", err) + } + } + + // Should only have 2 listed! + snaps, err := snap.List() + if err != nil { + t.Fatalf("err: %v", err) + } + if len(snaps) != 2 { + t.Fatalf("expect 2 snapshots: %v", snaps) + } + + // Check they are the latest + if snaps[0].Index != 14 { + t.Fatalf("bad snap: %#v", *snaps[0]) + } + if snaps[1].Index != 13 { + t.Fatalf("bad snap: %#v", *snaps[1]) + } +} + +func TestFileSS_BadPerm(t *testing.T) { + if runtime.GOOS == "windows" { + t.Skip("skipping file permission test on windows") + } + + // Create a temp dir + dir1, err := ioutil.TempDir("", "raft") + if err != nil { + t.Fatalf("err: %s", err) + } + defer os.RemoveAll(dir1) + + // Create a sub dir and remove all permissions + dir2, err := ioutil.TempDir(dir1, "badperm") + if err != nil { + t.Fatalf("err: %s", err) + } + if err := os.Chmod(dir2, 000); err != nil { + t.Fatalf("err: %s", err) + } + defer os.Chmod(dir2, 777) // Set perms back for delete + + // Should fail + if _, err := NewFileSnapshotStore(dir2, 3, nil); err == nil { + t.Fatalf("should fail to use dir with bad perms") + } +} + +func TestFileSS_MissingParentDir(t *testing.T) { + parent, err := ioutil.TempDir("", "raft") + if err != nil { + t.Fatalf("err: %v ", err) + } + defer os.RemoveAll(parent) + + dir, err := ioutil.TempDir(parent, "raft") + if err != nil { + t.Fatalf("err: %v ", err) + } + + os.RemoveAll(parent) + _, err = NewFileSnapshotStore(dir, 3, nil) + if err != nil { + t.Fatalf("should not fail when using non existing parent") + } +} + +func TestFileSS_Ordering(t *testing.T) { + // Create a test dir + dir, err := ioutil.TempDir("", "raft") + if err != nil { + t.Fatalf("err: %v ", err) + } + defer os.RemoveAll(dir) + + snap, err := NewFileSnapshotStore(dir, 3, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + + // Create a new sink + peers := []byte("all my lovely friends") + + sink, err := snap.Create(130350, 5, peers) + if err != nil { + t.Fatalf("err: %v", err) + } + err = sink.Close() + if err != nil { + t.Fatalf("err: %v", err) + } + + sink, err = snap.Create(204917, 36, peers) + if err != nil { + t.Fatalf("err: %v", err) + } + err = sink.Close() + if err != nil { + t.Fatalf("err: %v", err) + } + + // Should only have 2 listed! + snaps, err := snap.List() + if err != nil { + t.Fatalf("err: %v", err) + } + if len(snaps) != 2 { + t.Fatalf("expect 2 snapshots: %v", snaps) + } + + // Check they are ordered + if snaps[0].Term != 36 { + t.Fatalf("bad snap: %#v", *snaps[0]) + } + if snaps[1].Term != 5 { + t.Fatalf("bad snap: %#v", *snaps[1]) + } +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/fsm.go b/Godeps/_workspace/src/github.com/hashicorp/raft/fsm.go new file mode 100644 index 0000000000000..ea8ab548dbcb0 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/fsm.go @@ -0,0 +1,37 @@ +package raft + +import ( + "io" +) + +// FSM provides an interface that can be implemented by +// clients to make use of the replicated log. +type FSM interface { + // Apply log is invoked once a log entry is committed. + Apply(*Log) interface{} + + // Snapshot is used to support log compaction. This call should + // return an FSMSnapshot which can be used to save a point-in-time + // snapshot of the FSM. Apply and Snapshot are not called in multiple + // threads, but Apply will be called concurrently with Persist. This means + // the FSM should be implemented in a fashion that allows for concurrent + // updates while a snapshot is happening. + Snapshot() (FSMSnapshot, error) + + // Restore is used to restore an FSM from a snapshot. It is not called + // concurrently with any other command. The FSM must discard all previous + // state. + Restore(io.ReadCloser) error +} + +// FSMSnapshot is returned by an FSM in response to a Snapshot +// It must be safe to invoke FSMSnapshot methods with concurrent +// calls to Apply. +type FSMSnapshot interface { + // Persist should dump all necessary state to the WriteCloser 'sink', + // and call sink.Close() when finished or call sink.Cancel() on error. + Persist(sink SnapshotSink) error + + // Release is invoked when we are finished with the snapshot. + Release() +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/future.go b/Godeps/_workspace/src/github.com/hashicorp/raft/future.go new file mode 100644 index 0000000000000..854e1ac927be5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/future.go @@ -0,0 +1,182 @@ +package raft + +import ( + "sync" + "time" +) + +// Future is used to represent an action that may occur in the future. +type Future interface { + Error() error +} + +// ApplyFuture is used for Apply() and can returns the FSM response. +type ApplyFuture interface { + Future + Response() interface{} + Index() uint64 +} + +// errorFuture is used to return a static error. +type errorFuture struct { + err error +} + +func (e errorFuture) Error() error { + return e.err +} + +func (e errorFuture) Response() interface{} { + return nil +} + +func (e errorFuture) Index() uint64 { + return 0 +} + +// deferError can be embedded to allow a future +// to provide an error in the future. +type deferError struct { + err error + errCh chan error + responded bool +} + +func (d *deferError) init() { + d.errCh = make(chan error, 1) +} + +func (d *deferError) Error() error { + if d.err != nil { + return d.err + } + if d.errCh == nil { + panic("waiting for response on nil channel") + } + d.err = <-d.errCh + return d.err +} + +func (d *deferError) respond(err error) { + if d.errCh == nil { + return + } + if d.responded { + return + } + d.errCh <- err + close(d.errCh) + d.responded = true +} + +// logFuture is used to apply a log entry and waits until +// the log is considered committed. +type logFuture struct { + deferError + log Log + policy quorumPolicy + response interface{} + dispatch time.Time +} + +func (l *logFuture) Response() interface{} { + return l.response +} + +func (l *logFuture) Index() uint64 { + return l.log.Index +} + +type peerFuture struct { + deferError + peers []string +} + +type shutdownFuture struct { + raft *Raft +} + +func (s *shutdownFuture) Error() error { + for s.raft.getRoutines() > 0 { + time.Sleep(5 * time.Millisecond) + } + return nil +} + +// snapshotFuture is used for waiting on a snapshot to complete. +type snapshotFuture struct { + deferError +} + +// reqSnapshotFuture is used for requesting a snapshot start. +// It is only used internally. +type reqSnapshotFuture struct { + deferError + + // snapshot details provided by the FSM runner before responding + index uint64 + term uint64 + peers []string + snapshot FSMSnapshot +} + +// restoreFuture is used for requesting an FSM to perform a +// snapshot restore. Used internally only. +type restoreFuture struct { + deferError + ID string +} + +// verifyFuture is used to verify the current node is still +// the leader. This is to prevent a stale read. +type verifyFuture struct { + deferError + notifyCh chan *verifyFuture + quorumSize int + votes int + voteLock sync.Mutex +} + +// vote is used to respond to a verifyFuture. +// This may block when responding on the notifyCh. +func (v *verifyFuture) vote(leader bool) { + v.voteLock.Lock() + defer v.voteLock.Unlock() + + // Guard against having notified already + if v.notifyCh == nil { + return + } + + if leader { + v.votes++ + if v.votes >= v.quorumSize { + v.notifyCh <- v + v.notifyCh = nil + } + } else { + v.notifyCh <- v + v.notifyCh = nil + } +} + +// appendFuture is used for waiting on a pipelined append +// entries RPC. +type appendFuture struct { + deferError + start time.Time + args *AppendEntriesRequest + resp *AppendEntriesResponse +} + +func (a *appendFuture) Start() time.Time { + return a.start +} + +func (a *appendFuture) Request() *AppendEntriesRequest { + return a.args +} + +func (a *appendFuture) Response() *AppendEntriesResponse { + return a.resp +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/inflight.go b/Godeps/_workspace/src/github.com/hashicorp/raft/inflight.go new file mode 100644 index 0000000000000..7014ff50394dd --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/inflight.go @@ -0,0 +1,213 @@ +package raft + +import ( + "container/list" + "sync" +) + +// QuorumPolicy allows individual logFutures to have different +// commitment rules while still using the inflight mechanism. +type quorumPolicy interface { + // Checks if a commit from a given peer is enough to + // satisfy the commitment rules + Commit() bool + + // Checks if a commit is committed + IsCommitted() bool +} + +// MajorityQuorum is used by Apply transactions and requires +// a simple majority of nodes. +type majorityQuorum struct { + count int + votesNeeded int +} + +func newMajorityQuorum(clusterSize int) *majorityQuorum { + votesNeeded := (clusterSize / 2) + 1 + return &majorityQuorum{count: 0, votesNeeded: votesNeeded} +} + +func (m *majorityQuorum) Commit() bool { + m.count++ + return m.count >= m.votesNeeded +} + +func (m *majorityQuorum) IsCommitted() bool { + return m.count >= m.votesNeeded +} + +// Inflight is used to track operations that are still in-flight. +type inflight struct { + sync.Mutex + committed *list.List + commitCh chan struct{} + minCommit uint64 + maxCommit uint64 + operations map[uint64]*logFuture + stopCh chan struct{} +} + +// NewInflight returns an inflight struct that notifies +// the provided channel when logs are finished committing. +func newInflight(commitCh chan struct{}) *inflight { + return &inflight{ + committed: list.New(), + commitCh: commitCh, + minCommit: 0, + maxCommit: 0, + operations: make(map[uint64]*logFuture), + stopCh: make(chan struct{}), + } +} + +// Start is used to mark a logFuture as being inflight. It +// also commits the entry, as it is assumed the leader is +// starting. +func (i *inflight) Start(l *logFuture) { + i.Lock() + defer i.Unlock() + i.start(l) +} + +// StartAll is used to mark a list of logFuture's as being +// inflight. It also commits each entry as the leader is +// assumed to be starting. +func (i *inflight) StartAll(logs []*logFuture) { + i.Lock() + defer i.Unlock() + for _, l := range logs { + i.start(l) + } +} + +// start is used to mark a single entry as inflight, +// must be invoked with the lock held. +func (i *inflight) start(l *logFuture) { + idx := l.log.Index + i.operations[idx] = l + + if idx > i.maxCommit { + i.maxCommit = idx + } + if i.minCommit == 0 { + i.minCommit = idx + } + i.commit(idx) +} + +// Cancel is used to cancel all in-flight operations. +// This is done when the leader steps down, and all futures +// are sent the given error. +func (i *inflight) Cancel(err error) { + // Close the channel first to unblock any pending commits + close(i.stopCh) + + // Lock after close to avoid deadlock + i.Lock() + defer i.Unlock() + + // Respond to all inflight operations + for _, op := range i.operations { + op.respond(err) + } + + // Clear all the committed but not processed + for e := i.committed.Front(); e != nil; e = e.Next() { + e.Value.(*logFuture).respond(err) + } + + // Clear the map + i.operations = make(map[uint64]*logFuture) + + // Clear the list of committed + i.committed = list.New() + + // Close the commmitCh + close(i.commitCh) + + // Reset indexes + i.minCommit = 0 + i.maxCommit = 0 +} + +// Committed returns all the committed operations in order. +func (i *inflight) Committed() (l *list.List) { + i.Lock() + l, i.committed = i.committed, list.New() + i.Unlock() + return l +} + +// Commit is used by leader replication routines to indicate that +// a follower was finished committing a log to disk. +func (i *inflight) Commit(index uint64) { + i.Lock() + defer i.Unlock() + i.commit(index) +} + +// CommitRange is used to commit a range of indexes inclusively. +// It is optimized to avoid commits for indexes that are not tracked. +func (i *inflight) CommitRange(minIndex, maxIndex uint64) { + i.Lock() + defer i.Unlock() + + // Update the minimum index + minIndex = max(i.minCommit, minIndex) + + // Commit each index + for idx := minIndex; idx <= maxIndex; idx++ { + i.commit(idx) + } +} + +// commit is used to commit a single index. Must be called with the lock held. +func (i *inflight) commit(index uint64) { + op, ok := i.operations[index] + if !ok { + // Ignore if not in the map, as it may be committed already + return + } + + // Check if we've satisfied the commit + if !op.policy.Commit() { + return + } + + // Cannot commit if this is not the minimum inflight. This can happen + // if the quorum size changes, meaning a previous commit requires a larger + // quorum that this commit. We MUST block until the previous log is committed, + // otherwise logs will be applied out of order. + if index != i.minCommit { + return + } + +NOTIFY: + // Add the operation to the committed list + i.committed.PushBack(op) + + // Stop tracking since it is committed + delete(i.operations, index) + + // Update the indexes + if index == i.maxCommit { + i.minCommit = 0 + i.maxCommit = 0 + + } else { + i.minCommit++ + } + + // Check if the next in-flight operation is ready + if i.minCommit != 0 { + op = i.operations[i.minCommit] + if op.policy.IsCommitted() { + index = i.minCommit + goto NOTIFY + } + } + + // Async notify of ready operations + asyncNotifyCh(i.commitCh) +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/inflight_test.go b/Godeps/_workspace/src/github.com/hashicorp/raft/inflight_test.go new file mode 100644 index 0000000000000..a9f57d6ead756 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/inflight_test.go @@ -0,0 +1,150 @@ +package raft + +import ( + "fmt" + "testing" +) + +func TestInflight_StartCommit(t *testing.T) { + commitCh := make(chan struct{}, 1) + in := newInflight(commitCh) + + // Commit a transaction as being in flight + l := &logFuture{log: Log{Index: 1}} + l.policy = newMajorityQuorum(5) + in.Start(l) + + // Commit 3 times + in.Commit(1) + if in.Committed().Len() != 0 { + t.Fatalf("should not be commited") + } + + in.Commit(1) + if in.Committed().Len() != 1 { + t.Fatalf("should be commited") + } + + // Already committed but should work anyways + in.Commit(1) +} + +func TestInflight_Cancel(t *testing.T) { + commitCh := make(chan struct{}, 1) + in := newInflight(commitCh) + + // Commit a transaction as being in flight + l := &logFuture{ + log: Log{Index: 1}, + } + l.init() + l.policy = newMajorityQuorum(3) + in.Start(l) + + // Cancel with an error + err := fmt.Errorf("error 1") + in.Cancel(err) + + // Should get an error return + if l.Error() != err { + t.Fatalf("expected error") + } +} + +func TestInflight_StartAll(t *testing.T) { + commitCh := make(chan struct{}, 1) + in := newInflight(commitCh) + + // Commit a few transaction as being in flight + l1 := &logFuture{log: Log{Index: 2}} + l1.policy = newMajorityQuorum(5) + l2 := &logFuture{log: Log{Index: 3}} + l2.policy = newMajorityQuorum(5) + l3 := &logFuture{log: Log{Index: 4}} + l3.policy = newMajorityQuorum(5) + + // Start all the entries + in.StartAll([]*logFuture{l1, l2, l3}) + + // Commit ranges + in.CommitRange(1, 5) + in.CommitRange(1, 4) + in.CommitRange(1, 10) + + // Should get 3 back + if in.Committed().Len() != 3 { + t.Fatalf("expected all 3 to commit") + } +} + +func TestInflight_CommitRange(t *testing.T) { + commitCh := make(chan struct{}, 1) + in := newInflight(commitCh) + + // Commit a few transaction as being in flight + l1 := &logFuture{log: Log{Index: 2}} + l1.policy = newMajorityQuorum(5) + in.Start(l1) + + l2 := &logFuture{log: Log{Index: 3}} + l2.policy = newMajorityQuorum(5) + in.Start(l2) + + l3 := &logFuture{log: Log{Index: 4}} + l3.policy = newMajorityQuorum(5) + in.Start(l3) + + // Commit ranges + in.CommitRange(1, 5) + in.CommitRange(1, 4) + in.CommitRange(1, 10) + + // Should get 3 back + if in.Committed().Len() != 3 { + t.Fatalf("expected all 3 to commit") + } +} + +// Should panic if we commit non contiguously! +func TestInflight_NonContiguous(t *testing.T) { + commitCh := make(chan struct{}, 1) + in := newInflight(commitCh) + + // Commit a few transaction as being in flight + l1 := &logFuture{log: Log{Index: 2}} + l1.policy = newMajorityQuorum(5) + in.Start(l1) + + l2 := &logFuture{log: Log{Index: 3}} + l2.policy = newMajorityQuorum(5) + in.Start(l2) + + in.Commit(3) + in.Commit(3) + in.Commit(3) // panic! + + if in.Committed().Len() != 0 { + t.Fatalf("should not commit") + } + + in.Commit(2) + in.Commit(2) + in.Commit(2) // panic! + + committed := in.Committed() + if committed.Len() != 2 { + t.Fatalf("should commit both") + } + + current := committed.Front() + l := current.Value.(*logFuture) + if l.log.Index != 2 { + t.Fatalf("bad: %v", *l) + } + + current = current.Next() + l = current.Value.(*logFuture) + if l.log.Index != 3 { + t.Fatalf("bad: %v", *l) + } +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/inmem_store.go b/Godeps/_workspace/src/github.com/hashicorp/raft/inmem_store.go new file mode 100644 index 0000000000000..6e4dfd020f7b1 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/inmem_store.go @@ -0,0 +1,116 @@ +package raft + +import ( + "sync" +) + +// InmemStore implements the LogStore and StableStore interface. +// It should NOT EVER be used for production. It is used only for +// unit tests. Use the MDBStore implementation instead. +type InmemStore struct { + l sync.RWMutex + lowIndex uint64 + highIndex uint64 + logs map[uint64]*Log + kv map[string][]byte + kvInt map[string]uint64 +} + +// NewInmemStore returns a new in-memory backend. Do not ever +// use for production. Only for testing. +func NewInmemStore() *InmemStore { + i := &InmemStore{ + logs: make(map[uint64]*Log), + kv: make(map[string][]byte), + kvInt: make(map[string]uint64), + } + return i +} + +// FirstIndex implements the LogStore interface. +func (i *InmemStore) FirstIndex() (uint64, error) { + i.l.RLock() + defer i.l.RUnlock() + return i.lowIndex, nil +} + +// LastIndex implements the LogStore interface. +func (i *InmemStore) LastIndex() (uint64, error) { + i.l.RLock() + defer i.l.RUnlock() + return i.highIndex, nil +} + +// GetLog implements the LogStore interface. +func (i *InmemStore) GetLog(index uint64, log *Log) error { + i.l.RLock() + defer i.l.RUnlock() + l, ok := i.logs[index] + if !ok { + return ErrLogNotFound + } + *log = *l + return nil +} + +// StoreLog implements the LogStore interface. +func (i *InmemStore) StoreLog(log *Log) error { + return i.StoreLogs([]*Log{log}) +} + +// StoreLogs implements the LogStore interface. +func (i *InmemStore) StoreLogs(logs []*Log) error { + i.l.Lock() + defer i.l.Unlock() + for _, l := range logs { + i.logs[l.Index] = l + if i.lowIndex == 0 { + i.lowIndex = l.Index + } + if l.Index > i.highIndex { + i.highIndex = l.Index + } + } + return nil +} + +// DeleteRange implements the LogStore interface. +func (i *InmemStore) DeleteRange(min, max uint64) error { + i.l.Lock() + defer i.l.Unlock() + for j := min; j <= max; j++ { + delete(i.logs, j) + } + i.lowIndex = max + 1 + return nil +} + +// Set implements the StableStore interface. +func (i *InmemStore) Set(key []byte, val []byte) error { + i.l.Lock() + defer i.l.Unlock() + i.kv[string(key)] = val + return nil +} + +// Get implements the StableStore interface. +func (i *InmemStore) Get(key []byte) ([]byte, error) { + i.l.RLock() + defer i.l.RUnlock() + return i.kv[string(key)], nil +} + +// SetUint64 implements the StableStore interface. +func (i *InmemStore) SetUint64(key []byte, val uint64) error { + i.l.Lock() + defer i.l.Unlock() + i.kvInt[string(key)] = val + return nil +} + +// GetUint64 implements the StableStore interface. +func (i *InmemStore) GetUint64(key []byte) (uint64, error) { + i.l.RLock() + defer i.l.RUnlock() + return i.kvInt[string(key)], nil +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/inmem_transport.go b/Godeps/_workspace/src/github.com/hashicorp/raft/inmem_transport.go new file mode 100644 index 0000000000000..994d06d8fad65 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/inmem_transport.go @@ -0,0 +1,315 @@ +package raft + +import ( + "fmt" + "io" + "sync" + "time" +) + +// NewInmemAddr returns a new in-memory addr with +// a randomly generate UUID as the ID. +func NewInmemAddr() string { + return generateUUID() +} + +// inmemPipeline is used to pipeline requests for the in-mem transport. +type inmemPipeline struct { + trans *InmemTransport + peer *InmemTransport + peerAddr string + + doneCh chan AppendFuture + inprogressCh chan *inmemPipelineInflight + + shutdown bool + shutdownCh chan struct{} + shutdownLock sync.Mutex +} + +type inmemPipelineInflight struct { + future *appendFuture + respCh <-chan RPCResponse +} + +// InmemTransport Implements the Transport interface, to allow Raft to be +// tested in-memory without going over a network. +type InmemTransport struct { + sync.RWMutex + consumerCh chan RPC + localAddr string + peers map[string]*InmemTransport + pipelines []*inmemPipeline + timeout time.Duration +} + +// NewInmemTransport is used to initialize a new transport +// and generates a random local address. +func NewInmemTransport() (string, *InmemTransport) { + addr := NewInmemAddr() + trans := &InmemTransport{ + consumerCh: make(chan RPC, 16), + localAddr: addr, + peers: make(map[string]*InmemTransport), + timeout: 50 * time.Millisecond, + } + return addr, trans +} + +// SetHeartbeatHandler is used to set optional fast-path for +// heartbeats, not supported for this transport. +func (i *InmemTransport) SetHeartbeatHandler(cb func(RPC)) { +} + +// Consumer implements the Transport interface. +func (i *InmemTransport) Consumer() <-chan RPC { + return i.consumerCh +} + +// LocalAddr implements the Transport interface. +func (i *InmemTransport) LocalAddr() string { + return i.localAddr +} + +// AppendEntriesPipeline returns an interface that can be used to pipeline +// AppendEntries requests. +func (i *InmemTransport) AppendEntriesPipeline(target string) (AppendPipeline, error) { + i.RLock() + peer, ok := i.peers[target] + i.RUnlock() + if !ok { + return nil, fmt.Errorf("failed to connect to peer: %v", target) + } + pipeline := newInmemPipeline(i, peer, target) + i.Lock() + i.pipelines = append(i.pipelines, pipeline) + i.Unlock() + return pipeline, nil +} + +// AppendEntries implements the Transport interface. +func (i *InmemTransport) AppendEntries(target string, args *AppendEntriesRequest, resp *AppendEntriesResponse) error { + rpcResp, err := i.makeRPC(target, args, nil, i.timeout) + if err != nil { + return err + } + + // Copy the result back + out := rpcResp.Response.(*AppendEntriesResponse) + *resp = *out + return nil +} + +// RequestVote implements the Transport interface. +func (i *InmemTransport) RequestVote(target string, args *RequestVoteRequest, resp *RequestVoteResponse) error { + rpcResp, err := i.makeRPC(target, args, nil, i.timeout) + if err != nil { + return err + } + + // Copy the result back + out := rpcResp.Response.(*RequestVoteResponse) + *resp = *out + return nil +} + +// InstallSnapshot implements the Transport interface. +func (i *InmemTransport) InstallSnapshot(target string, args *InstallSnapshotRequest, resp *InstallSnapshotResponse, data io.Reader) error { + rpcResp, err := i.makeRPC(target, args, data, 10*i.timeout) + if err != nil { + return err + } + + // Copy the result back + out := rpcResp.Response.(*InstallSnapshotResponse) + *resp = *out + return nil +} + +func (i *InmemTransport) makeRPC(target string, args interface{}, r io.Reader, timeout time.Duration) (rpcResp RPCResponse, err error) { + i.RLock() + peer, ok := i.peers[target] + i.RUnlock() + + if !ok { + err = fmt.Errorf("failed to connect to peer: %v", target) + return + } + + // Send the RPC over + respCh := make(chan RPCResponse) + peer.consumerCh <- RPC{ + Command: args, + Reader: r, + RespChan: respCh, + } + + // Wait for a response + select { + case rpcResp = <-respCh: + if rpcResp.Error != nil { + err = rpcResp.Error + } + case <-time.After(timeout): + err = fmt.Errorf("command timed out") + } + return +} + +// EncodePeer implements the Transport interface. It uses the UUID as the +// address directly. +func (i *InmemTransport) EncodePeer(p string) []byte { + return []byte(p) +} + +// DecodePeer implements the Transport interface. It wraps the UUID in an +// InmemAddr. +func (i *InmemTransport) DecodePeer(buf []byte) string { + return string(buf) +} + +// Connect is used to connect this transport to another transport for +// a given peer name. This allows for local routing. +func (i *InmemTransport) Connect(peer string, trans *InmemTransport) { + i.Lock() + defer i.Unlock() + i.peers[peer] = trans +} + +// Disconnect is used to remove the ability to route to a given peer. +func (i *InmemTransport) Disconnect(peer string) { + i.Lock() + defer i.Unlock() + delete(i.peers, peer) + + // Disconnect any pipelines + n := len(i.pipelines) + for idx := 0; idx < n; idx++ { + if i.pipelines[idx].peerAddr == peer { + i.pipelines[idx].Close() + i.pipelines[idx], i.pipelines[n-1] = i.pipelines[n-1], nil + idx-- + n-- + } + } + i.pipelines = i.pipelines[:n] +} + +// DisconnectAll is used to remove all routes to peers. +func (i *InmemTransport) DisconnectAll() { + i.Lock() + defer i.Unlock() + i.peers = make(map[string]*InmemTransport) + + // Handle pipelines + for _, pipeline := range i.pipelines { + pipeline.Close() + } + i.pipelines = nil +} + +func newInmemPipeline(trans *InmemTransport, peer *InmemTransport, addr string) *inmemPipeline { + i := &inmemPipeline{ + trans: trans, + peer: peer, + peerAddr: addr, + doneCh: make(chan AppendFuture, 16), + inprogressCh: make(chan *inmemPipelineInflight, 16), + shutdownCh: make(chan struct{}), + } + go i.decodeResponses() + return i +} + +func (i *inmemPipeline) decodeResponses() { + timeout := i.trans.timeout + for { + select { + case inp := <-i.inprogressCh: + var timeoutCh <-chan time.Time + if timeout > 0 { + timeoutCh = time.After(timeout) + } + + select { + case rpcResp := <-inp.respCh: + // Copy the result back + *inp.future.resp = *rpcResp.Response.(*AppendEntriesResponse) + inp.future.respond(rpcResp.Error) + + select { + case i.doneCh <- inp.future: + case <-i.shutdownCh: + return + } + + case <-timeoutCh: + inp.future.respond(fmt.Errorf("command timed out")) + select { + case i.doneCh <- inp.future: + case <-i.shutdownCh: + return + } + + case <-i.shutdownCh: + return + } + case <-i.shutdownCh: + return + } + } +} + +func (i *inmemPipeline) AppendEntries(args *AppendEntriesRequest, resp *AppendEntriesResponse) (AppendFuture, error) { + // Create a new future + future := &appendFuture{ + start: time.Now(), + args: args, + resp: resp, + } + future.init() + + // Handle a timeout + var timeout <-chan time.Time + if i.trans.timeout > 0 { + timeout = time.After(i.trans.timeout) + } + + // Send the RPC over + respCh := make(chan RPCResponse, 1) + rpc := RPC{ + Command: args, + RespChan: respCh, + } + select { + case i.peer.consumerCh <- rpc: + case <-timeout: + return nil, fmt.Errorf("command enqueue timeout") + case <-i.shutdownCh: + return nil, ErrPipelineShutdown + } + + // Send to be decoded + select { + case i.inprogressCh <- &inmemPipelineInflight{future, respCh}: + return future, nil + case <-i.shutdownCh: + return nil, ErrPipelineShutdown + } +} + +func (i *inmemPipeline) Consumer() <-chan AppendFuture { + return i.doneCh +} + +func (i *inmemPipeline) Close() error { + i.shutdownLock.Lock() + defer i.shutdownLock.Unlock() + if i.shutdown { + return nil + } + + i.shutdown = true + close(i.shutdownCh) + return nil +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/inmem_transport_test.go b/Godeps/_workspace/src/github.com/hashicorp/raft/inmem_transport_test.go new file mode 100644 index 0000000000000..2086a23897f2f --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/inmem_transport_test.go @@ -0,0 +1,12 @@ +package raft + +import ( + "testing" +) + +func TestInmemTransportImpl(t *testing.T) { + var inm interface{} = &InmemTransport{} + if _, ok := inm.(Transport); !ok { + t.Fatalf("InmemTransport is not a Transport") + } +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/integ_test.go b/Godeps/_workspace/src/github.com/hashicorp/raft/integ_test.go new file mode 100644 index 0000000000000..1d071e139a47e --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/integ_test.go @@ -0,0 +1,266 @@ +package raft + +import ( + "bytes" + "fmt" + "io/ioutil" + "log" + "os" + "testing" + "time" +) + +// CheckInteg will skip a test if integration testing is not enabled. +func CheckInteg(t *testing.T) { + if !IsInteg() { + t.SkipNow() + } +} + +// IsInteg returns a boolean telling you if we're in integ testing mode. +func IsInteg() bool { + return os.Getenv("INTEG_TESTS") != "" +} + +type RaftEnv struct { + dir string + conf *Config + fsm *MockFSM + store *InmemStore + snapshot *FileSnapshotStore + peers *JSONPeers + trans *NetworkTransport + raft *Raft +} + +func (r *RaftEnv) Release() { + log.Printf("[WARN] Release node at %v", r.raft.localAddr) + f := r.raft.Shutdown() + if err := f.Error(); err != nil { + panic(err) + } + r.trans.Close() + os.RemoveAll(r.dir) +} + +func MakeRaft(t *testing.T, conf *Config) *RaftEnv { + env := &RaftEnv{} + + // Set the config + if conf == nil { + conf = inmemConfig() + } + env.conf = conf + + dir, err := ioutil.TempDir("", "raft") + if err != nil { + t.Fatalf("err: %v ", err) + } + env.dir = dir + + stable := NewInmemStore() + env.store = stable + + snap, err := NewFileSnapshotStore(dir, 3, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + env.snapshot = snap + + env.fsm = &MockFSM{} + + trans, err := NewTCPTransport("127.0.0.1:0", nil, 2, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + env.trans = trans + + env.peers = NewJSONPeers(dir, trans) + + log.Printf("[INFO] Starting node at %v", trans.LocalAddr()) + raft, err := NewRaft(conf, env.fsm, stable, stable, snap, env.peers, trans) + if err != nil { + t.Fatalf("err: %v", err) + } + env.raft = raft + return env +} + +func WaitFor(env *RaftEnv, state RaftState) error { + limit := time.Now().Add(200 * time.Millisecond) + for env.raft.State() != state { + if time.Now().Before(limit) { + time.Sleep(10 * time.Millisecond) + } else { + return fmt.Errorf("failed to transition to state %v", state) + } + } + return nil +} + +func WaitForAny(state RaftState, envs []*RaftEnv) (*RaftEnv, error) { + limit := time.Now().Add(200 * time.Millisecond) +CHECK: + for _, env := range envs { + if env.raft.State() == state { + return env, nil + } + } + if time.Now().Before(limit) { + goto WAIT + } + return nil, fmt.Errorf("failed to find node in %v state", state) +WAIT: + time.Sleep(10 * time.Millisecond) + goto CHECK +} + +func WaitFuture(f Future, t *testing.T) error { + timer := time.AfterFunc(200*time.Millisecond, func() { + panic(fmt.Errorf("timeout waiting for future %v", f)) + }) + defer timer.Stop() + return f.Error() +} + +func NoErr(err error, t *testing.T) { + if err != nil { + t.Fatalf("err: %v", err) + } +} + +func CheckConsistent(envs []*RaftEnv, t *testing.T) { + limit := time.Now().Add(400 * time.Millisecond) + first := envs[0] + var err error +CHECK: + l1 := len(first.fsm.logs) + for i := 1; i < len(envs); i++ { + env := envs[i] + l2 := len(env.fsm.logs) + if l1 != l2 { + err = fmt.Errorf("log length mismatch %d %d", l1, l2) + goto ERR + } + for idx, log := range first.fsm.logs { + other := env.fsm.logs[idx] + if bytes.Compare(log, other) != 0 { + err = fmt.Errorf("log %d mismatch %v %v", idx, log, other) + goto ERR + } + } + } + return +ERR: + if time.Now().After(limit) { + t.Fatalf("%v", err) + } + time.Sleep(20 * time.Millisecond) + goto CHECK +} + +// Tests Raft by creating a cluster, growing it to 5 nodes while +// causing various stressful conditions +func TestRaft_Integ(t *testing.T) { + CheckInteg(t) + conf := DefaultConfig() + conf.HeartbeatTimeout = 50 * time.Millisecond + conf.ElectionTimeout = 50 * time.Millisecond + conf.LeaderLeaseTimeout = 50 * time.Millisecond + conf.CommitTimeout = 5 * time.Millisecond + conf.SnapshotThreshold = 100 + conf.TrailingLogs = 10 + conf.EnableSingleNode = true + + // Create a single node + env1 := MakeRaft(t, conf) + NoErr(WaitFor(env1, Leader), t) + + // Do some commits + var futures []Future + for i := 0; i < 100; i++ { + futures = append(futures, env1.raft.Apply([]byte(fmt.Sprintf("test%d", i)), 0)) + } + for _, f := range futures { + NoErr(WaitFuture(f, t), t) + log.Printf("[DEBUG] Applied %v", f) + } + + // Do a snapshot + NoErr(WaitFuture(env1.raft.Snapshot(), t), t) + + // Join a few nodes! + var envs []*RaftEnv + for i := 0; i < 4; i++ { + env := MakeRaft(t, conf) + addr := env.trans.LocalAddr() + NoErr(WaitFuture(env1.raft.AddPeer(addr), t), t) + envs = append(envs, env) + } + + // Wait for a leader + leader, err := WaitForAny(Leader, append([]*RaftEnv{env1}, envs...)) + NoErr(err, t) + + // Do some more commits + futures = nil + for i := 0; i < 100; i++ { + futures = append(futures, leader.raft.Apply([]byte(fmt.Sprintf("test%d", i)), 0)) + } + for _, f := range futures { + NoErr(WaitFuture(f, t), t) + log.Printf("[DEBUG] Applied %v", f) + } + + // Shoot two nodes in the head! + rm1, rm2 := envs[0], envs[1] + rm1.Release() + rm2.Release() + envs = envs[2:] + time.Sleep(10 * time.Millisecond) + + // Wait for a leader + leader, err = WaitForAny(Leader, append([]*RaftEnv{env1}, envs...)) + NoErr(err, t) + + // Do some more commits + futures = nil + for i := 0; i < 100; i++ { + futures = append(futures, leader.raft.Apply([]byte(fmt.Sprintf("test%d", i)), 0)) + } + for _, f := range futures { + NoErr(WaitFuture(f, t), t) + log.Printf("[DEBUG] Applied %v", f) + } + + // Join a few new nodes! + for i := 0; i < 2; i++ { + env := MakeRaft(t, conf) + addr := env.trans.LocalAddr() + NoErr(WaitFuture(leader.raft.AddPeer(addr), t), t) + envs = append(envs, env) + } + + // Remove the old nodes + NoErr(WaitFuture(leader.raft.RemovePeer(rm1.raft.localAddr), t), t) + NoErr(WaitFuture(leader.raft.RemovePeer(rm2.raft.localAddr), t), t) + + // Shoot the leader + env1.Release() + time.Sleep(3 * conf.HeartbeatTimeout) + + // Wait for a leader + leader, err = WaitForAny(Leader, envs) + NoErr(err, t) + + allEnvs := append([]*RaftEnv{env1}, envs...) + CheckConsistent(allEnvs, t) + + if len(env1.fsm.logs) != 300 { + t.Fatalf("should apply 300 logs! %d", len(env1.fsm.logs)) + } + + for _, e := range envs { + e.Release() + } +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/log.go b/Godeps/_workspace/src/github.com/hashicorp/raft/log.go new file mode 100644 index 0000000000000..a8c5a40eabf05 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/log.go @@ -0,0 +1,60 @@ +package raft + +// LogType describes various types of log entries. +type LogType uint8 + +const ( + // LogCommand is applied to a user FSM. + LogCommand LogType = iota + + // LogNoop is used to assert leadership. + LogNoop + + // LogAddPeer is used to add a new peer. + LogAddPeer + + // LogRemovePeer is used to remove an existing peer. + LogRemovePeer + + // LogBarrier is used to ensure all preceding operations have been + // applied to the FSM. It is similar to LogNoop, but instead of returning + // once committed, it only returns once the FSM manager acks it. Otherwise + // it is possible there are operations committed but not yet applied to + // the FSM. + LogBarrier +) + +// Log entries are replicated to all members of the Raft cluster +// and form the heart of the replicated state machine. +type Log struct { + Index uint64 + Term uint64 + Type LogType + Data []byte + + // peer is not exported since it is not transmitted, only used + // internally to construct the Data field. + peer string +} + +// LogStore is used to provide an interface for storing +// and retrieving logs in a durable fashion. +type LogStore interface { + // Returns the first index written. 0 for no entries. + FirstIndex() (uint64, error) + + // Returns the last index written. 0 for no entries. + LastIndex() (uint64, error) + + // Gets a log entry at a given index. + GetLog(index uint64, log *Log) error + + // Stores a log entry. + StoreLog(log *Log) error + + // Stores multiple log entries. + StoreLogs(logs []*Log) error + + // Deletes a range of log entries. The range is inclusive. + DeleteRange(min, max uint64) error +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/log_cache.go b/Godeps/_workspace/src/github.com/hashicorp/raft/log_cache.go new file mode 100644 index 0000000000000..952e98c22826f --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/log_cache.go @@ -0,0 +1,79 @@ +package raft + +import ( + "fmt" + "sync" +) + +// LogCache wraps any LogStore implementation to provide an +// in-memory ring buffer. This is used to cache access to +// the recently written entries. For implementations that do not +// cache themselves, this can provide a substantial boost by +// avoiding disk I/O on recent entries. +type LogCache struct { + store LogStore + + cache []*Log + l sync.RWMutex +} + +// NewLogCache is used to create a new LogCache with the +// given capacity and backend store. +func NewLogCache(capacity int, store LogStore) (*LogCache, error) { + if capacity <= 0 { + return nil, fmt.Errorf("capacity must be positive") + } + c := &LogCache{ + store: store, + cache: make([]*Log, capacity), + } + return c, nil +} + +func (c *LogCache) GetLog(idx uint64, log *Log) error { + // Check the buffer for an entry + c.l.RLock() + cached := c.cache[idx%uint64(len(c.cache))] + c.l.RUnlock() + + // Check if entry is valid + if cached != nil && cached.Index == idx { + *log = *cached + return nil + } + + // Forward request on cache miss + return c.store.GetLog(idx, log) +} + +func (c *LogCache) StoreLog(log *Log) error { + return c.StoreLogs([]*Log{log}) +} + +func (c *LogCache) StoreLogs(logs []*Log) error { + // Insert the logs into the ring buffer + c.l.Lock() + for _, l := range logs { + c.cache[l.Index%uint64(len(c.cache))] = l + } + c.l.Unlock() + + return c.store.StoreLogs(logs) +} + +func (c *LogCache) FirstIndex() (uint64, error) { + return c.store.FirstIndex() +} + +func (c *LogCache) LastIndex() (uint64, error) { + return c.store.LastIndex() +} + +func (c *LogCache) DeleteRange(min, max uint64) error { + // Invalidate the cache on deletes + c.l.Lock() + c.cache = make([]*Log, len(c.cache)) + c.l.Unlock() + + return c.store.DeleteRange(min, max) +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/log_cache_test.go b/Godeps/_workspace/src/github.com/hashicorp/raft/log_cache_test.go new file mode 100644 index 0000000000000..7569e78ee7006 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/log_cache_test.go @@ -0,0 +1,88 @@ +package raft + +import ( + "testing" +) + +func TestLogCache(t *testing.T) { + store := NewInmemStore() + c, _ := NewLogCache(16, store) + + // Insert into the in-mem store + for i := 0; i < 32; i++ { + log := &Log{Index: uint64(i) + 1} + store.StoreLog(log) + } + + // Check the indexes + if idx, _ := c.FirstIndex(); idx != 1 { + t.Fatalf("bad: %d", idx) + } + if idx, _ := c.LastIndex(); idx != 32 { + t.Fatalf("bad: %d", idx) + } + + // Try get log with a miss + var out Log + err := c.GetLog(1, &out) + if err != nil { + t.Fatalf("err: %v", err) + } + if out.Index != 1 { + t.Fatalf("bad: %#v", out) + } + + // Store logs + l1 := &Log{Index: 33} + l2 := &Log{Index: 34} + err = c.StoreLogs([]*Log{l1, l2}) + if err != nil { + t.Fatalf("err: %v", err) + } + + if idx, _ := c.LastIndex(); idx != 34 { + t.Fatalf("bad: %d", idx) + } + + // Check that it wrote-through + err = store.GetLog(33, &out) + if err != nil { + t.Fatalf("err: %v", err) + } + err = store.GetLog(34, &out) + if err != nil { + t.Fatalf("err: %v", err) + } + + // Delete in the backend + err = store.DeleteRange(33, 34) + if err != nil { + t.Fatalf("err: %v", err) + } + + // Should be in the ring buffer + err = c.GetLog(33, &out) + if err != nil { + t.Fatalf("err: %v", err) + } + err = c.GetLog(34, &out) + if err != nil { + t.Fatalf("err: %v", err) + } + + // Purge the ring buffer + err = c.DeleteRange(33, 34) + if err != nil { + t.Fatalf("err: %v", err) + } + + // Should not be in the ring buffer + err = c.GetLog(33, &out) + if err != ErrLogNotFound { + t.Fatalf("err: %v", err) + } + err = c.GetLog(34, &out) + if err != ErrLogNotFound { + t.Fatalf("err: %v", err) + } +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/net_transport.go b/Godeps/_workspace/src/github.com/hashicorp/raft/net_transport.go new file mode 100644 index 0000000000000..3f3ed31fdfa38 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/net_transport.go @@ -0,0 +1,606 @@ +package raft + +import ( + "bufio" + "errors" + "fmt" + "io" + "log" + "net" + "os" + "sync" + "time" + + "github.com/hashicorp/go-msgpack/codec" +) + +const ( + rpcAppendEntries uint8 = iota + rpcRequestVote + rpcInstallSnapshot + + // DefaultTimeoutScale is the default TimeoutScale in a NetworkTransport. + DefaultTimeoutScale = 256 * 1024 // 256KB + + // rpcMaxPipeline controls the maximum number of outstanding + // AppendEntries RPC calls. + rpcMaxPipeline = 128 +) + +var ( + // ErrTransportShutdown is returned when operations on a transport are + // invoked after it's been terminated. + ErrTransportShutdown = errors.New("transport shutdown") + + // ErrPipelineShutdown is returned when the pipeline is closed. + ErrPipelineShutdown = errors.New("append pipeline closed") +) + +/* + +NetworkTransport provides a network based transport that can be +used to communicate with Raft on remote machines. It requires +an underlying stream layer to provide a stream abstraction, which can +be simple TCP, TLS, etc. + +This transport is very simple and lightweight. Each RPC request is +framed by sending a byte that indicates the message type, followed +by the MsgPack encoded request. + +The response is an error string followed by the response object, +both are encoded using MsgPack. + +InstallSnapshot is special, in that after the RPC request we stream +the entire state. That socket is not re-used as the connection state +is not known if there is an error. + +*/ +type NetworkTransport struct { + connPool map[string][]*netConn + connPoolLock sync.Mutex + + consumeCh chan RPC + + heartbeatFn func(RPC) + heartbeatFnLock sync.Mutex + + logger *log.Logger + + maxPool int + + shutdown bool + shutdownCh chan struct{} + shutdownLock sync.Mutex + + stream StreamLayer + + timeout time.Duration + TimeoutScale int +} + +// StreamLayer is used with the NetworkTransport to provide +// the low level stream abstraction. +type StreamLayer interface { + net.Listener + + // Dial is used to create a new outgoing connection + Dial(address string, timeout time.Duration) (net.Conn, error) +} + +type netConn struct { + target string + conn net.Conn + r *bufio.Reader + w *bufio.Writer + dec *codec.Decoder + enc *codec.Encoder +} + +func (n *netConn) Release() error { + return n.conn.Close() +} + +type netPipeline struct { + conn *netConn + trans *NetworkTransport + + doneCh chan AppendFuture + inprogressCh chan *appendFuture + + shutdown bool + shutdownCh chan struct{} + shutdownLock sync.Mutex +} + +// NewNetworkTransport creates a new network transport with the given dialer +// and listener. The maxPool controls how many connections we will pool. The +// timeout is used to apply I/O deadlines. For InstallSnapshot, we multiply +// the timeout by (SnapshotSize / TimeoutScale). +func NewNetworkTransport( + stream StreamLayer, + maxPool int, + timeout time.Duration, + logOutput io.Writer, +) *NetworkTransport { + if logOutput == nil { + logOutput = os.Stderr + } + trans := &NetworkTransport{ + connPool: make(map[string][]*netConn), + consumeCh: make(chan RPC), + logger: log.New(logOutput, "", log.LstdFlags), + maxPool: maxPool, + shutdownCh: make(chan struct{}), + stream: stream, + timeout: timeout, + TimeoutScale: DefaultTimeoutScale, + } + go trans.listen() + return trans +} + +// SetHeartbeatHandler is used to setup a heartbeat handler +// as a fast-pass. This is to avoid head-of-line blocking from +// disk IO. +func (n *NetworkTransport) SetHeartbeatHandler(cb func(rpc RPC)) { + n.heartbeatFnLock.Lock() + defer n.heartbeatFnLock.Unlock() + n.heartbeatFn = cb +} + +// Close is used to stop the network transport. +func (n *NetworkTransport) Close() error { + n.shutdownLock.Lock() + defer n.shutdownLock.Unlock() + + if !n.shutdown { + close(n.shutdownCh) + n.stream.Close() + n.shutdown = true + } + return nil +} + +// Consumer implements the Transport interface. +func (n *NetworkTransport) Consumer() <-chan RPC { + return n.consumeCh +} + +// LocalAddr implements the Transport interface. +func (n *NetworkTransport) LocalAddr() string { + return n.stream.Addr().String() +} + +// IsShutdown is used to check if the transport is shutdown. +func (n *NetworkTransport) IsShutdown() bool { + select { + case <-n.shutdownCh: + return true + default: + return false + } +} + +// getExistingConn is used to grab a pooled connection. +func (n *NetworkTransport) getPooledConn(target string) *netConn { + n.connPoolLock.Lock() + defer n.connPoolLock.Unlock() + + conns, ok := n.connPool[target] + if !ok || len(conns) == 0 { + return nil + } + + var conn *netConn + num := len(conns) + conn, conns[num-1] = conns[num-1], nil + n.connPool[target] = conns[:num-1] + return conn +} + +// getConn is used to get a connection from the pool. +func (n *NetworkTransport) getConn(target string) (*netConn, error) { + // Check for a pooled conn + if conn := n.getPooledConn(target); conn != nil { + return conn, nil + } + + // Dial a new connection + conn, err := n.stream.Dial(target, n.timeout) + if err != nil { + return nil, err + } + + // Wrap the conn + netConn := &netConn{ + target: target, + conn: conn, + r: bufio.NewReader(conn), + w: bufio.NewWriter(conn), + } + + // Setup encoder/decoders + netConn.dec = codec.NewDecoder(netConn.r, &codec.MsgpackHandle{}) + netConn.enc = codec.NewEncoder(netConn.w, &codec.MsgpackHandle{}) + + // Done + return netConn, nil +} + +// returnConn returns a connection back to the pool. +func (n *NetworkTransport) returnConn(conn *netConn) { + n.connPoolLock.Lock() + defer n.connPoolLock.Unlock() + + key := conn.target + conns, _ := n.connPool[key] + + if !n.IsShutdown() && len(conns) < n.maxPool { + n.connPool[key] = append(conns, conn) + } else { + conn.Release() + } +} + +// AppendEntriesPipeline returns an interface that can be used to pipeline +// AppendEntries requests. +func (n *NetworkTransport) AppendEntriesPipeline(target string) (AppendPipeline, error) { + // Get a connection + conn, err := n.getConn(target) + if err != nil { + return nil, err + } + + // Create the pipeline + return newNetPipeline(n, conn), nil +} + +// AppendEntries implements the Transport interface. +func (n *NetworkTransport) AppendEntries(target string, args *AppendEntriesRequest, resp *AppendEntriesResponse) error { + return n.genericRPC(target, rpcAppendEntries, args, resp) +} + +// RequestVote implements the Transport interface. +func (n *NetworkTransport) RequestVote(target string, args *RequestVoteRequest, resp *RequestVoteResponse) error { + return n.genericRPC(target, rpcRequestVote, args, resp) +} + +// genericRPC handles a simple request/response RPC. +func (n *NetworkTransport) genericRPC(target string, rpcType uint8, args interface{}, resp interface{}) error { + // Get a conn + conn, err := n.getConn(target) + if err != nil { + return err + } + + // Set a deadline + if n.timeout > 0 { + conn.conn.SetDeadline(time.Now().Add(n.timeout)) + } + + // Send the RPC + if err := sendRPC(conn, rpcType, args); err != nil { + return err + } + + // Decode the response + canReturn, err := decodeResponse(conn, resp) + if canReturn { + n.returnConn(conn) + } + return err +} + +// InstallSnapshot implements the Transport interface. +func (n *NetworkTransport) InstallSnapshot(target string, args *InstallSnapshotRequest, resp *InstallSnapshotResponse, data io.Reader) error { + // Get a conn, always close for InstallSnapshot + conn, err := n.getConn(target) + if err != nil { + return err + } + defer conn.Release() + + // Set a deadline, scaled by request size + if n.timeout > 0 { + timeout := n.timeout * time.Duration(args.Size/int64(n.TimeoutScale)) + if timeout < n.timeout { + timeout = n.timeout + } + conn.conn.SetDeadline(time.Now().Add(timeout)) + } + + // Send the RPC + if err := sendRPC(conn, rpcInstallSnapshot, args); err != nil { + return err + } + + // Stream the state + if _, err := io.Copy(conn.w, data); err != nil { + return err + } + + // Flush + if err := conn.w.Flush(); err != nil { + return err + } + + // Decode the response, do not return conn + _, err = decodeResponse(conn, resp) + return err +} + +// EncodePeer implements the Transport interface. +func (n *NetworkTransport) EncodePeer(p string) []byte { + return []byte(p) +} + +// DecodePeer implements the Transport interface. +func (n *NetworkTransport) DecodePeer(buf []byte) string { + return string(buf) +} + +// listen is used to handling incoming connections. +func (n *NetworkTransport) listen() { + for { + // Accept incoming connections + conn, err := n.stream.Accept() + if err != nil { + if n.IsShutdown() { + return + } + n.logger.Printf("[ERR] raft-net: Failed to accept connection: %v", err) + continue + } + n.logger.Printf("[DEBUG] raft-net: %v accepted connection from: %v", n.LocalAddr(), conn.RemoteAddr()) + + // Handle the connection in dedicated routine + go n.handleConn(conn) + } +} + +// handleConn is used to handle an inbound connection for its lifespan. +func (n *NetworkTransport) handleConn(conn net.Conn) { + defer conn.Close() + r := bufio.NewReader(conn) + w := bufio.NewWriter(conn) + dec := codec.NewDecoder(r, &codec.MsgpackHandle{}) + enc := codec.NewEncoder(w, &codec.MsgpackHandle{}) + + for { + if err := n.handleCommand(r, dec, enc); err != nil { + if err != io.EOF { + n.logger.Printf("[ERR] raft-net: Failed to decode incoming command: %v", err) + } + return + } + if err := w.Flush(); err != nil { + n.logger.Printf("[ERR] raft-net: Failed to flush response: %v", err) + return + } + } +} + +// handleCommand is used to decode and dispatch a single command. +func (n *NetworkTransport) handleCommand(r *bufio.Reader, dec *codec.Decoder, enc *codec.Encoder) error { + // Get the rpc type + rpcType, err := r.ReadByte() + if err != nil { + return err + } + + // Create the RPC object + respCh := make(chan RPCResponse, 1) + rpc := RPC{ + RespChan: respCh, + } + + // Decode the command + isHeartbeat := false + switch rpcType { + case rpcAppendEntries: + var req AppendEntriesRequest + if err := dec.Decode(&req); err != nil { + return err + } + rpc.Command = &req + + // Check if this is a heartbeat + if req.Term != 0 && req.Leader != nil && + req.PrevLogEntry == 0 && req.PrevLogTerm == 0 && + len(req.Entries) == 0 && req.LeaderCommitIndex == 0 { + isHeartbeat = true + } + + case rpcRequestVote: + var req RequestVoteRequest + if err := dec.Decode(&req); err != nil { + return err + } + rpc.Command = &req + + case rpcInstallSnapshot: + var req InstallSnapshotRequest + if err := dec.Decode(&req); err != nil { + return err + } + rpc.Command = &req + rpc.Reader = io.LimitReader(r, req.Size) + + default: + return fmt.Errorf("unknown rpc type %d", rpcType) + } + + // Check for heartbeat fast-path + if isHeartbeat { + n.heartbeatFnLock.Lock() + fn := n.heartbeatFn + n.heartbeatFnLock.Unlock() + if fn != nil { + fn(rpc) + goto RESP + } + } + + // Dispatch the RPC + select { + case n.consumeCh <- rpc: + case <-n.shutdownCh: + return ErrTransportShutdown + } + + // Wait for response +RESP: + select { + case resp := <-respCh: + // Send the error first + respErr := "" + if resp.Error != nil { + respErr = resp.Error.Error() + } + if err := enc.Encode(respErr); err != nil { + return err + } + + // Send the response + if err := enc.Encode(resp.Response); err != nil { + return err + } + case <-n.shutdownCh: + return ErrTransportShutdown + } + return nil +} + +// decodeResponse is used to decode an RPC response and reports whether +// the connection can be reused. +func decodeResponse(conn *netConn, resp interface{}) (bool, error) { + // Decode the error if any + var rpcError string + if err := conn.dec.Decode(&rpcError); err != nil { + conn.Release() + return false, err + } + + // Decode the response + if err := conn.dec.Decode(resp); err != nil { + conn.Release() + return false, err + } + + // Format an error if any + if rpcError != "" { + return true, fmt.Errorf(rpcError) + } + return true, nil +} + +// sendRPC is used to encode and send the RPC. +func sendRPC(conn *netConn, rpcType uint8, args interface{}) error { + // Write the request type + if err := conn.w.WriteByte(rpcType); err != nil { + conn.Release() + return err + } + + // Send the request + if err := conn.enc.Encode(args); err != nil { + conn.Release() + return err + } + + // Flush + if err := conn.w.Flush(); err != nil { + conn.Release() + return err + } + return nil +} + +// newNetPipeline is used to construct a netPipeline from a given +// transport and connection. +func newNetPipeline(trans *NetworkTransport, conn *netConn) *netPipeline { + n := &netPipeline{ + conn: conn, + trans: trans, + doneCh: make(chan AppendFuture, rpcMaxPipeline), + inprogressCh: make(chan *appendFuture, rpcMaxPipeline), + shutdownCh: make(chan struct{}), + } + go n.decodeResponses() + return n +} + +// decodeResponses is a long running routine that decodes the responses +// sent on the connection. +func (n *netPipeline) decodeResponses() { + timeout := n.trans.timeout + for { + select { + case future := <-n.inprogressCh: + if timeout > 0 { + n.conn.conn.SetReadDeadline(time.Now().Add(timeout)) + } + + _, err := decodeResponse(n.conn, future.resp) + future.respond(err) + select { + case n.doneCh <- future: + case <-n.shutdownCh: + return + } + case <-n.shutdownCh: + return + } + } +} + +// AppendEntries is used to pipeline a new append entries request. +func (n *netPipeline) AppendEntries(args *AppendEntriesRequest, resp *AppendEntriesResponse) (AppendFuture, error) { + // Create a new future + future := &appendFuture{ + start: time.Now(), + args: args, + resp: resp, + } + future.init() + + // Add a send timeout + if timeout := n.trans.timeout; timeout > 0 { + n.conn.conn.SetWriteDeadline(time.Now().Add(timeout)) + } + + // Send the RPC + if err := sendRPC(n.conn, rpcAppendEntries, future.args); err != nil { + return nil, err + } + + // Hand-off for decoding, this can also cause back-pressure + // to prevent too many inflight requests + select { + case n.inprogressCh <- future: + return future, nil + case <-n.shutdownCh: + return nil, ErrPipelineShutdown + } +} + +// Consumer returns a channel that can be used to consume complete futures. +func (n *netPipeline) Consumer() <-chan AppendFuture { + return n.doneCh +} + +// Closed is used to shutdown the pipeline connection. +func (n *netPipeline) Close() error { + n.shutdownLock.Lock() + defer n.shutdownLock.Unlock() + if n.shutdown { + return nil + } + + // Release the connection + n.conn.Release() + + n.shutdown = true + close(n.shutdownCh) + return nil +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/net_transport_test.go b/Godeps/_workspace/src/github.com/hashicorp/raft/net_transport_test.go new file mode 100644 index 0000000000000..0127ac55e35fd --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/net_transport_test.go @@ -0,0 +1,449 @@ +package raft + +import ( + "bytes" + "reflect" + "sync" + "testing" + "time" +) + +func TestNetworkTransport_StartStop(t *testing.T) { + trans, err := NewTCPTransport("127.0.0.1:0", nil, 2, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + trans.Close() +} + +func TestNetworkTransport_Heartbeat_FastPath(t *testing.T) { + // Transport 1 is consumer + trans1, err := NewTCPTransport("127.0.0.1:0", nil, 2, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + defer trans1.Close() + + // Make the RPC request + args := AppendEntriesRequest{ + Term: 10, + Leader: []byte("cartman"), + } + resp := AppendEntriesResponse{ + Term: 4, + LastLog: 90, + Success: true, + } + + invoked := false + fastpath := func(rpc RPC) { + // Verify the command + req := rpc.Command.(*AppendEntriesRequest) + if !reflect.DeepEqual(req, &args) { + t.Fatalf("command mismatch: %#v %#v", *req, args) + } + + rpc.Respond(&resp, nil) + invoked = true + } + trans1.SetHeartbeatHandler(fastpath) + + // Transport 2 makes outbound request + trans2, err := NewTCPTransport("127.0.0.1:0", nil, 2, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + defer trans2.Close() + + var out AppendEntriesResponse + if err := trans2.AppendEntries(trans1.LocalAddr(), &args, &out); err != nil { + t.Fatalf("err: %v", err) + } + + // Verify the response + if !reflect.DeepEqual(resp, out) { + t.Fatalf("command mismatch: %#v %#v", resp, out) + } + + // Ensure fast-path is used + if !invoked { + t.Fatalf("fast-path not used") + } +} + +func TestNetworkTransport_AppendEntries(t *testing.T) { + // Transport 1 is consumer + trans1, err := NewTCPTransport("127.0.0.1:0", nil, 2, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + defer trans1.Close() + rpcCh := trans1.Consumer() + + // Make the RPC request + args := AppendEntriesRequest{ + Term: 10, + Leader: []byte("cartman"), + PrevLogEntry: 100, + PrevLogTerm: 4, + Entries: []*Log{ + &Log{ + Index: 101, + Term: 4, + Type: LogNoop, + }, + }, + LeaderCommitIndex: 90, + } + resp := AppendEntriesResponse{ + Term: 4, + LastLog: 90, + Success: true, + } + + // Listen for a request + go func() { + select { + case rpc := <-rpcCh: + // Verify the command + req := rpc.Command.(*AppendEntriesRequest) + if !reflect.DeepEqual(req, &args) { + t.Fatalf("command mismatch: %#v %#v", *req, args) + } + + rpc.Respond(&resp, nil) + + case <-time.After(200 * time.Millisecond): + t.Fatalf("timeout") + } + }() + + // Transport 2 makes outbound request + trans2, err := NewTCPTransport("127.0.0.1:0", nil, 2, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + defer trans2.Close() + + var out AppendEntriesResponse + if err := trans2.AppendEntries(trans1.LocalAddr(), &args, &out); err != nil { + t.Fatalf("err: %v", err) + } + + // Verify the response + if !reflect.DeepEqual(resp, out) { + t.Fatalf("command mismatch: %#v %#v", resp, out) + } +} + +func TestNetworkTransport_AppendEntriesPipeline(t *testing.T) { + // Transport 1 is consumer + trans1, err := NewTCPTransport("127.0.0.1:0", nil, 2, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + defer trans1.Close() + rpcCh := trans1.Consumer() + + // Make the RPC request + args := AppendEntriesRequest{ + Term: 10, + Leader: []byte("cartman"), + PrevLogEntry: 100, + PrevLogTerm: 4, + Entries: []*Log{ + &Log{ + Index: 101, + Term: 4, + Type: LogNoop, + }, + }, + LeaderCommitIndex: 90, + } + resp := AppendEntriesResponse{ + Term: 4, + LastLog: 90, + Success: true, + } + + // Listen for a request + go func() { + for i := 0; i < 10; i++ { + select { + case rpc := <-rpcCh: + // Verify the command + req := rpc.Command.(*AppendEntriesRequest) + if !reflect.DeepEqual(req, &args) { + t.Fatalf("command mismatch: %#v %#v", *req, args) + } + rpc.Respond(&resp, nil) + + case <-time.After(200 * time.Millisecond): + t.Fatalf("timeout") + } + } + }() + + // Transport 2 makes outbound request + trans2, err := NewTCPTransport("127.0.0.1:0", nil, 2, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + defer trans2.Close() + + pipeline, err := trans2.AppendEntriesPipeline(trans1.LocalAddr()) + if err != nil { + t.Fatalf("err: %v", err) + } + defer pipeline.Close() + for i := 0; i < 10; i++ { + out := new(AppendEntriesResponse) + if _, err := pipeline.AppendEntries(&args, out); err != nil { + t.Fatalf("err: %v", err) + } + } + + respCh := pipeline.Consumer() + for i := 0; i < 10; i++ { + select { + case ready := <-respCh: + // Verify the response + if !reflect.DeepEqual(&resp, ready.Response()) { + t.Fatalf("command mismatch: %#v %#v", &resp, ready.Response()) + } + case <-time.After(200 * time.Millisecond): + t.Fatalf("timeout") + } + } +} + +func TestNetworkTransport_RequestVote(t *testing.T) { + // Transport 1 is consumer + trans1, err := NewTCPTransport("127.0.0.1:0", nil, 2, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + defer trans1.Close() + rpcCh := trans1.Consumer() + + // Make the RPC request + args := RequestVoteRequest{ + Term: 20, + Candidate: []byte("butters"), + LastLogIndex: 100, + LastLogTerm: 19, + } + resp := RequestVoteResponse{ + Term: 100, + Peers: []byte("blah"), + Granted: false, + } + + // Listen for a request + go func() { + select { + case rpc := <-rpcCh: + // Verify the command + req := rpc.Command.(*RequestVoteRequest) + if !reflect.DeepEqual(req, &args) { + t.Fatalf("command mismatch: %#v %#v", *req, args) + } + + rpc.Respond(&resp, nil) + + case <-time.After(200 * time.Millisecond): + t.Fatalf("timeout") + } + }() + + // Transport 2 makes outbound request + trans2, err := NewTCPTransport("127.0.0.1:0", nil, 2, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + defer trans2.Close() + + var out RequestVoteResponse + if err := trans2.RequestVote(trans1.LocalAddr(), &args, &out); err != nil { + t.Fatalf("err: %v", err) + } + + // Verify the response + if !reflect.DeepEqual(resp, out) { + t.Fatalf("command mismatch: %#v %#v", resp, out) + } +} + +func TestNetworkTransport_InstallSnapshot(t *testing.T) { + // Transport 1 is consumer + trans1, err := NewTCPTransport("127.0.0.1:0", nil, 2, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + defer trans1.Close() + rpcCh := trans1.Consumer() + + // Make the RPC request + args := InstallSnapshotRequest{ + Term: 10, + Leader: []byte("kyle"), + LastLogIndex: 100, + LastLogTerm: 9, + Peers: []byte("blah blah"), + Size: 10, + } + resp := InstallSnapshotResponse{ + Term: 10, + Success: true, + } + + // Listen for a request + go func() { + select { + case rpc := <-rpcCh: + // Verify the command + req := rpc.Command.(*InstallSnapshotRequest) + if !reflect.DeepEqual(req, &args) { + t.Fatalf("command mismatch: %#v %#v", *req, args) + } + + // Try to read the bytes + buf := make([]byte, 10) + rpc.Reader.Read(buf) + + // Compare + if bytes.Compare(buf, []byte("0123456789")) != 0 { + t.Fatalf("bad buf %v", buf) + } + + rpc.Respond(&resp, nil) + + case <-time.After(200 * time.Millisecond): + t.Fatalf("timeout") + } + }() + + // Transport 2 makes outbound request + trans2, err := NewTCPTransport("127.0.0.1:0", nil, 2, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + defer trans2.Close() + + // Create a buffer + buf := bytes.NewBuffer([]byte("0123456789")) + + var out InstallSnapshotResponse + if err := trans2.InstallSnapshot(trans1.LocalAddr(), &args, &out, buf); err != nil { + t.Fatalf("err: %v", err) + } + + // Verify the response + if !reflect.DeepEqual(resp, out) { + t.Fatalf("command mismatch: %#v %#v", resp, out) + } +} + +func TestNetworkTransport_EncodeDecode(t *testing.T) { + // Transport 1 is consumer + trans1, err := NewTCPTransport("127.0.0.1:0", nil, 2, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + defer trans1.Close() + + local := trans1.LocalAddr() + enc := trans1.EncodePeer(local) + dec := trans1.DecodePeer(enc) + + if dec != local { + t.Fatalf("enc/dec fail: %v %v", dec, local) + } +} + +func TestNetworkTransport_PooledConn(t *testing.T) { + // Transport 1 is consumer + trans1, err := NewTCPTransport("127.0.0.1:0", nil, 2, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + defer trans1.Close() + rpcCh := trans1.Consumer() + + // Make the RPC request + args := AppendEntriesRequest{ + Term: 10, + Leader: []byte("cartman"), + PrevLogEntry: 100, + PrevLogTerm: 4, + Entries: []*Log{ + &Log{ + Index: 101, + Term: 4, + Type: LogNoop, + }, + }, + LeaderCommitIndex: 90, + } + resp := AppendEntriesResponse{ + Term: 4, + LastLog: 90, + Success: true, + } + + // Listen for a request + go func() { + for { + select { + case rpc := <-rpcCh: + // Verify the command + req := rpc.Command.(*AppendEntriesRequest) + if !reflect.DeepEqual(req, &args) { + t.Fatalf("command mismatch: %#v %#v", *req, args) + } + rpc.Respond(&resp, nil) + + case <-time.After(200 * time.Millisecond): + return + } + } + }() + + // Transport 2 makes outbound request, 3 conn pool + trans2, err := NewTCPTransport("127.0.0.1:0", nil, 3, time.Second, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + defer trans2.Close() + + // Create wait group + wg := &sync.WaitGroup{} + wg.Add(5) + + appendFunc := func() { + defer wg.Done() + var out AppendEntriesResponse + if err := trans2.AppendEntries(trans1.LocalAddr(), &args, &out); err != nil { + t.Fatalf("err: %v", err) + } + + // Verify the response + if !reflect.DeepEqual(resp, out) { + t.Fatalf("command mismatch: %#v %#v", resp, out) + } + } + + // Try to do parallel appends, should stress the conn pool + for i := 0; i < 5; i++ { + go appendFunc() + } + + // Wait for the routines to finish + wg.Wait() + + // Check the conn pool size + addr := trans1.LocalAddr() + if len(trans2.connPool[addr]) != 3 { + t.Fatalf("Expected 2 pooled conns!") + } +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/peer.go b/Godeps/_workspace/src/github.com/hashicorp/raft/peer.go new file mode 100644 index 0000000000000..6f3bcf8564507 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/peer.go @@ -0,0 +1,122 @@ +package raft + +import ( + "bytes" + "encoding/json" + "io/ioutil" + "os" + "path/filepath" + "sync" +) + +const ( + jsonPeerPath = "peers.json" +) + +// PeerStore provides an interface for persistent storage and +// retrieval of peers. We use a separate interface than StableStore +// since the peers may need to be edited by a human operator. For example, +// in a two node cluster, the failure of either node requires human intervention +// since consensus is impossible. +type PeerStore interface { + // Peers returns the list of known peers. + Peers() ([]string, error) + + // SetPeers sets the list of known peers. This is invoked when a peer is + // added or removed. + SetPeers([]string) error +} + +// StaticPeers is used to provide a static list of peers. +type StaticPeers struct { + StaticPeers []string + l sync.Mutex +} + +// Peers implements the PeerStore interface. +func (s *StaticPeers) Peers() ([]string, error) { + s.l.Lock() + peers := s.StaticPeers + s.l.Unlock() + return peers, nil +} + +// SetPeers implements the PeerStore interface. +func (s *StaticPeers) SetPeers(p []string) error { + s.l.Lock() + s.StaticPeers = p + s.l.Unlock() + return nil +} + +// JSONPeers is used to provide peer persistence on disk in the form +// of a JSON file. This allows human operators to manipulate the file. +type JSONPeers struct { + l sync.Mutex + path string + trans Transport +} + +// NewJSONPeers creates a new JSONPeers store. Requires a transport +// to handle the serialization of network addresses. +func NewJSONPeers(base string, trans Transport) *JSONPeers { + path := filepath.Join(base, jsonPeerPath) + store := &JSONPeers{ + path: path, + trans: trans, + } + return store +} + +// Peers implements the PeerStore interface. +func (j *JSONPeers) Peers() ([]string, error) { + j.l.Lock() + defer j.l.Unlock() + + // Read the file + buf, err := ioutil.ReadFile(j.path) + if err != nil && !os.IsNotExist(err) { + return nil, err + } + + // Check for no peers + if len(buf) == 0 { + return nil, nil + } + + // Decode the peers + var peerSet []string + dec := json.NewDecoder(bytes.NewReader(buf)) + if err := dec.Decode(&peerSet); err != nil { + return nil, err + } + + // Deserialize each peer + var peers []string + for _, p := range peerSet { + peers = append(peers, j.trans.DecodePeer([]byte(p))) + } + return peers, nil +} + +// SetPeers implements the PeerStore interface. +func (j *JSONPeers) SetPeers(peers []string) error { + j.l.Lock() + defer j.l.Unlock() + + // Encode each peer + var peerSet []string + for _, p := range peers { + peerSet = append(peerSet, string(j.trans.EncodePeer(p))) + } + + // Convert to JSON + var buf bytes.Buffer + enc := json.NewEncoder(&buf) + if err := enc.Encode(peerSet); err != nil { + return err + } + + // Write out as JSON + return ioutil.WriteFile(j.path, buf.Bytes(), 0755) +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/peer_test.go b/Godeps/_workspace/src/github.com/hashicorp/raft/peer_test.go new file mode 100644 index 0000000000000..1cb1159e24068 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/peer_test.go @@ -0,0 +1,44 @@ +package raft + +import ( + "io/ioutil" + "os" + "testing" +) + +func TestJSONPeers(t *testing.T) { + // Create a test dir + dir, err := ioutil.TempDir("", "raft") + if err != nil { + t.Fatalf("err: %v ", err) + } + defer os.RemoveAll(dir) + + // Create the store + _, trans := NewInmemTransport() + store := NewJSONPeers(dir, trans) + + // Try a read, should get nothing + peers, err := store.Peers() + if err != nil { + t.Fatalf("err: %v", err) + } + if len(peers) != 0 { + t.Fatalf("peers: %v", peers) + } + + // Initialize some peers + newPeers := []string{NewInmemAddr(), NewInmemAddr(), NewInmemAddr()} + if err := store.SetPeers(newPeers); err != nil { + t.Fatalf("err: %v", err) + } + + // Try a read, should peers + peers, err = store.Peers() + if err != nil { + t.Fatalf("err: %v", err) + } + if len(peers) != 3 { + t.Fatalf("peers: %v", peers) + } +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/raft.go b/Godeps/_workspace/src/github.com/hashicorp/raft/raft.go new file mode 100644 index 0000000000000..2fdc6d796c1cc --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/raft.go @@ -0,0 +1,1781 @@ +package raft + +import ( + "bytes" + "errors" + "fmt" + "io" + "log" + "os" + "strconv" + "sync" + "time" + + "github.com/armon/go-metrics" +) + +const ( + minCheckInterval = 10 * time.Millisecond +) + +var ( + keyCurrentTerm = []byte("CurrentTerm") + keyLastVoteTerm = []byte("LastVoteTerm") + keyLastVoteCand = []byte("LastVoteCand") + + // ErrLeader is returned when an operation can't be completed on a + // leader node. + ErrLeader = errors.New("node is the leader") + + // ErrNotLeader is returned when an operation can't be completed on a + // follower or candidate node. + ErrNotLeader = errors.New("node is not the leader") + + // ErrLeadershipLost is returned when a leader fails to commit a log entry + // because it's been deposed in the process. + ErrLeadershipLost = errors.New("leadership lost while committing log") + + // ErrRaftShutdown is returned when operations are requested against an + // inactive Raft. + ErrRaftShutdown = errors.New("raft is already shutdown") + + // ErrEnqueueTimeout is returned when a command fails due to a timeout. + ErrEnqueueTimeout = errors.New("timed out enqueuing operation") + + // ErrKnownPeer is returned when trying to add a peer to the configuration + // that already exists. + ErrKnownPeer = errors.New("peer already known") + + // ErrUnknownPeer is returned when trying to remove a peer from the + // configuration that doesn't exist. + ErrUnknownPeer = errors.New("peer is unknown") +) + +// commitTuple is used to send an index that was committed, +// with an optional associated future that should be invoked. +type commitTuple struct { + log *Log + future *logFuture +} + +// leaderState is state that is used while we are a leader. +type leaderState struct { + commitCh chan struct{} + inflight *inflight + replState map[string]*followerReplication + notify map[*verifyFuture]struct{} + stepDown chan struct{} +} + +// Raft implements a Raft node. +type Raft struct { + raftState + + // applyCh is used to async send logs to the main thread to + // be committed and applied to the FSM. + applyCh chan *logFuture + + // Configuration provided at Raft initialization + conf *Config + + // FSM is the client state machine to apply commands to + fsm FSM + + // fsmCommitCh is used to trigger async application of logs to the fsm + fsmCommitCh chan commitTuple + + // fsmRestoreCh is used to trigger a restore from snapshot + fsmRestoreCh chan *restoreFuture + + // fsmSnapshotCh is used to trigger a new snapshot being taken + fsmSnapshotCh chan *reqSnapshotFuture + + // lastContact is the last time we had contact from the + // leader node. This can be used to gauge staleness. + lastContact time.Time + lastContactLock sync.RWMutex + + // Leader is the current cluster leader + leader string + leaderLock sync.RWMutex + + // leaderCh is used to notify of leadership changes + leaderCh chan bool + + // leaderState used only while state is leader + leaderState leaderState + + // Stores our local addr + localAddr string + + // Used for our logging + logger *log.Logger + + // LogStore provides durable storage for logs + logs LogStore + + // Track our known peers + peerCh chan *peerFuture + peers []string + peerStore PeerStore + + // RPC chan comes from the transport layer + rpcCh <-chan RPC + + // Shutdown channel to exit, protected to prevent concurrent exits + shutdown bool + shutdownCh chan struct{} + shutdownLock sync.Mutex + + // snapshots is used to store and retrieve snapshots + snapshots SnapshotStore + + // snapshotCh is used for user triggered snapshots + snapshotCh chan *snapshotFuture + + // stable is a StableStore implementation for durable state + // It provides stable storage for many fields in raftState + stable StableStore + + // The transport layer we use + trans Transport + + // verifyCh is used to async send verify futures to the main thread + // to verify we are still the leader + verifyCh chan *verifyFuture +} + +// NewRaft is used to construct a new Raft node. It takes a configuration, as well +// as implementations of various interfaces that are required. If we have any old state, +// such as snapshots, logs, peers, etc, all those will be restored when creating the +// Raft node. +func NewRaft(conf *Config, fsm FSM, logs LogStore, stable StableStore, snaps SnapshotStore, + peerStore PeerStore, trans Transport) (*Raft, error) { + // Validate the configuration + if err := ValidateConfig(conf); err != nil { + return nil, err + } + + // Ensure we have a LogOutput + var logger *log.Logger + if conf.Logger != nil { + logger = conf.Logger + } else { + if conf.LogOutput == nil { + conf.LogOutput = os.Stderr + } + logger = log.New(conf.LogOutput, "", log.LstdFlags) + } + + // Try to restore the current term + currentTerm, err := stable.GetUint64(keyCurrentTerm) + if err != nil && err.Error() != "not found" { + return nil, fmt.Errorf("failed to load current term: %v", err) + } + + // Read the last log value + lastIdx, err := logs.LastIndex() + if err != nil { + return nil, fmt.Errorf("failed to find last log: %v", err) + } + + // Get the log + var lastLog Log + if lastIdx > 0 { + if err := logs.GetLog(lastIdx, &lastLog); err != nil { + return nil, fmt.Errorf("failed to get last log: %v", err) + } + } + + // Construct the list of peers that excludes us + localAddr := trans.LocalAddr() + peers, err := peerStore.Peers() + if err != nil { + return nil, fmt.Errorf("failed to get list of peers: %v", err) + } + peers = ExcludePeer(peers, localAddr) + + // Create Raft struct + r := &Raft{ + applyCh: make(chan *logFuture), + conf: conf, + fsm: fsm, + fsmCommitCh: make(chan commitTuple, 128), + fsmRestoreCh: make(chan *restoreFuture), + fsmSnapshotCh: make(chan *reqSnapshotFuture), + leaderCh: make(chan bool), + localAddr: localAddr, + logger: logger, + logs: logs, + peerCh: make(chan *peerFuture), + peers: peers, + peerStore: peerStore, + rpcCh: trans.Consumer(), + snapshots: snaps, + snapshotCh: make(chan *snapshotFuture), + shutdownCh: make(chan struct{}), + stable: stable, + trans: trans, + verifyCh: make(chan *verifyFuture, 64), + } + + // Initialize as a follower + r.setState(Follower) + + // Restore the current term and the last log + r.setCurrentTerm(currentTerm) + r.setLastLogIndex(lastLog.Index) + r.setLastLogTerm(lastLog.Term) + + // Attempt to restore a snapshot if there are any + if err := r.restoreSnapshot(); err != nil { + return nil, err + } + + // Setup a heartbeat fast-path to avoid head-of-line + // blocking where possible. It MUST be safe for this + // to be called concurrently with a blocking RPC. + trans.SetHeartbeatHandler(r.processHeartbeat) + + // Start the background work + r.goFunc(r.run) + r.goFunc(r.runFSM) + r.goFunc(r.runSnapshots) + return r, nil +} + +// Leader is used to return the current leader of the cluster. +// It may return empty string if there is no current leader +// or the leader is unknown. +func (r *Raft) Leader() string { + r.leaderLock.RLock() + leader := r.leader + r.leaderLock.RUnlock() + return leader +} + +// setLeader is used to modify the current leader of the cluster +func (r *Raft) setLeader(leader string) { + r.leaderLock.Lock() + r.leader = leader + r.leaderLock.Unlock() +} + +// Apply is used to apply a command to the FSM in a highly consistent +// manner. This returns a future that can be used to wait on the application. +// An optional timeout can be provided to limit the amount of time we wait +// for the command to be started. This must be run on the leader or it +// will fail. +func (r *Raft) Apply(cmd []byte, timeout time.Duration) ApplyFuture { + metrics.IncrCounter([]string{"raft", "apply"}, 1) + var timer <-chan time.Time + if timeout > 0 { + timer = time.After(timeout) + } + + // Create a log future, no index or term yet + logFuture := &logFuture{ + log: Log{ + Type: LogCommand, + Data: cmd, + }, + } + logFuture.init() + + select { + case <-timer: + return errorFuture{ErrEnqueueTimeout} + case <-r.shutdownCh: + return errorFuture{ErrRaftShutdown} + case r.applyCh <- logFuture: + return logFuture + } +} + +// Barrier is used to issue a command that blocks until all preceeding +// operations have been applied to the FSM. It can be used to ensure the +// FSM reflects all queued writes. An optional timeout can be provided to +// limit the amount of time we wait for the command to be started. This +// must be run on the leader or it will fail. +func (r *Raft) Barrier(timeout time.Duration) Future { + metrics.IncrCounter([]string{"raft", "barrier"}, 1) + var timer <-chan time.Time + if timeout > 0 { + timer = time.After(timeout) + } + + // Create a log future, no index or term yet + logFuture := &logFuture{ + log: Log{ + Type: LogBarrier, + }, + } + logFuture.init() + + select { + case <-timer: + return errorFuture{ErrEnqueueTimeout} + case <-r.shutdownCh: + return errorFuture{ErrRaftShutdown} + case r.applyCh <- logFuture: + return logFuture + } +} + +// VerifyLeader is used to ensure the current node is still +// the leader. This can be done to prevent stale reads when a +// new leader has potentially been elected. +func (r *Raft) VerifyLeader() Future { + metrics.IncrCounter([]string{"raft", "verify_leader"}, 1) + verifyFuture := &verifyFuture{} + verifyFuture.init() + select { + case <-r.shutdownCh: + return errorFuture{ErrRaftShutdown} + case r.verifyCh <- verifyFuture: + return verifyFuture + } +} + +// AddPeer is used to add a new peer into the cluster. This must be +// run on the leader or it will fail. +func (r *Raft) AddPeer(peer string) Future { + logFuture := &logFuture{ + log: Log{ + Type: LogAddPeer, + peer: peer, + }, + } + logFuture.init() + select { + case r.applyCh <- logFuture: + return logFuture + case <-r.shutdownCh: + return errorFuture{ErrRaftShutdown} + } +} + +// RemovePeer is used to remove a peer from the cluster. If the +// current leader is being removed, it will cause a new election +// to occur. This must be run on the leader or it will fail. +func (r *Raft) RemovePeer(peer string) Future { + logFuture := &logFuture{ + log: Log{ + Type: LogRemovePeer, + peer: peer, + }, + } + logFuture.init() + select { + case r.applyCh <- logFuture: + return logFuture + case <-r.shutdownCh: + return errorFuture{ErrRaftShutdown} + } +} + +// SetPeers is used to forcibly replace the set of internal peers and +// the peerstore with the ones specified. This can be considered unsafe. +func (r *Raft) SetPeers(p []string) Future { + peerFuture := &peerFuture{ + peers: p, + } + peerFuture.init() + + select { + case r.peerCh <- peerFuture: + return peerFuture + case <-r.shutdownCh: + return errorFuture{ErrRaftShutdown} + } +} + +// Shutdown is used to stop the Raft background routines. +// This is not a graceful operation. Provides a future that +// can be used to block until all background routines have exited. +func (r *Raft) Shutdown() Future { + r.shutdownLock.Lock() + defer r.shutdownLock.Unlock() + + if !r.shutdown { + close(r.shutdownCh) + r.shutdown = true + r.setState(Shutdown) + } + + return &shutdownFuture{r} +} + +// Snapshot is used to manually force Raft to take a snapshot. +// Returns a future that can be used to block until complete. +func (r *Raft) Snapshot() Future { + snapFuture := &snapshotFuture{} + snapFuture.init() + select { + case r.snapshotCh <- snapFuture: + return snapFuture + case <-r.shutdownCh: + return errorFuture{ErrRaftShutdown} + } + +} + +// State is used to return the current raft state. +func (r *Raft) State() RaftState { + return r.getState() +} + +// LeaderCh is used to get a channel which delivers signals on +// acquiring or losing leadership. It sends true if we become +// the leader, and false if we lose it. The channel is not buffered, +// and does not block on writes. +func (r *Raft) LeaderCh() <-chan bool { + return r.leaderCh +} + +func (r *Raft) String() string { + return fmt.Sprintf("Node at %s [%v]", r.localAddr, r.getState()) +} + +// LastContact returns the time of last contact by a leader. +// This only makes sense if we are currently a follower. +func (r *Raft) LastContact() time.Time { + r.lastContactLock.RLock() + last := r.lastContact + r.lastContactLock.RUnlock() + return last +} + +// Stats is used to return a map of various internal stats. This should only +// be used for informative purposes or debugging. +func (r *Raft) Stats() map[string]string { + toString := func(v uint64) string { + return strconv.FormatUint(v, 10) + } + s := map[string]string{ + "state": r.getState().String(), + "term": toString(r.getCurrentTerm()), + "last_log_index": toString(r.getLastLogIndex()), + "last_log_term": toString(r.getLastLogTerm()), + "commit_index": toString(r.getCommitIndex()), + "applied_index": toString(r.getLastApplied()), + "fsm_pending": toString(uint64(len(r.fsmCommitCh))), + "last_snapshot_index": toString(r.getLastSnapshotIndex()), + "last_snapshot_term": toString(r.getLastSnapshotTerm()), + "num_peers": toString(uint64(len(r.peers))), + } + last := r.LastContact() + if last.IsZero() { + s["last_contact"] = "never" + } else if r.getState() == Leader { + s["last_contact"] = "0" + } else { + s["last_contact"] = fmt.Sprintf("%v", time.Now().Sub(last)) + } + return s +} + +// LastIndex returns the last index in stable storage, +// either from the last log or from the last snapshot. +func (r *Raft) LastIndex() uint64 { + return r.getLastIndex() +} + +// AppliedIndex returns the last index applied to the FSM. +// This is generally lagging behind the last index, especially +// for indexes that are persisted but have not yet been considered +// committed by the leader. +func (r *Raft) AppliedIndex() uint64 { + return r.getLastApplied() +} + +// runFSM is a long running goroutine responsible for applying logs +// to the FSM. This is done async of other logs since we don't want +// the FSM to block our internal operations. +func (r *Raft) runFSM() { + var lastIndex, lastTerm uint64 + for { + select { + case req := <-r.fsmRestoreCh: + // Open the snapshot + meta, source, err := r.snapshots.Open(req.ID) + if err != nil { + req.respond(fmt.Errorf("failed to open snapshot %v: %v", req.ID, err)) + continue + } + + // Attempt to restore + start := time.Now() + if err := r.fsm.Restore(source); err != nil { + req.respond(fmt.Errorf("failed to restore snapshot %v: %v", req.ID, err)) + source.Close() + continue + } + source.Close() + metrics.MeasureSince([]string{"raft", "fsm", "restore"}, start) + + // Update the last index and term + lastIndex = meta.Index + lastTerm = meta.Term + req.respond(nil) + + case req := <-r.fsmSnapshotCh: + // Get our peers + peers, err := r.peerStore.Peers() + if err != nil { + req.respond(err) + } + + // Start a snapshot + start := time.Now() + snap, err := r.fsm.Snapshot() + metrics.MeasureSince([]string{"raft", "fsm", "snapshot"}, start) + + // Respond to the request + req.index = lastIndex + req.term = lastTerm + req.peers = peers + req.snapshot = snap + req.respond(err) + + case commitTuple := <-r.fsmCommitCh: + // Apply the log if a command + var resp interface{} + if commitTuple.log.Type == LogCommand { + start := time.Now() + resp = r.fsm.Apply(commitTuple.log) + metrics.MeasureSince([]string{"raft", "fsm", "apply"}, start) + } + + // Update the indexes + lastIndex = commitTuple.log.Index + lastTerm = commitTuple.log.Term + + // Invoke the future if given + if commitTuple.future != nil { + commitTuple.future.response = resp + commitTuple.future.respond(nil) + } + case <-r.shutdownCh: + return + } + } +} + +// run is a long running goroutine that runs the Raft FSM. +func (r *Raft) run() { + for { + // Check if we are doing a shutdown + select { + case <-r.shutdownCh: + // Clear the leader to prevent forwarding + r.setLeader("") + return + default: + } + + // Enter into a sub-FSM + switch r.getState() { + case Follower: + r.runFollower() + case Candidate: + r.runCandidate() + case Leader: + r.runLeader() + } + } +} + +// runFollower runs the FSM for a follower. +func (r *Raft) runFollower() { + didWarn := false + r.logger.Printf("[INFO] raft: %v entering Follower state", r) + heartbeatTimer := randomTimeout(r.conf.HeartbeatTimeout) + for { + select { + case rpc := <-r.rpcCh: + r.processRPC(rpc) + + case a := <-r.applyCh: + // Reject any operations since we are not the leader + a.respond(ErrNotLeader) + + case v := <-r.verifyCh: + // Reject any operations since we are not the leader + v.respond(ErrNotLeader) + + case p := <-r.peerCh: + // Set the peers + r.peers = ExcludePeer(p.peers, r.localAddr) + p.respond(r.peerStore.SetPeers(p.peers)) + + case <-heartbeatTimer: + // Restart the heartbeat timer + heartbeatTimer = randomTimeout(r.conf.HeartbeatTimeout) + + // Check if we have had a successful contact + lastContact := r.LastContact() + if time.Now().Sub(lastContact) < r.conf.HeartbeatTimeout { + continue + } + + // Heartbeat failed! Transition to the candidate state + r.setLeader("") + if len(r.peers) == 0 && !r.conf.EnableSingleNode { + if !didWarn { + r.logger.Printf("[WARN] raft: EnableSingleNode disabled, and no known peers. Aborting election.") + didWarn = true + } + } else { + r.logger.Printf("[WARN] raft: Heartbeat timeout reached, starting election") + r.setState(Candidate) + return + } + + case <-r.shutdownCh: + return + } + } +} + +// runCandidate runs the FSM for a candidate. +func (r *Raft) runCandidate() { + r.logger.Printf("[INFO] raft: %v entering Candidate state", r) + + // Start vote for us, and set a timeout + voteCh := r.electSelf() + electionTimer := randomTimeout(r.conf.ElectionTimeout) + + // Tally the votes, need a simple majority + grantedVotes := 0 + votesNeeded := r.quorumSize() + r.logger.Printf("[DEBUG] raft: Votes needed: %d", votesNeeded) + + for r.getState() == Candidate { + select { + case rpc := <-r.rpcCh: + r.processRPC(rpc) + + case vote := <-voteCh: + // Check if the term is greater than ours, bail + if vote.Term > r.getCurrentTerm() { + r.logger.Printf("[DEBUG] raft: Newer term discovered, fallback to follower") + r.setState(Follower) + r.setCurrentTerm(vote.Term) + return + } + + // Check if the vote is granted + if vote.Granted { + grantedVotes++ + r.logger.Printf("[DEBUG] raft: Vote granted. Tally: %d", grantedVotes) + } + + // Check if we've become the leader + if grantedVotes >= votesNeeded { + r.logger.Printf("[INFO] raft: Election won. Tally: %d", grantedVotes) + r.setState(Leader) + r.setLeader(r.localAddr) + return + } + + case a := <-r.applyCh: + // Reject any operations since we are not the leader + a.respond(ErrNotLeader) + + case v := <-r.verifyCh: + // Reject any operations since we are not the leader + v.respond(ErrNotLeader) + + case p := <-r.peerCh: + // Set the peers + r.peers = ExcludePeer(p.peers, r.localAddr) + p.respond(r.peerStore.SetPeers(p.peers)) + // Become a follower again + r.setState(Follower) + return + + case <-electionTimer: + // Election failed! Restart the election. We simply return, + // which will kick us back into runCandidate + r.logger.Printf("[WARN] raft: Election timeout reached, restarting election") + return + + case <-r.shutdownCh: + return + } + } +} + +// runLeader runs the FSM for a leader. Do the setup here and drop into +// the leaderLoop for the hot loop. +func (r *Raft) runLeader() { + r.logger.Printf("[INFO] raft: %v entering Leader state", r) + + // Notify that we are the leader + asyncNotifyBool(r.leaderCh, true) + + // Setup leader state + r.leaderState.commitCh = make(chan struct{}, 1) + r.leaderState.inflight = newInflight(r.leaderState.commitCh) + r.leaderState.replState = make(map[string]*followerReplication) + r.leaderState.notify = make(map[*verifyFuture]struct{}) + r.leaderState.stepDown = make(chan struct{}, 1) + + // Cleanup state on step down + defer func() { + // Stop replication + for _, p := range r.leaderState.replState { + close(p.stopCh) + } + + // Cancel inflight requests + r.leaderState.inflight.Cancel(ErrLeadershipLost) + + // Respond to any pending verify requests + for future := range r.leaderState.notify { + future.respond(ErrLeadershipLost) + } + + // Clear all the state + r.leaderState.commitCh = nil + r.leaderState.inflight = nil + r.leaderState.replState = nil + r.leaderState.notify = nil + r.leaderState.stepDown = nil + + // If we are stepping down for some reason, no known leader. + // We may have stepped down due to an RPC call, which would + // provide the leader, so we cannot always blank this out. + r.leaderLock.Lock() + if r.leader == r.localAddr { + r.leader = "" + } + r.leaderLock.Unlock() + + // Notify that we are not the leader + asyncNotifyBool(r.leaderCh, false) + }() + + // Start a replication routine for each peer + for _, peer := range r.peers { + r.startReplication(peer) + } + + // Dispatch a no-op log first. Instead of LogNoop, + // we use a LogAddPeer with our peerset. This acts like + // a no-op as well, but when doing an initial bootstrap, ensures + // that all nodes share a common peerset. + peerSet := append([]string{r.localAddr}, r.peers...) + noop := &logFuture{ + log: Log{ + Type: LogAddPeer, + Data: encodePeers(peerSet, r.trans), + }, + } + r.dispatchLogs([]*logFuture{noop}) + + // Disable EnableSingleNode after we've been elected leader. + // This is to prevent a split brain in the future, if we are removed + // from the cluster and then elect ourself as leader. + if r.conf.DisableBootstrapAfterElect && r.conf.EnableSingleNode { + r.logger.Printf("[INFO] raft: Disabling EnableSingleNode (bootstrap)") + r.conf.EnableSingleNode = false + } + + // Sit in the leader loop until we step down + r.leaderLoop() +} + +// startReplication is a helper to setup state and start async replication to a peer. +func (r *Raft) startReplication(peer string) { + lastIdx := r.getLastIndex() + s := &followerReplication{ + peer: peer, + inflight: r.leaderState.inflight, + stopCh: make(chan uint64, 1), + triggerCh: make(chan struct{}, 1), + currentTerm: r.getCurrentTerm(), + matchIndex: 0, + nextIndex: lastIdx + 1, + lastContact: time.Now(), + notifyCh: make(chan struct{}, 1), + stepDown: r.leaderState.stepDown, + } + r.leaderState.replState[peer] = s + r.goFunc(func() { r.replicate(s) }) + asyncNotifyCh(s.triggerCh) +} + +// leaderLoop is the hot loop for a leader. It is invoked +// after all the various leader setup is done. +func (r *Raft) leaderLoop() { + lease := time.After(r.conf.LeaderLeaseTimeout) + for r.getState() == Leader { + select { + case rpc := <-r.rpcCh: + r.processRPC(rpc) + + case <-r.leaderState.stepDown: + r.setState(Follower) + + case <-r.leaderState.commitCh: + // Get the committed messages + committed := r.leaderState.inflight.Committed() + for e := committed.Front(); e != nil; e = e.Next() { + // Measure the commit time + commitLog := e.Value.(*logFuture) + metrics.MeasureSince([]string{"raft", "commitTime"}, commitLog.dispatch) + + // Increment the commit index + idx := commitLog.log.Index + r.setCommitIndex(idx) + r.processLogs(idx, commitLog) + } + + case v := <-r.verifyCh: + if v.quorumSize == 0 { + // Just dispatched, start the verification + r.verifyLeader(v) + + } else if v.votes < v.quorumSize { + // Early return, means there must be a new leader + r.logger.Printf("[WARN] raft: New leader elected, stepping down") + r.setState(Follower) + delete(r.leaderState.notify, v) + v.respond(ErrNotLeader) + + } else { + // Quorum of members agree, we are still leader + delete(r.leaderState.notify, v) + v.respond(nil) + } + + case p := <-r.peerCh: + p.respond(ErrLeader) + + case newLog := <-r.applyCh: + // Group commit, gather all the ready commits + ready := []*logFuture{newLog} + for i := 0; i < r.conf.MaxAppendEntries; i++ { + select { + case newLog := <-r.applyCh: + ready = append(ready, newLog) + default: + break + } + } + + // Handle any peer set changes + n := len(ready) + for i := 0; i < n; i++ { + // Special case AddPeer and RemovePeer + log := ready[i] + if log.log.Type != LogAddPeer && log.log.Type != LogRemovePeer { + continue + } + + // Check if this log should be ignored + if !r.preparePeerChange(log) { + ready[i], ready[n-1] = ready[n-1], nil + n-- + i-- + continue + } + + // Apply peer set changes early + r.processLog(&log.log, nil, true) + } + + // Nothing to do if all logs are invalid + if n == 0 { + continue + } + + // Dispatch the logs + ready = ready[:n] + r.dispatchLogs(ready) + + case <-lease: + // Check if we've exceeded the lease, potentially stepping down + maxDiff := r.checkLeaderLease() + + // Next check interval should adjust for the last node we've + // contacted, without going negative + checkInterval := r.conf.LeaderLeaseTimeout - maxDiff + if checkInterval < minCheckInterval { + checkInterval = minCheckInterval + } + + // Renew the lease timer + lease = time.After(checkInterval) + + case <-r.shutdownCh: + return + } + } +} + +// verifyLeader must be called from the main thread for safety. +// Causes the followers to attempt an immediate heartbeat. +func (r *Raft) verifyLeader(v *verifyFuture) { + // Current leader always votes for self + v.votes = 1 + + // Set the quorum size, hot-path for single node + v.quorumSize = r.quorumSize() + if v.quorumSize == 1 { + v.respond(nil) + return + } + + // Track this request + v.notifyCh = r.verifyCh + r.leaderState.notify[v] = struct{}{} + + // Trigger immediate heartbeats + for _, repl := range r.leaderState.replState { + repl.notifyLock.Lock() + repl.notify = append(repl.notify, v) + repl.notifyLock.Unlock() + asyncNotifyCh(repl.notifyCh) + } +} + +// checkLeaderLease is used to check if we can contact a quorum of nodes +// within the last leader lease interval. If not, we need to step down, +// as we may have lost connectivity. Returns the maximum duration without +// contact. +func (r *Raft) checkLeaderLease() time.Duration { + // Track contacted nodes, we can always contact ourself + contacted := 1 + + // Check each follower + var maxDiff time.Duration + now := time.Now() + for peer, f := range r.leaderState.replState { + diff := now.Sub(f.LastContact()) + if diff <= r.conf.LeaderLeaseTimeout { + contacted++ + if diff > maxDiff { + maxDiff = diff + } + } else { + // Log at least once at high value, then debug. Otherwise it gets very verbose. + if diff <= 3*r.conf.LeaderLeaseTimeout { + r.logger.Printf("[WARN] raft: Failed to contact %v in %v", peer, diff) + } else { + r.logger.Printf("[DEBUG] raft: Failed to contact %v in %v", peer, diff) + } + } + metrics.AddSample([]string{"raft", "leader", "lastContact"}, float32(diff/time.Millisecond)) + } + + // Verify we can contact a quorum + quorum := r.quorumSize() + if contacted < quorum { + r.logger.Printf("[WARN] raft: Failed to contact quorum of nodes, stepping down") + r.setState(Follower) + } + return maxDiff +} + +// quorumSize is used to return the quorum size +func (r *Raft) quorumSize() int { + return ((len(r.peers) + 1) / 2) + 1 +} + +// preparePeerChange checks if a LogAddPeer or LogRemovePeer should be performed, +// and properly formats the data field on the log before dispatching it. +func (r *Raft) preparePeerChange(l *logFuture) bool { + // Check if this is a known peer + p := l.log.peer + knownPeer := PeerContained(r.peers, p) || r.localAddr == p + + // Ignore known peers on add + if l.log.Type == LogAddPeer && knownPeer { + l.respond(ErrKnownPeer) + return false + } + + // Ignore unknown peers on remove + if l.log.Type == LogRemovePeer && !knownPeer { + l.respond(ErrUnknownPeer) + return false + } + + // Construct the peer set + var peerSet []string + if l.log.Type == LogAddPeer { + peerSet = append([]string{p, r.localAddr}, r.peers...) + } else { + peerSet = ExcludePeer(append([]string{r.localAddr}, r.peers...), p) + } + + // Setup the log + l.log.Data = encodePeers(peerSet, r.trans) + return true +} + +// dispatchLog is called to push a log to disk, mark it +// as inflight and begin replication of it. +func (r *Raft) dispatchLogs(applyLogs []*logFuture) { + now := time.Now() + defer metrics.MeasureSince([]string{"raft", "leader", "dispatchLog"}, now) + + term := r.getCurrentTerm() + lastIndex := r.getLastIndex() + logs := make([]*Log, len(applyLogs)) + + for idx, applyLog := range applyLogs { + applyLog.dispatch = now + applyLog.log.Index = lastIndex + uint64(idx) + 1 + applyLog.log.Term = term + applyLog.policy = newMajorityQuorum(len(r.peers) + 1) + logs[idx] = &applyLog.log + } + + // Write the log entry locally + if err := r.logs.StoreLogs(logs); err != nil { + r.logger.Printf("[ERR] raft: Failed to commit logs: %v", err) + for _, applyLog := range applyLogs { + applyLog.respond(err) + } + r.setState(Follower) + return + } + + // Add this to the inflight logs, commit + r.leaderState.inflight.StartAll(applyLogs) + + // Update the last log since it's on disk now + r.setLastLogIndex(lastIndex + uint64(len(applyLogs))) + r.setLastLogTerm(term) + + // Notify the replicators of the new log + for _, f := range r.leaderState.replState { + asyncNotifyCh(f.triggerCh) + } +} + +// processLogs is used to process all the logs from the lastApplied +// up to the given index. +func (r *Raft) processLogs(index uint64, future *logFuture) { + // Reject logs we've applied already + lastApplied := r.getLastApplied() + if index <= lastApplied { + r.logger.Printf("[WARN] raft: Skipping application of old log: %d", index) + return + } + + // Apply all the preceding logs + for idx := r.getLastApplied() + 1; idx <= index; idx++ { + // Get the log, either from the future or from our log store + if future != nil && future.log.Index == idx { + r.processLog(&future.log, future, false) + + } else { + l := new(Log) + if err := r.logs.GetLog(idx, l); err != nil { + r.logger.Printf("[ERR] raft: Failed to get log at %d: %v", idx, err) + panic(err) + } + r.processLog(l, nil, false) + } + + // Update the lastApplied index and term + r.setLastApplied(idx) + } +} + +// processLog is invoked to process the application of a single committed log. +func (r *Raft) processLog(l *Log, future *logFuture, precommit bool) { + switch l.Type { + case LogBarrier: + // Barrier is handled by the FSM + fallthrough + + case LogCommand: + // Forward to the fsm handler + select { + case r.fsmCommitCh <- commitTuple{l, future}: + case <-r.shutdownCh: + if future != nil { + future.respond(ErrRaftShutdown) + } + } + + // Return so that the future is only responded to + // by the FSM handler when the application is done + return + + case LogAddPeer: + fallthrough + case LogRemovePeer: + peers := decodePeers(l.Data, r.trans) + r.logger.Printf("[DEBUG] raft: Node %v updated peer set (%v): %v", r.localAddr, l.Type, peers) + + // If the peer set does not include us, remove all other peers + removeSelf := !PeerContained(peers, r.localAddr) && l.Type == LogRemovePeer + if removeSelf { + r.peers = nil + r.peerStore.SetPeers([]string{r.localAddr}) + } else { + r.peers = ExcludePeer(peers, r.localAddr) + r.peerStore.SetPeers(peers) + } + + // Handle replication if we are the leader + if r.getState() == Leader { + for _, p := range r.peers { + if _, ok := r.leaderState.replState[p]; !ok { + r.logger.Printf("[INFO] raft: Added peer %v, starting replication", p) + r.startReplication(p) + } + } + } + + // Stop replication for old nodes + if r.getState() == Leader && !precommit { + var toDelete []string + for _, repl := range r.leaderState.replState { + if !PeerContained(r.peers, repl.peer) { + r.logger.Printf("[INFO] raft: Removed peer %v, stopping replication (Index: %d)", repl.peer, l.Index) + + // Replicate up to this index and stop + repl.stopCh <- l.Index + close(repl.stopCh) + toDelete = append(toDelete, repl.peer) + } + } + for _, name := range toDelete { + delete(r.leaderState.replState, name) + } + } + + // Handle removing ourself + if removeSelf && !precommit { + if r.conf.ShutdownOnRemove { + r.logger.Printf("[INFO] raft: Removed ourself, shutting down") + r.Shutdown() + } else { + r.logger.Printf("[INFO] raft: Removed ourself, transitioning to follower") + r.setState(Follower) + } + } + + case LogNoop: + // Ignore the no-op + default: + r.logger.Printf("[ERR] raft: Got unrecognized log type: %#v", l) + } + + // Invoke the future if given + if future != nil && !precommit { + future.respond(nil) + } +} + +// processRPC is called to handle an incoming RPC request. +func (r *Raft) processRPC(rpc RPC) { + switch cmd := rpc.Command.(type) { + case *AppendEntriesRequest: + r.appendEntries(rpc, cmd) + case *RequestVoteRequest: + r.requestVote(rpc, cmd) + case *InstallSnapshotRequest: + r.installSnapshot(rpc, cmd) + default: + r.logger.Printf("[ERR] raft: Got unexpected command: %#v", rpc.Command) + rpc.Respond(nil, fmt.Errorf("unexpected command")) + } +} + +// processHeartbeat is a special handler used just for heartbeat requests +// so that they can be fast-pathed if a transport supports it. +func (r *Raft) processHeartbeat(rpc RPC) { + defer metrics.MeasureSince([]string{"raft", "rpc", "processHeartbeat"}, time.Now()) + + // Check if we are shutdown, just ignore the RPC + select { + case <-r.shutdownCh: + return + default: + } + + // Ensure we are only handling a heartbeat + switch cmd := rpc.Command.(type) { + case *AppendEntriesRequest: + r.appendEntries(rpc, cmd) + default: + r.logger.Printf("[ERR] raft: Expected heartbeat, got command: %#v", rpc.Command) + rpc.Respond(nil, fmt.Errorf("unexpected command")) + } +} + +// appendEntries is invoked when we get an append entries RPC call. +func (r *Raft) appendEntries(rpc RPC, a *AppendEntriesRequest) { + defer metrics.MeasureSince([]string{"raft", "rpc", "appendEntries"}, time.Now()) + // Setup a response + resp := &AppendEntriesResponse{ + Term: r.getCurrentTerm(), + LastLog: r.getLastIndex(), + Success: false, + } + var rpcErr error + defer func() { + rpc.Respond(resp, rpcErr) + }() + + // Ignore an older term + if a.Term < r.getCurrentTerm() { + return + } + + // Increase the term if we see a newer one, also transition to follower + // if we ever get an appendEntries call + if a.Term > r.getCurrentTerm() || r.getState() != Follower { + // Ensure transition to follower + r.setState(Follower) + r.setCurrentTerm(a.Term) + resp.Term = a.Term + } + + // Save the current leader + r.setLeader(r.trans.DecodePeer(a.Leader)) + + // Verify the last log entry + if a.PrevLogEntry > 0 { + lastIdx, lastTerm := r.getLastEntry() + + var prevLogTerm uint64 + if a.PrevLogEntry == lastIdx { + prevLogTerm = lastTerm + + } else { + var prevLog Log + if err := r.logs.GetLog(a.PrevLogEntry, &prevLog); err != nil { + r.logger.Printf("[WARN] raft: Failed to get previous log: %d %v (last: %d)", + a.PrevLogEntry, err, lastIdx) + return + } + prevLogTerm = prevLog.Term + } + + if a.PrevLogTerm != prevLogTerm { + r.logger.Printf("[WARN] raft: Previous log term mis-match: ours: %d remote: %d", + prevLogTerm, a.PrevLogTerm) + return + } + } + + // Process any new entries + if n := len(a.Entries); n > 0 { + start := time.Now() + first := a.Entries[0] + last := a.Entries[n-1] + + // Delete any conflicting entries + lastLogIdx := r.getLastLogIndex() + if first.Index <= lastLogIdx { + r.logger.Printf("[WARN] raft: Clearing log suffix from %d to %d", first.Index, lastLogIdx) + if err := r.logs.DeleteRange(first.Index, lastLogIdx); err != nil { + r.logger.Printf("[ERR] raft: Failed to clear log suffix: %v", err) + return + } + } + + // Append the entry + if err := r.logs.StoreLogs(a.Entries); err != nil { + r.logger.Printf("[ERR] raft: Failed to append to logs: %v", err) + return + } + + // Update the lastLog + r.setLastLogIndex(last.Index) + r.setLastLogTerm(last.Term) + metrics.MeasureSince([]string{"raft", "rpc", "appendEntries", "storeLogs"}, start) + } + + // Update the commit index + if a.LeaderCommitIndex > 0 && a.LeaderCommitIndex > r.getCommitIndex() { + start := time.Now() + idx := min(a.LeaderCommitIndex, r.getLastIndex()) + r.setCommitIndex(idx) + r.processLogs(idx, nil) + metrics.MeasureSince([]string{"raft", "rpc", "appendEntries", "processLogs"}, start) + } + + // Everything went well, set success + resp.Success = true + r.lastContactLock.Lock() + r.lastContact = time.Now() + r.lastContactLock.Unlock() + return +} + +// requestVote is invoked when we get an request vote RPC call. +func (r *Raft) requestVote(rpc RPC, req *RequestVoteRequest) { + defer metrics.MeasureSince([]string{"raft", "rpc", "requestVote"}, time.Now()) + // Setup a response + resp := &RequestVoteResponse{ + Term: r.getCurrentTerm(), + Peers: encodePeers(r.peers, r.trans), + Granted: false, + } + var rpcErr error + defer func() { + rpc.Respond(resp, rpcErr) + }() + + // Check if we have an existing leader + if leader := r.Leader(); leader != "" { + r.logger.Printf("[WARN] raft: Rejecting vote from %v since we have a leader: %v", + r.trans.DecodePeer(req.Candidate), leader) + return + } + + // Ignore an older term + if req.Term < r.getCurrentTerm() { + return + } + + // Increase the term if we see a newer one + if req.Term > r.getCurrentTerm() { + // Ensure transition to follower + r.setState(Follower) + r.setCurrentTerm(req.Term) + resp.Term = req.Term + } + + // Check if we have voted yet + lastVoteTerm, err := r.stable.GetUint64(keyLastVoteTerm) + if err != nil && err.Error() != "not found" { + r.logger.Printf("[ERR] raft: Failed to get last vote term: %v", err) + return + } + lastVoteCandBytes, err := r.stable.Get(keyLastVoteCand) + if err != nil && err.Error() != "not found" { + r.logger.Printf("[ERR] raft: Failed to get last vote candidate: %v", err) + return + } + + // Check if we've voted in this election before + if lastVoteTerm == req.Term && lastVoteCandBytes != nil { + r.logger.Printf("[INFO] raft: Duplicate RequestVote for same term: %d", req.Term) + if bytes.Compare(lastVoteCandBytes, req.Candidate) == 0 { + r.logger.Printf("[WARN] raft: Duplicate RequestVote from candidate: %s", req.Candidate) + resp.Granted = true + } + return + } + + // Reject if their term is older + lastIdx, lastTerm := r.getLastEntry() + if lastTerm > req.LastLogTerm { + r.logger.Printf("[WARN] raft: Rejecting vote from %v since our last term is greater (%d, %d)", + r.trans.DecodePeer(req.Candidate), lastTerm, req.LastLogTerm) + return + } + + if lastIdx > req.LastLogIndex { + r.logger.Printf("[WARN] raft: Rejecting vote from %v since our last index is greater (%d, %d)", + r.trans.DecodePeer(req.Candidate), lastIdx, req.LastLogIndex) + return + } + + // Persist a vote for safety + if err := r.persistVote(req.Term, req.Candidate); err != nil { + r.logger.Printf("[ERR] raft: Failed to persist vote: %v", err) + return + } + + resp.Granted = true + return +} + +// installSnapshot is invoked when we get a InstallSnapshot RPC call. +// We must be in the follower state for this, since it means we are +// too far behind a leader for log replay. +func (r *Raft) installSnapshot(rpc RPC, req *InstallSnapshotRequest) { + defer metrics.MeasureSince([]string{"raft", "rpc", "installSnapshot"}, time.Now()) + // Setup a response + resp := &InstallSnapshotResponse{ + Term: r.getCurrentTerm(), + Success: false, + } + var rpcErr error + defer func() { + rpc.Respond(resp, rpcErr) + }() + + // Ignore an older term + if req.Term < r.getCurrentTerm() { + return + } + + // Increase the term if we see a newer one + if req.Term > r.getCurrentTerm() { + // Ensure transition to follower + r.setState(Follower) + r.setCurrentTerm(req.Term) + resp.Term = req.Term + } + + // Save the current leader + r.setLeader(r.trans.DecodePeer(req.Leader)) + + // Create a new snapshot + sink, err := r.snapshots.Create(req.LastLogIndex, req.LastLogTerm, req.Peers) + if err != nil { + r.logger.Printf("[ERR] raft: Failed to create snapshot to install: %v", err) + rpcErr = fmt.Errorf("failed to create snapshot: %v", err) + return + } + + // Spill the remote snapshot to disk + n, err := io.Copy(sink, rpc.Reader) + if err != nil { + sink.Cancel() + r.logger.Printf("[ERR] raft: Failed to copy snapshot: %v", err) + rpcErr = err + return + } + + // Check that we received it all + if n != req.Size { + sink.Cancel() + r.logger.Printf("[ERR] raft: Failed to receive whole snapshot: %d / %d", n, req.Size) + rpcErr = fmt.Errorf("short read") + return + } + + // Finalize the snapshot + if err := sink.Close(); err != nil { + r.logger.Printf("[ERR] raft: Failed to finalize snapshot: %v", err) + rpcErr = err + return + } + r.logger.Printf("[INFO] raft: Copied %d bytes to local snapshot", n) + + // Restore snapshot + future := &restoreFuture{ID: sink.ID()} + future.init() + select { + case r.fsmRestoreCh <- future: + case <-r.shutdownCh: + future.respond(ErrRaftShutdown) + return + } + + // Wait for the restore to happen + if err := future.Error(); err != nil { + r.logger.Printf("[ERR] raft: Failed to restore snapshot: %v", err) + rpcErr = err + return + } + + // Update the lastApplied so we don't replay old logs + r.setLastApplied(req.LastLogIndex) + + // Update the last stable snapshot info + r.setLastSnapshotIndex(req.LastLogIndex) + r.setLastSnapshotTerm(req.LastLogTerm) + + // Restore the peer set + peers := decodePeers(req.Peers, r.trans) + r.peers = ExcludePeer(peers, r.localAddr) + r.peerStore.SetPeers(peers) + + // Compact logs, continue even if this fails + if err := r.compactLogs(req.LastLogIndex); err != nil { + r.logger.Printf("[ERR] raft: Failed to compact logs: %v", err) + } + + r.logger.Printf("[INFO] raft: Installed remote snapshot") + resp.Success = true + r.lastContactLock.Lock() + r.lastContact = time.Now() + r.lastContactLock.Unlock() + return +} + +// electSelf is used to send a RequestVote RPC to all peers, +// and vote for ourself. This has the side affecting of incrementing +// the current term. The response channel returned is used to wait +// for all the responses (including a vote for ourself). +func (r *Raft) electSelf() <-chan *RequestVoteResponse { + // Create a response channel + respCh := make(chan *RequestVoteResponse, len(r.peers)+1) + + // Increment the term + r.setCurrentTerm(r.getCurrentTerm() + 1) + + // Construct the request + lastIdx, lastTerm := r.getLastEntry() + req := &RequestVoteRequest{ + Term: r.getCurrentTerm(), + Candidate: r.trans.EncodePeer(r.localAddr), + LastLogIndex: lastIdx, + LastLogTerm: lastTerm, + } + + // Construct a function to ask for a vote + askPeer := func(peer string) { + r.goFunc(func() { + defer metrics.MeasureSince([]string{"raft", "candidate", "electSelf"}, time.Now()) + resp := new(RequestVoteResponse) + err := r.trans.RequestVote(peer, req, resp) + if err != nil { + r.logger.Printf("[ERR] raft: Failed to make RequestVote RPC to %v: %v", peer, err) + resp.Term = req.Term + resp.Granted = false + } + + // If we are not a peer, we could have been removed but failed + // to receive the log message. OR it could mean an improperly configured + // cluster. Either way, we should warn + if err == nil { + peerSet := decodePeers(resp.Peers, r.trans) + if !PeerContained(peerSet, r.localAddr) { + r.logger.Printf("[WARN] raft: Remote peer %v does not have local node %v as a peer", + peer, r.localAddr) + } + } + + respCh <- resp + }) + } + + // For each peer, request a vote + for _, peer := range r.peers { + askPeer(peer) + } + + // Persist a vote for ourselves + if err := r.persistVote(req.Term, req.Candidate); err != nil { + r.logger.Printf("[ERR] raft: Failed to persist vote : %v", err) + return nil + } + + // Include our own vote + respCh <- &RequestVoteResponse{ + Term: req.Term, + Granted: true, + } + return respCh +} + +// persistVote is used to persist our vote for safety. +func (r *Raft) persistVote(term uint64, candidate []byte) error { + if err := r.stable.SetUint64(keyLastVoteTerm, term); err != nil { + return err + } + if err := r.stable.Set(keyLastVoteCand, candidate); err != nil { + return err + } + return nil +} + +// setCurrentTerm is used to set the current term in a durable manner. +func (r *Raft) setCurrentTerm(t uint64) { + // Persist to disk first + if err := r.stable.SetUint64(keyCurrentTerm, t); err != nil { + panic(fmt.Errorf("failed to save current term: %v", err)) + } + r.raftState.setCurrentTerm(t) +} + +// setState is used to update the current state. Any state +// transition causes the known leader to be cleared. This means +// that leader should be set only after updating the state. +func (r *Raft) setState(state RaftState) { + r.setLeader("") + r.raftState.setState(state) +} + +// runSnapshots is a long running goroutine used to manage taking +// new snapshots of the FSM. It runs in parallel to the FSM and +// main goroutines, so that snapshots do not block normal operation. +func (r *Raft) runSnapshots() { + for { + select { + case <-randomTimeout(r.conf.SnapshotInterval): + // Check if we should snapshot + if !r.shouldSnapshot() { + continue + } + + // Trigger a snapshot + if err := r.takeSnapshot(); err != nil { + r.logger.Printf("[ERR] raft: Failed to take snapshot: %v", err) + } + + case future := <-r.snapshotCh: + // User-triggered, run immediately + err := r.takeSnapshot() + if err != nil { + r.logger.Printf("[ERR] raft: Failed to take snapshot: %v", err) + } + future.respond(err) + + case <-r.shutdownCh: + return + } + } +} + +// shouldSnapshot checks if we meet the conditions to take +// a new snapshot. +func (r *Raft) shouldSnapshot() bool { + // Check the last snapshot index + lastSnap := r.getLastSnapshotIndex() + + // Check the last log index + lastIdx, err := r.logs.LastIndex() + if err != nil { + r.logger.Printf("[ERR] raft: Failed to get last log index: %v", err) + return false + } + + // Compare the delta to the threshold + delta := lastIdx - lastSnap + return delta >= r.conf.SnapshotThreshold +} + +// takeSnapshot is used to take a new snapshot. +func (r *Raft) takeSnapshot() error { + defer metrics.MeasureSince([]string{"raft", "snapshot", "takeSnapshot"}, time.Now()) + // Create a snapshot request + req := &reqSnapshotFuture{} + req.init() + + // Wait for dispatch or shutdown + select { + case r.fsmSnapshotCh <- req: + case <-r.shutdownCh: + return ErrRaftShutdown + } + + // Wait until we get a response + if err := req.Error(); err != nil { + return fmt.Errorf("failed to start snapshot: %v", err) + } + defer req.snapshot.Release() + + // Log that we are starting the snapshot + r.logger.Printf("[INFO] raft: Starting snapshot up to %d", req.index) + + // Encode the peerset + peerSet := encodePeers(req.peers, r.trans) + + // Create a new snapshot + start := time.Now() + sink, err := r.snapshots.Create(req.index, req.term, peerSet) + if err != nil { + return fmt.Errorf("failed to create snapshot: %v", err) + } + metrics.MeasureSince([]string{"raft", "snapshot", "create"}, start) + + // Try to persist the snapshot + start = time.Now() + if err := req.snapshot.Persist(sink); err != nil { + sink.Cancel() + return fmt.Errorf("failed to persist snapshot: %v", err) + } + metrics.MeasureSince([]string{"raft", "snapshot", "persist"}, start) + + // Close and check for error + if err := sink.Close(); err != nil { + return fmt.Errorf("failed to close snapshot: %v", err) + } + + // Update the last stable snapshot info + r.setLastSnapshotIndex(req.index) + r.setLastSnapshotTerm(req.term) + + // Compact the logs + if err := r.compactLogs(req.index); err != nil { + return err + } + + // Log completion + r.logger.Printf("[INFO] raft: Snapshot to %d complete", req.index) + return nil +} + +// compactLogs takes the last inclusive index of a snapshot +// and trims the logs that are no longer needed. +func (r *Raft) compactLogs(snapIdx uint64) error { + defer metrics.MeasureSince([]string{"raft", "compactLogs"}, time.Now()) + // Determine log ranges to compact + minLog, err := r.logs.FirstIndex() + if err != nil { + return fmt.Errorf("failed to get first log index: %v", err) + } + + // Check if we have enough logs to truncate + if r.getLastLogIndex() <= r.conf.TrailingLogs { + return nil + } + + // Truncate up to the end of the snapshot, or `TrailingLogs` + // back from the head, which ever is further back. This ensures + // at least `TrailingLogs` entries, but does not allow logs + // after the snapshot to be removed. + maxLog := min(snapIdx, r.getLastLogIndex()-r.conf.TrailingLogs) + + // Log this + r.logger.Printf("[INFO] raft: Compacting logs from %d to %d", minLog, maxLog) + + // Compact the logs + if err := r.logs.DeleteRange(minLog, maxLog); err != nil { + return fmt.Errorf("log compaction failed: %v", err) + } + return nil +} + +// restoreSnapshot attempts to restore the latest snapshots, and fails +// if none of them can be restored. This is called at initialization time, +// and is completely unsafe to call at any other time. +func (r *Raft) restoreSnapshot() error { + snapshots, err := r.snapshots.List() + if err != nil { + r.logger.Printf("[ERR] raft: Failed to list snapshots: %v", err) + return err + } + + // Try to load in order of newest to oldest + for _, snapshot := range snapshots { + _, source, err := r.snapshots.Open(snapshot.ID) + if err != nil { + r.logger.Printf("[ERR] raft: Failed to open snapshot %v: %v", snapshot.ID, err) + continue + } + defer source.Close() + + if err := r.fsm.Restore(source); err != nil { + r.logger.Printf("[ERR] raft: Failed to restore snapshot %v: %v", snapshot.ID, err) + continue + } + + // Log success + r.logger.Printf("[INFO] raft: Restored from snapshot %v", snapshot.ID) + + // Update the lastApplied so we don't replay old logs + r.setLastApplied(snapshot.Index) + + // Update the last stable snapshot info + r.setLastSnapshotIndex(snapshot.Index) + r.setLastSnapshotTerm(snapshot.Term) + + // Success! + return nil + } + + // If we had snapshots and failed to load them, its an error + if len(snapshots) > 0 { + return fmt.Errorf("failed to load any existing snapshots") + } + return nil +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/raft_test.go b/Godeps/_workspace/src/github.com/hashicorp/raft/raft_test.go new file mode 100644 index 0000000000000..284a5dd0eaa53 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/raft_test.go @@ -0,0 +1,1454 @@ +package raft + +import ( + "bytes" + "fmt" + "io" + "io/ioutil" + "log" + "os" + "reflect" + "sync" + "sync/atomic" + "testing" + "time" + + "github.com/hashicorp/go-msgpack/codec" +) + +// MockFSM is an implementation of the FSM interface, and just stores +// the logs sequentially. +type MockFSM struct { + sync.Mutex + logs [][]byte +} + +type MockSnapshot struct { + logs [][]byte + maxIndex int +} + +func (m *MockFSM) Apply(log *Log) interface{} { + m.Lock() + defer m.Unlock() + m.logs = append(m.logs, log.Data) + return len(m.logs) +} + +func (m *MockFSM) Snapshot() (FSMSnapshot, error) { + m.Lock() + defer m.Unlock() + return &MockSnapshot{m.logs, len(m.logs)}, nil +} + +func (m *MockFSM) Restore(inp io.ReadCloser) error { + m.Lock() + defer m.Unlock() + defer inp.Close() + hd := codec.MsgpackHandle{} + dec := codec.NewDecoder(inp, &hd) + + m.logs = nil + return dec.Decode(&m.logs) +} + +func (m *MockSnapshot) Persist(sink SnapshotSink) error { + hd := codec.MsgpackHandle{} + enc := codec.NewEncoder(sink, &hd) + if err := enc.Encode(m.logs[:m.maxIndex]); err != nil { + sink.Cancel() + return err + } + sink.Close() + return nil +} + +func (m *MockSnapshot) Release() { +} + +// Return configurations optimized for in-memory +func inmemConfig() *Config { + conf := DefaultConfig() + conf.HeartbeatTimeout = 50 * time.Millisecond + conf.ElectionTimeout = 50 * time.Millisecond + conf.LeaderLeaseTimeout = 50 * time.Millisecond + conf.CommitTimeout = time.Millisecond + return conf +} + +type cluster struct { + dirs []string + stores []*InmemStore + fsms []*MockFSM + snaps []*FileSnapshotStore + trans []*InmemTransport + rafts []*Raft +} + +func (c *cluster) Merge(other *cluster) { + c.dirs = append(c.dirs, other.dirs...) + c.stores = append(c.stores, other.stores...) + c.fsms = append(c.fsms, other.fsms...) + c.snaps = append(c.snaps, other.snaps...) + c.trans = append(c.trans, other.trans...) + c.rafts = append(c.rafts, other.rafts...) +} + +func (c *cluster) Close() { + var futures []Future + for _, r := range c.rafts { + futures = append(futures, r.Shutdown()) + } + + // Wait for shutdown + timer := time.AfterFunc(200*time.Millisecond, func() { + panic("timed out waiting for shutdown") + }) + + for _, f := range futures { + if err := f.Error(); err != nil { + panic(fmt.Errorf("shutdown future err: %v", err)) + } + } + timer.Stop() + + for _, d := range c.dirs { + os.RemoveAll(d) + } +} + +func (c *cluster) GetInState(s RaftState) []*Raft { + in := make([]*Raft, 0, 1) + for _, r := range c.rafts { + if r.State() == s { + in = append(in, r) + } + } + return in +} + +func (c *cluster) Leader() *Raft { + timeout := time.AfterFunc(400*time.Millisecond, func() { + panic("timeout waiting for leader") + }) + defer timeout.Stop() + + for len(c.GetInState(Leader)) < 1 { + time.Sleep(time.Millisecond) + } + leaders := c.GetInState(Leader) + if len(leaders) != 1 { + panic(fmt.Errorf("expected one leader: %v", leaders)) + } + return leaders[0] +} + +func (c *cluster) FullyConnect() { + log.Printf("[WARN] Fully Connecting") + for i, t1 := range c.trans { + for j, t2 := range c.trans { + if i != j { + t1.Connect(t2.LocalAddr(), t2) + t2.Connect(t1.LocalAddr(), t1) + } + } + } +} + +func (c *cluster) Disconnect(a string) { + log.Printf("[WARN] Disconnecting %v", a) + for _, t := range c.trans { + if t.localAddr == a { + t.DisconnectAll() + } else { + t.Disconnect(a) + } + } +} + +func (c *cluster) EnsureLeader(t *testing.T, expect string) { + limit := time.Now().Add(400 * time.Millisecond) +CHECK: + for _, r := range c.rafts { + leader := r.Leader() + if expect == "" { + if leader != "" { + if time.Now().After(limit) { + t.Fatalf("leader %v expected nil", leader) + } else { + goto WAIT + } + } + } else { + if leader == "" || leader != expect { + if time.Now().After(limit) { + t.Fatalf("leader %v expected %v", leader, expect) + } else { + goto WAIT + } + } + } + } + + return +WAIT: + time.Sleep(10 * time.Millisecond) + goto CHECK +} + +func (c *cluster) EnsureSame(t *testing.T) { + limit := time.Now().Add(400 * time.Millisecond) + first := c.fsms[0] + +CHECK: + first.Lock() + for i, fsm := range c.fsms { + if i == 0 { + continue + } + fsm.Lock() + + if len(first.logs) != len(fsm.logs) { + fsm.Unlock() + if time.Now().After(limit) { + t.Fatalf("length mismatch: %d %d", + len(first.logs), len(fsm.logs)) + } else { + goto WAIT + } + } + + for idx := 0; idx < len(first.logs); idx++ { + if bytes.Compare(first.logs[idx], fsm.logs[idx]) != 0 { + fsm.Unlock() + if time.Now().After(limit) { + t.Fatalf("log mismatch at index %d", idx) + } else { + goto WAIT + } + } + } + fsm.Unlock() + } + + first.Unlock() + return + +WAIT: + first.Unlock() + time.Sleep(20 * time.Millisecond) + goto CHECK +} + +func raftToPeerSet(r *Raft) map[string]struct{} { + peers := make(map[string]struct{}) + peers[r.localAddr] = struct{}{} + + raftPeers, _ := r.peerStore.Peers() + for _, p := range raftPeers { + peers[p] = struct{}{} + } + return peers +} + +func (c *cluster) EnsureSamePeers(t *testing.T) { + limit := time.Now().Add(400 * time.Millisecond) + peerSet := raftToPeerSet(c.rafts[0]) + +CHECK: + for i, raft := range c.rafts { + if i == 0 { + continue + } + + otherSet := raftToPeerSet(raft) + if !reflect.DeepEqual(peerSet, otherSet) { + if time.Now().After(limit) { + t.Fatalf("peer mismatch: %v %v", peerSet, otherSet) + } else { + goto WAIT + } + } + } + return + +WAIT: + time.Sleep(20 * time.Millisecond) + goto CHECK +} + +func MakeCluster(n int, t *testing.T, conf *Config) *cluster { + c := &cluster{} + peers := make([]string, 0, n) + + // Setup the stores and transports + for i := 0; i < n; i++ { + dir, err := ioutil.TempDir("", "raft") + if err != nil { + t.Fatalf("err: %v ", err) + } + store := NewInmemStore() + c.dirs = append(c.dirs, dir) + c.stores = append(c.stores, store) + c.fsms = append(c.fsms, &MockFSM{}) + + dir2, snap := FileSnapTest(t) + c.dirs = append(c.dirs, dir2) + c.snaps = append(c.snaps, snap) + + addr, trans := NewInmemTransport() + c.trans = append(c.trans, trans) + peers = append(peers, addr) + } + + // Wire the transports together + c.FullyConnect() + + // Create all the rafts + for i := 0; i < n; i++ { + if conf == nil { + conf = inmemConfig() + } + if n == 1 { + conf.EnableSingleNode = true + } + + logs := c.stores[i] + store := c.stores[i] + snap := c.snaps[i] + trans := c.trans[i] + peerStore := &StaticPeers{StaticPeers: peers} + + raft, err := NewRaft(conf, c.fsms[i], logs, store, snap, peerStore, trans) + if err != nil { + t.Fatalf("err: %v", err) + } + c.rafts = append(c.rafts, raft) + } + + return c +} + +func MakeClusterNoPeers(n int, t *testing.T, conf *Config) *cluster { + c := &cluster{} + + // Setup the stores and transports + for i := 0; i < n; i++ { + dir, err := ioutil.TempDir("", "raft") + if err != nil { + t.Fatalf("err: %v ", err) + } + store := NewInmemStore() + c.dirs = append(c.dirs, dir) + c.stores = append(c.stores, store) + c.fsms = append(c.fsms, &MockFSM{}) + + dir2, snap := FileSnapTest(t) + c.dirs = append(c.dirs, dir2) + c.snaps = append(c.snaps, snap) + + _, trans := NewInmemTransport() + c.trans = append(c.trans, trans) + } + + // Wire the transports together + c.FullyConnect() + + // Create all the rafts + for i := 0; i < n; i++ { + if conf == nil { + conf = inmemConfig() + } + + logs := c.stores[i] + store := c.stores[i] + snap := c.snaps[i] + trans := c.trans[i] + peerStore := &StaticPeers{} + + raft, err := NewRaft(conf, c.fsms[i], logs, store, snap, peerStore, trans) + if err != nil { + t.Fatalf("err: %v", err) + } + c.rafts = append(c.rafts, raft) + } + + return c +} + +func TestRaft_StartStop(t *testing.T) { + c := MakeCluster(1, t, nil) + c.Close() +} + +func TestRaft_AfterShutdown(t *testing.T) { + c := MakeCluster(1, t, nil) + c.Close() + raft := c.rafts[0] + + // Everything should fail now + if f := raft.Apply(nil, 0); f.Error() != ErrRaftShutdown { + t.Fatalf("should be shutdown: %v", f.Error()) + } + if f := raft.AddPeer(NewInmemAddr()); f.Error() != ErrRaftShutdown { + t.Fatalf("should be shutdown: %v", f.Error()) + } + if f := raft.RemovePeer(NewInmemAddr()); f.Error() != ErrRaftShutdown { + t.Fatalf("should be shutdown: %v", f.Error()) + } + if f := raft.Snapshot(); f.Error() != ErrRaftShutdown { + t.Fatalf("should be shutdown: %v", f.Error()) + } + + // Should be idempotent + raft.Shutdown() +} + +func TestRaft_SingleNode(t *testing.T) { + conf := inmemConfig() + c := MakeCluster(1, t, conf) + defer c.Close() + raft := c.rafts[0] + + // Watch leaderCh for change + select { + case v := <-raft.LeaderCh(): + if !v { + t.Fatalf("should become leader") + } + case <-time.After(conf.HeartbeatTimeout * 3): + t.Fatalf("timeout becoming leader") + } + + // Should be leader + if s := raft.State(); s != Leader { + t.Fatalf("expected leader: %v", s) + } + + // Should be able to apply + future := raft.Apply([]byte("test"), time.Millisecond) + if err := future.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Check the response + if future.Response().(int) != 1 { + t.Fatalf("bad response: %v", future.Response()) + } + + // Check the index + if idx := future.Index(); idx == 0 { + t.Fatalf("bad index: %d", idx) + } + + // Check that it is applied to the FSM + if len(c.fsms[0].logs) != 1 { + t.Fatalf("did not apply to FSM!") + } +} + +func TestRaft_TripleNode(t *testing.T) { + // Make the cluster + c := MakeCluster(3, t, nil) + defer c.Close() + + // Should be one leader + leader := c.Leader() + c.EnsureLeader(t, leader.localAddr) + + // Should be able to apply + future := leader.Apply([]byte("test"), time.Millisecond) + if err := future.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Wait for replication + time.Sleep(30 * time.Millisecond) + + // Check that it is applied to the FSM + for _, fsm := range c.fsms { + fsm.Lock() + num := len(fsm.logs) + fsm.Unlock() + if num != 1 { + t.Fatalf("did not apply to FSM!") + } + } +} + +func TestRaft_LeaderFail(t *testing.T) { + // Make the cluster + c := MakeCluster(3, t, nil) + defer c.Close() + + // Should be one leader + leader := c.Leader() + + // Should be able to apply + future := leader.Apply([]byte("test"), time.Millisecond) + if err := future.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Wait for replication + time.Sleep(30 * time.Millisecond) + + // Disconnect the leader now + log.Printf("[INFO] Disconnecting %v", leader) + c.Disconnect(leader.localAddr) + + // Wait for new leader + limit := time.Now().Add(200 * time.Millisecond) + var newLead *Raft + for time.Now().Before(limit) && newLead == nil { + time.Sleep(10 * time.Millisecond) + leaders := c.GetInState(Leader) + if len(leaders) == 1 && leaders[0] != leader { + newLead = leaders[0] + } + } + if newLead == nil { + t.Fatalf("expected new leader") + } + + // Ensure the term is greater + if newLead.getCurrentTerm() <= leader.getCurrentTerm() { + t.Fatalf("expected newer term! %d %d", newLead.getCurrentTerm(), leader.getCurrentTerm()) + } + + // Apply should work not work on old leader + future1 := leader.Apply([]byte("fail"), time.Millisecond) + + // Apply should work on newer leader + future2 := newLead.Apply([]byte("apply"), time.Millisecond) + + // Future2 should work + if err := future2.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Reconnect the networks + log.Printf("[INFO] Reconnecting %v", leader) + c.FullyConnect() + + // Future1 should fail + if err := future1.Error(); err != ErrLeadershipLost && err != ErrNotLeader { + t.Fatalf("err: %v", err) + } + + // Wait for log replication + c.EnsureSame(t) + + // Check two entries are applied to the FSM + for _, fsm := range c.fsms { + fsm.Lock() + if len(fsm.logs) != 2 { + t.Fatalf("did not apply both to FSM! %v", fsm.logs) + } + if bytes.Compare(fsm.logs[0], []byte("test")) != 0 { + t.Fatalf("first entry should be 'test'") + } + if bytes.Compare(fsm.logs[1], []byte("apply")) != 0 { + t.Fatalf("second entry should be 'apply'") + } + fsm.Unlock() + } +} + +func TestRaft_BehindFollower(t *testing.T) { + // Make the cluster + c := MakeCluster(3, t, nil) + defer c.Close() + + // Disconnect one follower + leader := c.Leader() + followers := c.GetInState(Follower) + behind := followers[0] + c.Disconnect(behind.localAddr) + + // Commit a lot of things + var future Future + for i := 0; i < 100; i++ { + future = leader.Apply([]byte(fmt.Sprintf("test%d", i)), 0) + } + + // Wait for the last future to apply + if err := future.Error(); err != nil { + t.Fatalf("err: %v", err) + } else { + log.Printf("[INFO] Finished apply without behind follower") + } + + // Check that we have a non zero last contact + if behind.LastContact().IsZero() { + t.Fatalf("expected previous contact") + } + + // Reconnect the behind node + c.FullyConnect() + + // Ensure all the logs are the same + c.EnsureSame(t) + + // Ensure one leader + leader = c.Leader() + c.EnsureLeader(t, leader.localAddr) +} + +func TestRaft_ApplyNonLeader(t *testing.T) { + // Make the cluster + c := MakeCluster(3, t, nil) + defer c.Close() + + // Wait for a leader + c.Leader() + time.Sleep(10 * time.Millisecond) + + // Try to apply to them + followers := c.GetInState(Follower) + if len(followers) != 2 { + t.Fatalf("Expected 2 followers") + } + follower := followers[0] + + // Try to apply + future := follower.Apply([]byte("test"), time.Millisecond) + + if future.Error() != ErrNotLeader { + t.Fatalf("should not apply on follower") + } + + // Should be cached + if future.Error() != ErrNotLeader { + t.Fatalf("should not apply on follower") + } +} + +func TestRaft_ApplyConcurrent(t *testing.T) { + // Make the cluster + conf := inmemConfig() + conf.HeartbeatTimeout = 80 * time.Millisecond + conf.ElectionTimeout = 80 * time.Millisecond + c := MakeCluster(3, t, conf) + defer c.Close() + + // Wait for a leader + leader := c.Leader() + + // Create a wait group + var group sync.WaitGroup + group.Add(100) + + applyF := func(i int) { + defer group.Done() + future := leader.Apply([]byte(fmt.Sprintf("test%d", i)), 0) + if err := future.Error(); err != nil { + t.Fatalf("err: %v", err) + } + } + + // Concurrently apply + for i := 0; i < 100; i++ { + go applyF(i) + } + + // Wait to finish + doneCh := make(chan struct{}) + go func() { + group.Wait() + close(doneCh) + }() + select { + case <-doneCh: + case <-time.After(time.Second): + t.Fatalf("timeout") + } + + // Check the FSMs + c.EnsureSame(t) +} + +func TestRaft_ApplyConcurrent_Timeout(t *testing.T) { + // Make the cluster + conf := inmemConfig() + conf.HeartbeatTimeout = 80 * time.Millisecond + conf.ElectionTimeout = 80 * time.Millisecond + c := MakeCluster(1, t, conf) + defer c.Close() + + // Wait for a leader + leader := c.Leader() + + // Enough enqueues should cause at least one timeout... + var didTimeout int32 = 0 + for i := 0; i < 200; i++ { + go func(i int) { + future := leader.Apply([]byte(fmt.Sprintf("test%d", i)), time.Microsecond) + if future.Error() == ErrEnqueueTimeout { + atomic.StoreInt32(&didTimeout, 1) + } + }(i) + } + + // Wait + time.Sleep(20 * time.Millisecond) + + // Some should have failed + if atomic.LoadInt32(&didTimeout) == 0 { + t.Fatalf("expected a timeout") + } +} + +func TestRaft_JoinNode(t *testing.T) { + // Make a cluster + c := MakeCluster(2, t, nil) + defer c.Close() + + // Apply a log to this cluster to ensure it is 'newer' + var future Future + leader := c.Leader() + future = leader.Apply([]byte("first"), 0) + if err := future.Error(); err != nil { + t.Fatalf("err: %v", err) + } else { + log.Printf("[INFO] Applied log") + } + + // Make a new cluster of 1 + c1 := MakeCluster(1, t, nil) + + // Merge clusters + c.Merge(c1) + c.FullyConnect() + + // Wait until we have 2 leaders + limit := time.Now().Add(200 * time.Millisecond) + var leaders []*Raft + for time.Now().Before(limit) && len(leaders) != 2 { + time.Sleep(10 * time.Millisecond) + leaders = c.GetInState(Leader) + } + if len(leaders) != 2 { + t.Fatalf("expected two leader: %v", leaders) + } + + // Join the new node in + future = leader.AddPeer(c1.rafts[0].localAddr) + if err := future.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Wait until we have 2 followers + limit = time.Now().Add(200 * time.Millisecond) + var followers []*Raft + for time.Now().Before(limit) && len(followers) != 2 { + time.Sleep(10 * time.Millisecond) + followers = c.GetInState(Follower) + } + if len(followers) != 2 { + t.Fatalf("expected two followers: %v", followers) + } + + // Check the FSMs + c.EnsureSame(t) + + // Check the peers + c.EnsureSamePeers(t) + + // Ensure one leader + leader = c.Leader() + c.EnsureLeader(t, leader.localAddr) +} + +func TestRaft_RemoveFollower(t *testing.T) { + // Make a cluster + c := MakeCluster(3, t, nil) + defer c.Close() + + // Get the leader + leader := c.Leader() + + // Wait until we have 2 followers + limit := time.Now().Add(200 * time.Millisecond) + var followers []*Raft + for time.Now().Before(limit) && len(followers) != 2 { + time.Sleep(10 * time.Millisecond) + followers = c.GetInState(Follower) + } + if len(followers) != 2 { + t.Fatalf("expected two followers: %v", followers) + } + + // Remove a follower + follower := followers[0] + future := leader.RemovePeer(follower.localAddr) + if err := future.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Wait a while + time.Sleep(20 * time.Millisecond) + + // Other nodes should have fewer peers + if peers, _ := leader.peerStore.Peers(); len(peers) != 2 { + t.Fatalf("too many peers") + } + if peers, _ := followers[1].peerStore.Peers(); len(peers) != 2 { + t.Fatalf("too many peers") + } +} + +func TestRaft_RemoveLeader(t *testing.T) { + // Make a cluster + c := MakeCluster(3, t, nil) + defer c.Close() + + // Get the leader + leader := c.Leader() + + // Wait until we have 2 followers + limit := time.Now().Add(200 * time.Millisecond) + var followers []*Raft + for time.Now().Before(limit) && len(followers) != 2 { + time.Sleep(10 * time.Millisecond) + followers = c.GetInState(Follower) + } + if len(followers) != 2 { + t.Fatalf("expected two followers: %v", followers) + } + + // Remove the leader + leader.RemovePeer(leader.localAddr) + + // Wait a while + time.Sleep(20 * time.Millisecond) + + // Should have a new leader + newLeader := c.Leader() + + // Wait a bit for log application + time.Sleep(20 * time.Millisecond) + + // Other nodes should have fewer peers + if peers, _ := newLeader.peerStore.Peers(); len(peers) != 2 { + t.Fatalf("too many peers") + } + + // Old leader should be shutdown + if leader.State() != Shutdown { + t.Fatalf("leader should be shutdown") + } + + // Old leader should have no peers + if peers, _ := leader.peerStore.Peers(); len(peers) != 1 { + t.Fatalf("leader should have no peers") + } +} + +func TestRaft_RemoveLeader_NoShutdown(t *testing.T) { + // Make a cluster + conf := inmemConfig() + conf.ShutdownOnRemove = false + c := MakeCluster(3, t, conf) + defer c.Close() + + // Get the leader + leader := c.Leader() + + // Wait until we have 2 followers + limit := time.Now().Add(200 * time.Millisecond) + var followers []*Raft + for time.Now().Before(limit) && len(followers) != 2 { + time.Sleep(10 * time.Millisecond) + followers = c.GetInState(Follower) + } + if len(followers) != 2 { + t.Fatalf("expected two followers: %v", followers) + } + + // Remove the leader + leader.RemovePeer(leader.localAddr) + + // Wait a while + time.Sleep(20 * time.Millisecond) + + // Should have a new leader + newLeader := c.Leader() + + // Wait a bit for log application + time.Sleep(20 * time.Millisecond) + + // Other nodes should have fewer peers + if peers, _ := newLeader.peerStore.Peers(); len(peers) != 2 { + t.Fatalf("too many peers") + } + + // Old leader should be a follower + if leader.State() != Follower { + t.Fatalf("leader should be shutdown") + } + + // Old leader should have no peers + if peers, _ := leader.peerStore.Peers(); len(peers) != 1 { + t.Fatalf("leader should have no peers") + } +} + +func TestRaft_RemoveLeader_SplitCluster(t *testing.T) { + // Enable operation after a remove + conf := inmemConfig() + conf.EnableSingleNode = true + conf.ShutdownOnRemove = false + conf.DisableBootstrapAfterElect = false + + // Make a cluster + c := MakeCluster(3, t, conf) + defer c.Close() + + // Get the leader + leader := c.Leader() + + // Remove the leader + leader.RemovePeer(leader.localAddr) + + // Wait until we have 2 leaders + limit := time.Now().Add(200 * time.Millisecond) + var leaders []*Raft + for time.Now().Before(limit) && len(leaders) != 2 { + time.Sleep(10 * time.Millisecond) + leaders = c.GetInState(Leader) + } + if len(leaders) != 2 { + t.Fatalf("expected two leader: %v", leaders) + } + + // Old leader should have no peers + if len(leader.peers) != 0 { + t.Fatalf("leader should have no peers") + } +} + +func TestRaft_AddKnownPeer(t *testing.T) { + // Make a cluster + c := MakeCluster(3, t, nil) + defer c.Close() + + // Get the leader + leader := c.Leader() + followers := c.GetInState(Follower) + + // Add a follower + future := leader.AddPeer(followers[0].localAddr) + + // Should be already added + if err := future.Error(); err != ErrKnownPeer { + t.Fatalf("err: %v", err) + } +} + +func TestRaft_RemoveUnknownPeer(t *testing.T) { + // Make a cluster + c := MakeCluster(3, t, nil) + defer c.Close() + + // Get the leader + leader := c.Leader() + + // Remove unknown + future := leader.RemovePeer(NewInmemAddr()) + + // Should be already added + if err := future.Error(); err != ErrUnknownPeer { + t.Fatalf("err: %v", err) + } +} + +func TestRaft_SnapshotRestore(t *testing.T) { + // Make the cluster + conf := inmemConfig() + conf.TrailingLogs = 10 + c := MakeCluster(1, t, conf) + defer c.Close() + + // Commit a lot of things + leader := c.Leader() + var future Future + for i := 0; i < 100; i++ { + future = leader.Apply([]byte(fmt.Sprintf("test%d", i)), 0) + } + + // Wait for the last future to apply + if err := future.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Take a snapshot + snapFuture := leader.Snapshot() + if err := snapFuture.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Check for snapshot + if snaps, _ := leader.snapshots.List(); len(snaps) != 1 { + t.Fatalf("should have a snapshot") + } + + // Logs should be trimmed + if idx, _ := leader.logs.FirstIndex(); idx != 92 { + t.Fatalf("should trim logs to 92: %d", idx) + } + + // Shutdown + shutdown := leader.Shutdown() + if err := shutdown.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Restart the Raft + r := leader + r, err := NewRaft(r.conf, r.fsm, r.logs, r.stable, + r.snapshots, r.peerStore, r.trans) + if err != nil { + t.Fatalf("err: %v", err) + } + c.rafts[0] = r + + // We should have restored from the snapshot! + if last := r.getLastApplied(); last != 101 { + t.Fatalf("bad last: %v", last) + } +} + +func TestRaft_SnapshotRestore_PeerChange(t *testing.T) { + // Make the cluster + conf := inmemConfig() + conf.TrailingLogs = 10 + c := MakeCluster(3, t, conf) + defer c.Close() + + // Commit a lot of things + leader := c.Leader() + var future Future + for i := 0; i < 100; i++ { + future = leader.Apply([]byte(fmt.Sprintf("test%d", i)), 0) + } + + // Wait for the last future to apply + if err := future.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Take a snapshot + snapFuture := leader.Snapshot() + if err := snapFuture.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Shutdown + shutdown := leader.Shutdown() + if err := shutdown.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Make a separate cluster + c2 := MakeClusterNoPeers(2, t, conf) + defer c2.Close() + + // Kill the old cluster + for _, sec := range c.rafts { + if sec != leader { + sec.Shutdown() + } + } + + // Change the peer addresses + peers := []string{leader.trans.LocalAddr()} + for _, sec := range c2.rafts { + peers = append(peers, sec.trans.LocalAddr()) + } + + // Restart the Raft with new peers + r := leader + peerStore := &StaticPeers{StaticPeers: peers} + r, err := NewRaft(r.conf, r.fsm, r.logs, r.stable, + r.snapshots, peerStore, r.trans) + if err != nil { + t.Fatalf("err: %v", err) + } + c.rafts[0] = r + c2.rafts = append(c2.rafts, r) + c2.trans = append(c2.trans, r.trans.(*InmemTransport)) + c2.fsms = append(c2.fsms, r.fsm.(*MockFSM)) + c2.FullyConnect() + + // Wait a while + time.Sleep(50 * time.Millisecond) + + // Ensure we elect a leader, and that we replicate + // to our new followers + c2.EnsureSame(t) + + // We should have restored from the snapshot! + if last := r.getLastApplied(); last != 102 { + t.Fatalf("bad last: %v", last) + } +} + +func TestRaft_AutoSnapshot(t *testing.T) { + // Make the cluster + conf := inmemConfig() + conf.SnapshotInterval = 5 * time.Millisecond + conf.SnapshotThreshold = 50 + conf.TrailingLogs = 10 + c := MakeCluster(1, t, conf) + defer c.Close() + + // Commit a lot of things + leader := c.Leader() + var future Future + for i := 0; i < 100; i++ { + future = leader.Apply([]byte(fmt.Sprintf("test%d", i)), 0) + } + + // Wait for the last future to apply + if err := future.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Wait for a snapshot to happen + time.Sleep(50 * time.Millisecond) + + // Check for snapshot + if snaps, _ := leader.snapshots.List(); len(snaps) == 0 { + t.Fatalf("should have a snapshot") + } +} + +func TestRaft_SendSnapshotFollower(t *testing.T) { + // Make the cluster + conf := inmemConfig() + conf.TrailingLogs = 10 + c := MakeCluster(3, t, conf) + defer c.Close() + + // Disconnect one follower + followers := c.GetInState(Follower) + behind := followers[0] + c.Disconnect(behind.localAddr) + + // Commit a lot of things + leader := c.Leader() + var future Future + for i := 0; i < 100; i++ { + future = leader.Apply([]byte(fmt.Sprintf("test%d", i)), 0) + } + + // Wait for the last future to apply + if err := future.Error(); err != nil { + t.Fatalf("err: %v", err) + } else { + log.Printf("[INFO] Finished apply without behind follower") + } + + // Snapshot, this will truncate logs! + for _, r := range c.rafts { + future = r.Snapshot() + if err := future.Error(); err != nil { + t.Fatalf("err: %v", err) + } + } + + // Reconnect the behind node + c.FullyConnect() + + // Ensure all the logs are the same + c.EnsureSame(t) +} + +func TestRaft_ReJoinFollower(t *testing.T) { + // Enable operation after a remove + conf := inmemConfig() + conf.ShutdownOnRemove = false + + // Make a cluster + c := MakeCluster(3, t, conf) + defer c.Close() + + // Get the leader + leader := c.Leader() + + // Wait until we have 2 followers + limit := time.Now().Add(200 * time.Millisecond) + var followers []*Raft + for time.Now().Before(limit) && len(followers) != 2 { + time.Sleep(10 * time.Millisecond) + followers = c.GetInState(Follower) + } + if len(followers) != 2 { + t.Fatalf("expected two followers: %v", followers) + } + + // Remove a follower + follower := followers[0] + future := leader.RemovePeer(follower.localAddr) + if err := future.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Wait a while + time.Sleep(20 * time.Millisecond) + + // Other nodes should have fewer peers + if peers, _ := leader.peerStore.Peers(); len(peers) != 2 { + t.Fatalf("too many peers: %v", peers) + } + if peers, _ := followers[1].peerStore.Peers(); len(peers) != 2 { + t.Fatalf("too many peers: %v", peers) + } + + // Get the leader + time.Sleep(20 * time.Millisecond) + leader = c.Leader() + + // Rejoin. The follower will have a higher term than the leader, + // this will cause the leader to step down, and a new round of elections + // to take place. We should eventually re-stabilize. + future = leader.AddPeer(follower.localAddr) + if err := future.Error(); err != nil && err != ErrLeadershipLost { + t.Fatalf("err: %v", err) + } + + // Wait a while + time.Sleep(40 * time.Millisecond) + + // Other nodes should have fewer peers + if peers, _ := leader.peerStore.Peers(); len(peers) != 3 { + t.Fatalf("missing peers: %v", peers) + } + if peers, _ := followers[1].peerStore.Peers(); len(peers) != 3 { + t.Fatalf("missing peers: %v", peers) + } + + // Should be a follower now + if follower.State() != Follower { + t.Fatalf("bad state: %v", follower.State()) + } +} + +func TestRaft_LeaderLeaseExpire(t *testing.T) { + // Make a cluster + conf := inmemConfig() + c := MakeCluster(2, t, conf) + defer c.Close() + + // Get the leader + leader := c.Leader() + + // Wait until we have a followers + limit := time.Now().Add(200 * time.Millisecond) + var followers []*Raft + for time.Now().Before(limit) && len(followers) != 1 { + time.Sleep(10 * time.Millisecond) + followers = c.GetInState(Follower) + } + if len(followers) != 1 { + t.Fatalf("expected a followers: %v", followers) + } + + // Disconnect the follower now + follower := followers[0] + log.Printf("[INFO] Disconnecting %v", follower) + c.Disconnect(follower.localAddr) + + // Watch the leaderCh + select { + case v := <-leader.LeaderCh(): + if v { + t.Fatalf("should step down as leader") + } + case <-time.After(conf.LeaderLeaseTimeout * 2): + t.Fatalf("timeout stepping down as leader") + } + + // Should be no leaders + if len(c.GetInState(Leader)) != 0 { + t.Fatalf("expected step down") + } + + // Verify no further contact + last := follower.LastContact() + time.Sleep(50 * time.Millisecond) + + // Check that last contact has not changed + if last != follower.LastContact() { + t.Fatalf("unexpected further contact") + } + + // Ensure both have cleared their leader + if l := leader.Leader(); l != "" { + t.Fatalf("bad: %v", l) + } + if l := follower.Leader(); l != "" { + t.Fatalf("bad: %v", l) + } +} + +func TestRaft_Barrier(t *testing.T) { + // Make the cluster + c := MakeCluster(3, t, nil) + defer c.Close() + + // Get the leader + leader := c.Leader() + + // Commit a lot of things + for i := 0; i < 100; i++ { + leader.Apply([]byte(fmt.Sprintf("test%d", i)), 0) + } + + // Wait for a barrier complete + barrier := leader.Barrier(0) + + // Wait for the barrier future to apply + if err := barrier.Error(); err != nil { + t.Fatalf("err: %v", err) + } + + // Ensure all the logs are the same + c.EnsureSame(t) + if len(c.fsms[0].logs) != 100 { + t.Fatalf("Bad log length") + } +} + +func TestRaft_VerifyLeader(t *testing.T) { + // Make the cluster + c := MakeCluster(3, t, nil) + defer c.Close() + + // Get the leader + leader := c.Leader() + + // Verify we are leader + verify := leader.VerifyLeader() + + // Wait for the verify to apply + if err := verify.Error(); err != nil { + t.Fatalf("err: %v", err) + } +} + +func TestRaft_VerifyLeader_Single(t *testing.T) { + // Make the cluster + c := MakeCluster(1, t, nil) + defer c.Close() + + // Get the leader + leader := c.Leader() + + // Verify we are leader + verify := leader.VerifyLeader() + + // Wait for the verify to apply + if err := verify.Error(); err != nil { + t.Fatalf("err: %v", err) + } +} + +func TestRaft_VerifyLeader_Fail(t *testing.T) { + // Make a cluster + conf := inmemConfig() + c := MakeCluster(2, t, conf) + defer c.Close() + + // Get the leader + leader := c.Leader() + + // Wait until we have a followers + limit := time.Now().Add(200 * time.Millisecond) + var followers []*Raft + for time.Now().Before(limit) && len(followers) != 1 { + time.Sleep(10 * time.Millisecond) + followers = c.GetInState(Follower) + } + if len(followers) != 1 { + t.Fatalf("expected a followers: %v", followers) + } + + // Force follower to different term + follower := followers[0] + follower.setCurrentTerm(follower.getCurrentTerm() + 1) + + // Verify we are leader + verify := leader.VerifyLeader() + + // Wait for the leader to step down + if err := verify.Error(); err != ErrNotLeader && err != ErrLeadershipLost { + t.Fatalf("err: %v", err) + } + + // Ensure the known leader is cleared + if l := leader.Leader(); l != "" { + t.Fatalf("bad: %v", l) + } +} + +func TestRaft_VerifyLeader_ParitalConnect(t *testing.T) { + // Make a cluster + conf := inmemConfig() + c := MakeCluster(3, t, conf) + defer c.Close() + + // Get the leader + leader := c.Leader() + + // Wait until we have a followers + limit := time.Now().Add(200 * time.Millisecond) + var followers []*Raft + for time.Now().Before(limit) && len(followers) != 2 { + time.Sleep(10 * time.Millisecond) + followers = c.GetInState(Follower) + } + if len(followers) != 2 { + t.Fatalf("expected a followers: %v", followers) + } + + // Force partial disconnect + follower := followers[0] + log.Printf("[INFO] Disconnecting %v", follower) + c.Disconnect(follower.localAddr) + + // Verify we are leader + verify := leader.VerifyLeader() + + // Wait for the leader to step down + if err := verify.Error(); err != nil { + t.Fatalf("err: %v", err) + } +} + +func TestRaft_SettingPeers(t *testing.T) { + // Make the cluster + c := MakeClusterNoPeers(3, t, nil) + defer c.Close() + + peers := make([]string, 0) + for _, v := range c.rafts { + peers = append(peers, v.localAddr) + } + + for _, v := range c.rafts { + future := v.SetPeers(peers) + if err := future.Error(); err != nil { + t.Fatalf("error setting peers: %v", err) + } + } + + // Wait a while + time.Sleep(20 * time.Millisecond) + + // Should have a new leader + if leader := c.Leader(); leader == nil { + t.Fatalf("no leader?") + } +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/replication.go b/Godeps/_workspace/src/github.com/hashicorp/raft/replication.go new file mode 100644 index 0000000000000..30541952db05a --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/replication.go @@ -0,0 +1,513 @@ +package raft + +import ( + "errors" + "fmt" + "sync" + "time" + + "github.com/armon/go-metrics" +) + +const ( + maxFailureScale = 12 + failureWait = 10 * time.Millisecond +) + +var ( + // ErrLogNotFound indicates a given log entry is not available. + ErrLogNotFound = errors.New("log not found") + + // ErrPipelineReplicationNotSupported can be returned by the transport to + // signal that pipeline replication is not supported in general, and that + // no error message should be produced. + ErrPipelineReplicationNotSupported = errors.New("pipeline replication not supported") +) + +type followerReplication struct { + peer string + inflight *inflight + + stopCh chan uint64 + triggerCh chan struct{} + + currentTerm uint64 + matchIndex uint64 + nextIndex uint64 + + lastContact time.Time + lastContactLock sync.RWMutex + + failures uint64 + + notifyCh chan struct{} + notify []*verifyFuture + notifyLock sync.Mutex + + // stepDown is used to indicate to the leader that we + // should step down based on information from a follower. + stepDown chan struct{} + + // allowPipeline is used to control it seems like + // pipeline replication should be enabled. + allowPipeline bool +} + +// notifyAll is used to notify all the waiting verify futures +// if the follower believes we are still the leader. +func (s *followerReplication) notifyAll(leader bool) { + // Clear the waiting notifies minimizing lock time + s.notifyLock.Lock() + n := s.notify + s.notify = nil + s.notifyLock.Unlock() + + // Submit our votes + for _, v := range n { + v.vote(leader) + } +} + +// LastContact returns the time of last contact. +func (s *followerReplication) LastContact() time.Time { + s.lastContactLock.RLock() + last := s.lastContact + s.lastContactLock.RUnlock() + return last +} + +// setLastContact sets the last contact to the current time. +func (s *followerReplication) setLastContact() { + s.lastContactLock.Lock() + s.lastContact = time.Now() + s.lastContactLock.Unlock() +} + +// replicate is a long running routine that is used to manage +// the process of replicating logs to our followers. +func (r *Raft) replicate(s *followerReplication) { + // Start an async heartbeating routing + stopHeartbeat := make(chan struct{}) + defer close(stopHeartbeat) + r.goFunc(func() { r.heartbeat(s, stopHeartbeat) }) + +RPC: + shouldStop := false + for !shouldStop { + select { + case maxIndex := <-s.stopCh: + // Make a best effort to replicate up to this index + if maxIndex > 0 { + r.replicateTo(s, maxIndex) + } + return + case <-s.triggerCh: + shouldStop = r.replicateTo(s, r.getLastLogIndex()) + case <-randomTimeout(r.conf.CommitTimeout): + shouldStop = r.replicateTo(s, r.getLastLogIndex()) + } + + // If things looks healthy, switch to pipeline mode + if !shouldStop && s.allowPipeline { + goto PIPELINE + } + } + return + +PIPELINE: + // Disable until re-enabled + s.allowPipeline = false + + // Replicates using a pipeline for high performance. This method + // is not able to gracefully recover from errors, and so we fall back + // to standard mode on failure. + if err := r.pipelineReplicate(s); err != nil { + if err != ErrPipelineReplicationNotSupported { + r.logger.Printf("[ERR] raft: Failed to start pipeline replication to %s: %s", s.peer, err) + } + } + goto RPC +} + +// replicateTo is used to replicate the logs up to a given last index. +// If the follower log is behind, we take care to bring them up to date. +func (r *Raft) replicateTo(s *followerReplication, lastIndex uint64) (shouldStop bool) { + // Create the base request + var req AppendEntriesRequest + var resp AppendEntriesResponse + var start time.Time +START: + // Prevent an excessive retry rate on errors + if s.failures > 0 { + select { + case <-time.After(backoff(failureWait, s.failures, maxFailureScale)): + case <-r.shutdownCh: + } + } + + // Setup the request + if err := r.setupAppendEntries(s, &req, s.nextIndex, lastIndex); err == ErrLogNotFound { + goto SEND_SNAP + } else if err != nil { + return + } + + // Make the RPC call + start = time.Now() + if err := r.trans.AppendEntries(s.peer, &req, &resp); err != nil { + r.logger.Printf("[ERR] raft: Failed to AppendEntries to %v: %v", s.peer, err) + s.failures++ + return + } + appendStats(s.peer, start, float32(len(req.Entries))) + + // Check for a newer term, stop running + if resp.Term > req.Term { + r.handleStaleTerm(s) + return true + } + + // Update the last contact + s.setLastContact() + + // Update s based on success + if resp.Success { + // Update our replication state + updateLastAppended(s, &req) + + // Clear any failures, allow pipelining + s.failures = 0 + s.allowPipeline = true + } else { + s.nextIndex = max(min(s.nextIndex-1, resp.LastLog+1), 1) + s.matchIndex = s.nextIndex - 1 + s.failures++ + r.logger.Printf("[WARN] raft: AppendEntries to %v rejected, sending older logs (next: %d)", s.peer, s.nextIndex) + } + +CHECK_MORE: + // Check if there are more logs to replicate + if s.nextIndex <= lastIndex { + goto START + } + return + + // SEND_SNAP is used when we fail to get a log, usually because the follower + // is too far behind, and we must ship a snapshot down instead +SEND_SNAP: + if stop, err := r.sendLatestSnapshot(s); stop { + return true + } else if err != nil { + r.logger.Printf("[ERR] raft: Failed to send snapshot to %v: %v", s.peer, err) + return + } + + // Check if there is more to replicate + goto CHECK_MORE +} + +// sendLatestSnapshot is used to send the latest snapshot we have +// down to our follower. +func (r *Raft) sendLatestSnapshot(s *followerReplication) (bool, error) { + // Get the snapshots + snapshots, err := r.snapshots.List() + if err != nil { + r.logger.Printf("[ERR] raft: Failed to list snapshots: %v", err) + return false, err + } + + // Check we have at least a single snapshot + if len(snapshots) == 0 { + return false, fmt.Errorf("no snapshots found") + } + + // Open the most recent snapshot + snapID := snapshots[0].ID + meta, snapshot, err := r.snapshots.Open(snapID) + if err != nil { + r.logger.Printf("[ERR] raft: Failed to open snapshot %v: %v", snapID, err) + return false, err + } + defer snapshot.Close() + + // Setup the request + req := InstallSnapshotRequest{ + Term: s.currentTerm, + Leader: r.trans.EncodePeer(r.localAddr), + LastLogIndex: meta.Index, + LastLogTerm: meta.Term, + Peers: meta.Peers, + Size: meta.Size, + } + + // Make the call + start := time.Now() + var resp InstallSnapshotResponse + if err := r.trans.InstallSnapshot(s.peer, &req, &resp, snapshot); err != nil { + r.logger.Printf("[ERR] raft: Failed to install snapshot %v: %v", snapID, err) + s.failures++ + return false, err + } + metrics.MeasureSince([]string{"raft", "replication", "installSnapshot", s.peer}, start) + + // Check for a newer term, stop running + if resp.Term > req.Term { + r.handleStaleTerm(s) + return true, nil + } + + // Update the last contact + s.setLastContact() + + // Check for success + if resp.Success { + // Mark any inflight logs as committed + s.inflight.CommitRange(s.matchIndex+1, meta.Index) + + // Update the indexes + s.matchIndex = meta.Index + s.nextIndex = s.matchIndex + 1 + + // Clear any failures + s.failures = 0 + + // Notify we are still leader + s.notifyAll(true) + } else { + s.failures++ + r.logger.Printf("[WARN] raft: InstallSnapshot to %v rejected", s.peer) + } + return false, nil +} + +// heartbeat is used to periodically invoke AppendEntries on a peer +// to ensure they don't time out. This is done async of replicate(), +// since that routine could potentially be blocked on disk IO. +func (r *Raft) heartbeat(s *followerReplication, stopCh chan struct{}) { + var failures uint64 + req := AppendEntriesRequest{ + Term: s.currentTerm, + Leader: r.trans.EncodePeer(r.localAddr), + } + var resp AppendEntriesResponse + for { + // Wait for the next heartbeat interval or forced notify + select { + case <-s.notifyCh: + case <-randomTimeout(r.conf.HeartbeatTimeout / 10): + case <-stopCh: + return + } + + start := time.Now() + if err := r.trans.AppendEntries(s.peer, &req, &resp); err != nil { + r.logger.Printf("[ERR] raft: Failed to heartbeat to %v: %v", s.peer, err) + failures++ + select { + case <-time.After(backoff(failureWait, failures, maxFailureScale)): + case <-stopCh: + } + } else { + s.setLastContact() + failures = 0 + metrics.MeasureSince([]string{"raft", "replication", "heartbeat", s.peer}, start) + s.notifyAll(resp.Success) + } + } +} + +// pipelineReplicate is used when we have synchronized our state with the follower, +// and want to switch to a higher performance pipeline mode of replication. +// We only pipeline AppendEntries commands, and if we ever hit an error, we fall +// back to the standard replication which can handle more complex situations. +func (r *Raft) pipelineReplicate(s *followerReplication) error { + // Create a new pipeline + pipeline, err := r.trans.AppendEntriesPipeline(s.peer) + if err != nil { + return err + } + defer pipeline.Close() + + // Log start and stop of pipeline + r.logger.Printf("[INFO] raft: pipelining replication to peer %v", s.peer) + defer r.logger.Printf("[INFO] raft: aborting pipeline replication to peer %v", s.peer) + + // Create a shutdown and finish channel + stopCh := make(chan struct{}) + finishCh := make(chan struct{}) + + // Start a dedicated decoder + r.goFunc(func() { r.pipelineDecode(s, pipeline, stopCh, finishCh) }) + + // Start pipeline sends at the last good nextIndex + nextIndex := s.nextIndex + + shouldStop := false +SEND: + for !shouldStop { + select { + case <-finishCh: + break SEND + case maxIndex := <-s.stopCh: + if maxIndex > 0 { + r.pipelineSend(s, pipeline, &nextIndex, maxIndex) + } + break SEND + case <-s.triggerCh: + shouldStop = r.pipelineSend(s, pipeline, &nextIndex, r.getLastLogIndex()) + case <-randomTimeout(r.conf.CommitTimeout): + shouldStop = r.pipelineSend(s, pipeline, &nextIndex, r.getLastLogIndex()) + } + } + + // Stop our decoder, and wait for it to finish + close(stopCh) + select { + case <-finishCh: + case <-r.shutdownCh: + } + return nil +} + +// pipelineSend is used to send data over a pipeline. +func (r *Raft) pipelineSend(s *followerReplication, p AppendPipeline, nextIdx *uint64, lastIndex uint64) (shouldStop bool) { + // Create a new append request + req := new(AppendEntriesRequest) + if err := r.setupAppendEntries(s, req, *nextIdx, lastIndex); err != nil { + return true + } + + // Pipeline the append entries + if _, err := p.AppendEntries(req, new(AppendEntriesResponse)); err != nil { + r.logger.Printf("[ERR] raft: Failed to pipeline AppendEntries to %v: %v", s.peer, err) + return true + } + + // Increase the next send log to avoid re-sending old logs + if n := len(req.Entries); n > 0 { + last := req.Entries[n-1] + *nextIdx = last.Index + 1 + } + return false +} + +// pipelineDecode is used to decode the responses of pipelined requests. +func (r *Raft) pipelineDecode(s *followerReplication, p AppendPipeline, stopCh, finishCh chan struct{}) { + defer close(finishCh) + respCh := p.Consumer() + for { + select { + case ready := <-respCh: + req, resp := ready.Request(), ready.Response() + appendStats(s.peer, ready.Start(), float32(len(req.Entries))) + + // Check for a newer term, stop running + if resp.Term > req.Term { + r.handleStaleTerm(s) + return + } + + // Update the last contact + s.setLastContact() + + // Abort pipeline if not successful + if !resp.Success { + return + } + + // Update our replication state + updateLastAppended(s, req) + case <-stopCh: + return + } + } +} + +// setupAppendEntries is used to setup an append entries request. +func (r *Raft) setupAppendEntries(s *followerReplication, req *AppendEntriesRequest, nextIndex, lastIndex uint64) error { + req.Term = s.currentTerm + req.Leader = r.trans.EncodePeer(r.localAddr) + req.LeaderCommitIndex = r.getCommitIndex() + if err := r.setPreviousLog(req, nextIndex); err != nil { + return err + } + if err := r.setNewLogs(req, nextIndex, lastIndex); err != nil { + return err + } + return nil +} + +// setPreviousLog is used to setup the PrevLogEntry and PrevLogTerm for an +// AppendEntriesRequest given the next index to replicate. +func (r *Raft) setPreviousLog(req *AppendEntriesRequest, nextIndex uint64) error { + // Guard for the first index, since there is no 0 log entry + // Guard against the previous index being a snapshot as well + if nextIndex == 1 { + req.PrevLogEntry = 0 + req.PrevLogTerm = 0 + + } else if (nextIndex - 1) == r.getLastSnapshotIndex() { + req.PrevLogEntry = r.getLastSnapshotIndex() + req.PrevLogTerm = r.getLastSnapshotTerm() + + } else { + var l Log + if err := r.logs.GetLog(nextIndex-1, &l); err != nil { + r.logger.Printf("[ERR] raft: Failed to get log at index %d: %v", + nextIndex-1, err) + return err + } + + // Set the previous index and term (0 if nextIndex is 1) + req.PrevLogEntry = l.Index + req.PrevLogTerm = l.Term + } + return nil +} + +// setNewLogs is used to setup the logs which should be appended for a request. +func (r *Raft) setNewLogs(req *AppendEntriesRequest, nextIndex, lastIndex uint64) error { + // Append up to MaxAppendEntries or up to the lastIndex + req.Entries = make([]*Log, 0, r.conf.MaxAppendEntries) + maxIndex := min(nextIndex+uint64(r.conf.MaxAppendEntries)-1, lastIndex) + for i := nextIndex; i <= maxIndex; i++ { + oldLog := new(Log) + if err := r.logs.GetLog(i, oldLog); err != nil { + r.logger.Printf("[ERR] raft: Failed to get log at index %d: %v", i, err) + return err + } + req.Entries = append(req.Entries, oldLog) + } + return nil +} + +// appendStats is used to emit stats about an AppendEntries invocation. +func appendStats(peer string, start time.Time, logs float32) { + metrics.MeasureSince([]string{"raft", "replication", "appendEntries", "rpc", peer}, start) + metrics.IncrCounter([]string{"raft", "replication", "appendEntries", "logs", peer}, logs) +} + +// handleStaleTerm is used when a follower indicates that we have a stale term. +func (r *Raft) handleStaleTerm(s *followerReplication) { + r.logger.Printf("[ERR] raft: peer %v has newer term, stopping replication", s.peer) + s.notifyAll(false) // No longer leader + asyncNotifyCh(s.stepDown) +} + +// updateLastAppended is used to update follower replication state after a successful +// AppendEntries RPC. +func updateLastAppended(s *followerReplication, req *AppendEntriesRequest) { + // Mark any inflight logs as committed + if logs := req.Entries; len(logs) > 0 { + first := logs[0] + last := logs[len(logs)-1] + s.inflight.CommitRange(first.Index, last.Index) + + // Update the indexes + s.matchIndex = last.Index + s.nextIndex = last.Index + 1 + } + + // Notify still leader + s.notifyAll(true) +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/snapshot.go b/Godeps/_workspace/src/github.com/hashicorp/raft/snapshot.go new file mode 100644 index 0000000000000..7151f43ce2656 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/snapshot.go @@ -0,0 +1,40 @@ +package raft + +import ( + "io" +) + +// SnapshotMeta is for metadata of a snapshot. +type SnapshotMeta struct { + ID string // ID is opaque to the store, and is used for opening + Index uint64 + Term uint64 + Peers []byte + Size int64 +} + +// SnapshotStore interface is used to allow for flexible implementations +// of snapshot storage and retrieval. For example, a client could implement +// a shared state store such as S3, allowing new nodes to restore snapshots +// without steaming from the leader. +type SnapshotStore interface { + // Create is used to begin a snapshot at a given index and term, + // with the current peer set already encoded. + Create(index, term uint64, peers []byte) (SnapshotSink, error) + + // List is used to list the available snapshots in the store. + // It should return then in descending order, with the highest index first. + List() ([]*SnapshotMeta, error) + + // Open takes a snapshot ID and provides a ReadCloser. Once close is + // called it is assumed the snapshot is no longer needed. + Open(id string) (*SnapshotMeta, io.ReadCloser, error) +} + +// SnapshotSink is returned by StartSnapshot. The FSM will Write state +// to the sink and call Close on completion. On error, Cancel will be invoked. +type SnapshotSink interface { + io.WriteCloser + ID() string + Cancel() error +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/stable.go b/Godeps/_workspace/src/github.com/hashicorp/raft/stable.go new file mode 100644 index 0000000000000..4588ea8a9af5a --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/stable.go @@ -0,0 +1,15 @@ +package raft + +// StableStore is used to provide stable storage +// of key configurations to ensure safety. +type StableStore interface { + Set(key []byte, val []byte) error + + // Get returns the value for key, or an empty byte slice if key was not found. + Get(key []byte) ([]byte, error) + + SetUint64(key []byte, val uint64) error + + // GetUint64 returns the uint64 value for key, or 0 if key was not found. + GetUint64(key []byte) (uint64, error) +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/state.go b/Godeps/_workspace/src/github.com/hashicorp/raft/state.go new file mode 100644 index 0000000000000..41e80a1b51093 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/state.go @@ -0,0 +1,169 @@ +package raft + +import ( + "sync/atomic" +) + +// RaftState captures the state of a Raft node: Follower, Candidate, Leader, +// or Shutdown. +type RaftState uint32 + +const ( + // Follower is the initial state of a Raft node. + Follower RaftState = iota + + // Candidate is one of the valid states of a Raft node. + Candidate + + // Leader is one of the valid states of a Raft node. + Leader + + // Shutdown is the terminal state of a Raft node. + Shutdown +) + +func (s RaftState) String() string { + switch s { + case Follower: + return "Follower" + case Candidate: + return "Candidate" + case Leader: + return "Leader" + case Shutdown: + return "Shutdown" + default: + return "Unknown" + } +} + +// raftState is used to maintain various state variables +// and provides an interface to set/get the variables in a +// thread safe manner. +type raftState struct { + // The current term, cache of StableStore + currentTerm uint64 + + // Cache the latest log from LogStore + LastLogIndex uint64 + LastLogTerm uint64 + + // Highest committed log entry + commitIndex uint64 + + // Last applied log to the FSM + lastApplied uint64 + + // Cache the latest snapshot index/term + lastSnapshotIndex uint64 + lastSnapshotTerm uint64 + + // Tracks the number of live routines + runningRoutines int32 + + // The current state + state RaftState +} + +func (r *raftState) getState() RaftState { + stateAddr := (*uint32)(&r.state) + return RaftState(atomic.LoadUint32(stateAddr)) +} + +func (r *raftState) setState(s RaftState) { + stateAddr := (*uint32)(&r.state) + atomic.StoreUint32(stateAddr, uint32(s)) +} + +func (r *raftState) getCurrentTerm() uint64 { + return atomic.LoadUint64(&r.currentTerm) +} + +func (r *raftState) setCurrentTerm(term uint64) { + atomic.StoreUint64(&r.currentTerm, term) +} + +func (r *raftState) getLastLogIndex() uint64 { + return atomic.LoadUint64(&r.LastLogIndex) +} + +func (r *raftState) setLastLogIndex(term uint64) { + atomic.StoreUint64(&r.LastLogIndex, term) +} + +func (r *raftState) getLastLogTerm() uint64 { + return atomic.LoadUint64(&r.LastLogTerm) +} + +func (r *raftState) setLastLogTerm(term uint64) { + atomic.StoreUint64(&r.LastLogTerm, term) +} + +func (r *raftState) getCommitIndex() uint64 { + return atomic.LoadUint64(&r.commitIndex) +} + +func (r *raftState) setCommitIndex(term uint64) { + atomic.StoreUint64(&r.commitIndex, term) +} + +func (r *raftState) getLastApplied() uint64 { + return atomic.LoadUint64(&r.lastApplied) +} + +func (r *raftState) setLastApplied(term uint64) { + atomic.StoreUint64(&r.lastApplied, term) +} + +func (r *raftState) getLastSnapshotIndex() uint64 { + return atomic.LoadUint64(&r.lastSnapshotIndex) +} + +func (r *raftState) setLastSnapshotIndex(term uint64) { + atomic.StoreUint64(&r.lastSnapshotIndex, term) +} + +func (r *raftState) getLastSnapshotTerm() uint64 { + return atomic.LoadUint64(&r.lastSnapshotTerm) +} + +func (r *raftState) setLastSnapshotTerm(term uint64) { + atomic.StoreUint64(&r.lastSnapshotTerm, term) +} + +func (r *raftState) incrRoutines() { + atomic.AddInt32(&r.runningRoutines, 1) +} + +func (r *raftState) decrRoutines() { + atomic.AddInt32(&r.runningRoutines, -1) +} + +func (r *raftState) getRoutines() int32 { + return atomic.LoadInt32(&r.runningRoutines) +} + +// Start a goroutine and properly handle the race between a routine +// starting and incrementing, and exiting and decrementing. +func (r *raftState) goFunc(f func()) { + r.incrRoutines() + go func() { + defer r.decrRoutines() + f() + }() +} + +// getLastIndex returns the last index in stable storage. +// Either from the last log or from the last snapshot. +func (r *raftState) getLastIndex() uint64 { + return max(r.getLastLogIndex(), r.getLastSnapshotIndex()) +} + +// getLastEntry returns the last index and term in stable storage. +// Either from the last log or from the last snapshot. +func (r *raftState) getLastEntry() (uint64, uint64) { + if r.getLastLogIndex() >= r.getLastSnapshotIndex() { + return r.getLastLogIndex(), r.getLastLogTerm() + } + return r.getLastSnapshotIndex(), r.getLastSnapshotTerm() +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/tcp_transport.go b/Godeps/_workspace/src/github.com/hashicorp/raft/tcp_transport.go new file mode 100644 index 0000000000000..1b1ea9c352f3e --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/tcp_transport.go @@ -0,0 +1,80 @@ +package raft + +import ( + "errors" + "io" + "net" + "time" +) + +var ( + errNotAdvertisable = errors.New("local bind address is not advertisable") + errNotTCP = errors.New("local address is not a TCP address") +) + +// TCPStreamLayer implements StreamLayer interface for plain TCP. +type TCPStreamLayer struct { + advertise net.Addr + listener *net.TCPListener +} + +// NewTCPTransport returns a NetworkTransport that is built on top of +// a TCP streaming transport layer. +func NewTCPTransport( + bindAddr string, + advertise net.Addr, + maxPool int, + timeout time.Duration, + logOutput io.Writer, +) (*NetworkTransport, error) { + // Try to bind + list, err := net.Listen("tcp", bindAddr) + if err != nil { + return nil, err + } + + // Create stream + stream := &TCPStreamLayer{ + advertise: advertise, + listener: list.(*net.TCPListener), + } + + // Verify that we have a usable advertise address + addr, ok := stream.Addr().(*net.TCPAddr) + if !ok { + list.Close() + return nil, errNotTCP + } + if addr.IP.IsUnspecified() { + list.Close() + return nil, errNotAdvertisable + } + + // Create the network transport + trans := NewNetworkTransport(stream, maxPool, timeout, logOutput) + return trans, nil +} + +// Dial implements the StreamLayer interface. +func (t *TCPStreamLayer) Dial(address string, timeout time.Duration) (net.Conn, error) { + return net.DialTimeout("tcp", address, timeout) +} + +// Accept implements the net.Listener interface. +func (t *TCPStreamLayer) Accept() (c net.Conn, err error) { + return t.listener.Accept() +} + +// Close implements the net.Listener interface. +func (t *TCPStreamLayer) Close() (err error) { + return t.listener.Close() +} + +// Addr implements the net.Listener interface. +func (t *TCPStreamLayer) Addr() net.Addr { + // Use an advertise addr if provided + if t.advertise != nil { + return t.advertise + } + return t.listener.Addr() +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/tcp_transport_test.go b/Godeps/_workspace/src/github.com/hashicorp/raft/tcp_transport_test.go new file mode 100644 index 0000000000000..22d59da2adecd --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/tcp_transport_test.go @@ -0,0 +1,24 @@ +package raft + +import ( + "net" + "testing" +) + +func TestTCPTransport_BadAddr(t *testing.T) { + _, err := NewTCPTransport("0.0.0.0:0", nil, 1, 0, nil) + if err != errNotAdvertisable { + t.Fatalf("err: %v", err) + } +} + +func TestTCPTransport_WithAdvertise(t *testing.T) { + addr := &net.TCPAddr{IP: []byte{127, 0, 0, 1}, Port: 12345} + trans, err := NewTCPTransport("0.0.0.0:0", addr, 1, 0, nil) + if err != nil { + t.Fatalf("err: %v", err) + } + if trans.LocalAddr() != "127.0.0.1:12345" { + t.Fatalf("bad: %v", trans.LocalAddr()) + } +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/transport.go b/Godeps/_workspace/src/github.com/hashicorp/raft/transport.go new file mode 100644 index 0000000000000..8928de0c2fc07 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/transport.go @@ -0,0 +1,85 @@ +package raft + +import ( + "io" + "time" +) + +// RPCResponse captures both a response and a potential error. +type RPCResponse struct { + Response interface{} + Error error +} + +// RPC has a command, and provides a response mechanism. +type RPC struct { + Command interface{} + Reader io.Reader // Set only for InstallSnapshot + RespChan chan<- RPCResponse +} + +// Respond is used to respond with a response, error or both +func (r *RPC) Respond(resp interface{}, err error) { + r.RespChan <- RPCResponse{resp, err} +} + +// Transport provides an interface for network transports +// to allow Raft to communicate with other nodes. +type Transport interface { + // Consumer returns a channel that can be used to + // consume and respond to RPC requests. + Consumer() <-chan RPC + + // LocalAddr is used to return our local address to distinguish from our peers. + LocalAddr() string + + // AppendEntriesPipeline returns an interface that can be used to pipeline + // AppendEntries requests. + AppendEntriesPipeline(target string) (AppendPipeline, error) + + // AppendEntries sends the appropriate RPC to the target node. + AppendEntries(target string, args *AppendEntriesRequest, resp *AppendEntriesResponse) error + + // RequestVote sends the appropriate RPC to the target node. + RequestVote(target string, args *RequestVoteRequest, resp *RequestVoteResponse) error + + // InstallSnapshot is used to push a snapshot down to a follower. The data is read from + // the ReadCloser and streamed to the client. + InstallSnapshot(target string, args *InstallSnapshotRequest, resp *InstallSnapshotResponse, data io.Reader) error + + // EncodePeer is used to serialize a peer name. + EncodePeer(string) []byte + + // DecodePeer is used to deserialize a peer name. + DecodePeer([]byte) string + + // SetHeartbeatHandler is used to setup a heartbeat handler + // as a fast-pass. This is to avoid head-of-line blocking from + // disk IO. If a Transport does not support this, it can simply + // ignore the call, and push the heartbeat onto the Consumer channel. + SetHeartbeatHandler(cb func(rpc RPC)) +} + +// AppendPipeline is used for pipelining AppendEntries requests. It is used +// to increase the replication throughput by masking latency and better +// utilizing bandwidth. +type AppendPipeline interface { + // AppendEntries is used to add another request to the pipeline. + // The send may block which is an effective form of back-pressure. + AppendEntries(args *AppendEntriesRequest, resp *AppendEntriesResponse) (AppendFuture, error) + + // Consumer returns a channel that can be used to consume + // response futures when they are ready. + Consumer() <-chan AppendFuture + + // Closes pipeline and cancels all inflight RPCs + Close() error +} + +// AppendFuture is used to return information about a pipelined AppendEntries request. +type AppendFuture interface { + Future + Start() time.Time + Request() *AppendEntriesRequest + Response() *AppendEntriesResponse +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/util.go b/Godeps/_workspace/src/github.com/hashicorp/raft/util.go new file mode 100644 index 0000000000000..a6642c4c9e646 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/util.go @@ -0,0 +1,200 @@ +package raft + +import ( + "bytes" + crand "crypto/rand" + "encoding/binary" + "fmt" + "math" + "math/big" + "math/rand" + "time" + + "github.com/hashicorp/go-msgpack/codec" +) + +func init() { + // Ensure we use a high-entropy seed for the psuedo-random generator + rand.Seed(newSeed()) +} + +// returns an int64 from a crypto random source +// can be used to seed a source for a math/rand. +func newSeed() int64 { + r, err := crand.Int(crand.Reader, big.NewInt(math.MaxInt64)) + if err != nil { + panic(fmt.Errorf("failed to read random bytes: %v", err)) + } + return r.Int64() +} + +// randomTimeout returns a value that is between the minVal and 2x minVal. +func randomTimeout(minVal time.Duration) <-chan time.Time { + if minVal == 0 { + return nil + } + extra := (time.Duration(rand.Int63()) % minVal) + return time.After(minVal + extra) +} + +// min returns the minimum. +func min(a, b uint64) uint64 { + if a <= b { + return a + } + return b +} + +// max returns the maximum. +func max(a, b uint64) uint64 { + if a >= b { + return a + } + return b +} + +// generateUUID is used to generate a random UUID. +func generateUUID() string { + buf := make([]byte, 16) + if _, err := crand.Read(buf); err != nil { + panic(fmt.Errorf("failed to read random bytes: %v", err)) + } + + return fmt.Sprintf("%08x-%04x-%04x-%04x-%12x", + buf[0:4], + buf[4:6], + buf[6:8], + buf[8:10], + buf[10:16]) +} + +// asyncNotify is used to do an async channel send to +// a list of channels. This will not block. +func asyncNotify(chans []chan struct{}) { + for _, ch := range chans { + asyncNotifyCh(ch) + } +} + +// asyncNotifyCh is used to do an async channel send +// to a single channel without blocking. +func asyncNotifyCh(ch chan struct{}) { + select { + case ch <- struct{}{}: + default: + } +} + +// asyncNotifyBool is used to do an async notification +// on a bool channel. +func asyncNotifyBool(ch chan bool, v bool) { + select { + case ch <- v: + default: + } +} + +// ExcludePeer is used to exclude a single peer from a list of peers. +func ExcludePeer(peers []string, peer string) []string { + otherPeers := make([]string, 0, len(peers)) + for _, p := range peers { + if p != peer { + otherPeers = append(otherPeers, p) + } + } + return otherPeers +} + +// PeerContained checks if a given peer is contained in a list. +func PeerContained(peers []string, peer string) bool { + for _, p := range peers { + if p == peer { + return true + } + } + return false +} + +// AddUniquePeer is used to add a peer to a list of existing +// peers only if it is not already contained. +func AddUniquePeer(peers []string, peer string) []string { + if PeerContained(peers, peer) { + return peers + } + return append(peers, peer) +} + +// encodePeers is used to serialize a list of peers. +func encodePeers(peers []string, trans Transport) []byte { + // Encode each peer + var encPeers [][]byte + for _, p := range peers { + encPeers = append(encPeers, trans.EncodePeer(p)) + } + + // Encode the entire array + buf, err := encodeMsgPack(encPeers) + if err != nil { + panic(fmt.Errorf("failed to encode peers: %v", err)) + } + + return buf.Bytes() +} + +// decodePeers is used to deserialize a list of peers. +func decodePeers(buf []byte, trans Transport) []string { + // Decode the buffer first + var encPeers [][]byte + if err := decodeMsgPack(buf, &encPeers); err != nil { + panic(fmt.Errorf("failed to decode peers: %v", err)) + } + + // Deserialize each peer + var peers []string + for _, enc := range encPeers { + peers = append(peers, trans.DecodePeer(enc)) + } + + return peers +} + +// Decode reverses the encode operation on a byte slice input. +func decodeMsgPack(buf []byte, out interface{}) error { + r := bytes.NewBuffer(buf) + hd := codec.MsgpackHandle{} + dec := codec.NewDecoder(r, &hd) + return dec.Decode(out) +} + +// Encode writes an encoded object to a new bytes buffer. +func encodeMsgPack(in interface{}) (*bytes.Buffer, error) { + buf := bytes.NewBuffer(nil) + hd := codec.MsgpackHandle{} + enc := codec.NewEncoder(buf, &hd) + err := enc.Encode(in) + return buf, err +} + +// Converts bytes to an integer. +func bytesToUint64(b []byte) uint64 { + return binary.BigEndian.Uint64(b) +} + +// Converts a uint64 to a byte slice. +func uint64ToBytes(u uint64) []byte { + buf := make([]byte, 8) + binary.BigEndian.PutUint64(buf, u) + return buf +} + +// backoff is used to compute an exponential backoff +// duration. Base time is scaled by the current round, +// up to some maximum scale factor. +func backoff(base time.Duration, round, limit uint64) time.Duration { + power := min(round, limit) + for power > 2 { + base *= 2 + power-- + } + return base +} diff --git a/Godeps/_workspace/src/github.com/hashicorp/raft/util_test.go b/Godeps/_workspace/src/github.com/hashicorp/raft/util_test.go new file mode 100644 index 0000000000000..191510972c387 --- /dev/null +++ b/Godeps/_workspace/src/github.com/hashicorp/raft/util_test.go @@ -0,0 +1,187 @@ +package raft + +import ( + "reflect" + "regexp" + "testing" + "time" +) + +func TestRandomTimeout(t *testing.T) { + start := time.Now() + timeout := randomTimeout(time.Millisecond) + + select { + case <-timeout: + diff := time.Now().Sub(start) + if diff < time.Millisecond { + t.Fatalf("fired early") + } + case <-time.After(3 * time.Millisecond): + t.Fatalf("timeout") + } +} + +func TestNewSeed(t *testing.T) { + vals := make(map[int64]bool) + for i := 0; i < 1000; i++ { + seed := newSeed() + if _, exists := vals[seed]; exists { + t.Fatal("newSeed() return a value it'd previously returned") + } + vals[seed] = true + } +} + +func TestRandomTimeout_NoTime(t *testing.T) { + timeout := randomTimeout(0) + if timeout != nil { + t.Fatalf("expected nil channel") + } +} + +func TestMin(t *testing.T) { + if min(1, 1) != 1 { + t.Fatalf("bad min") + } + if min(2, 1) != 1 { + t.Fatalf("bad min") + } + if min(1, 2) != 1 { + t.Fatalf("bad min") + } +} + +func TestMax(t *testing.T) { + if max(1, 1) != 1 { + t.Fatalf("bad max") + } + if max(2, 1) != 2 { + t.Fatalf("bad max") + } + if max(1, 2) != 2 { + t.Fatalf("bad max") + } +} + +func TestGenerateUUID(t *testing.T) { + prev := generateUUID() + for i := 0; i < 100; i++ { + id := generateUUID() + if prev == id { + t.Fatalf("Should get a new ID!") + } + + matched, err := regexp.MatchString( + `[\da-f]{8}-[\da-f]{4}-[\da-f]{4}-[\da-f]{4}-[\da-f]{12}`, id) + if !matched || err != nil { + t.Fatalf("expected match %s %v %s", id, matched, err) + } + } +} + +func TestAsyncNotify(t *testing.T) { + chs := []chan struct{}{ + make(chan struct{}), + make(chan struct{}, 1), + make(chan struct{}, 2), + } + + // Should not block! + asyncNotify(chs) + asyncNotify(chs) + asyncNotify(chs) + + // Try to read + select { + case <-chs[0]: + t.Fatalf("should not have message!") + default: + } + select { + case <-chs[1]: + default: + t.Fatalf("should have message!") + } + select { + case <-chs[2]: + default: + t.Fatalf("should have message!") + } + select { + case <-chs[2]: + default: + t.Fatalf("should have message!") + } +} + +func TestExcludePeer(t *testing.T) { + peers := []string{NewInmemAddr(), NewInmemAddr(), NewInmemAddr()} + peer := peers[2] + + after := ExcludePeer(peers, peer) + if len(after) != 2 { + t.Fatalf("Bad length") + } + if after[0] == peer || after[1] == peer { + t.Fatalf("should not contain peer") + } +} + +func TestPeerContained(t *testing.T) { + peers := []string{NewInmemAddr(), NewInmemAddr(), NewInmemAddr()} + + if !PeerContained(peers, peers[2]) { + t.Fatalf("Expect contained") + } + if PeerContained(peers, NewInmemAddr()) { + t.Fatalf("unexpected contained") + } +} + +func TestAddUniquePeer(t *testing.T) { + peers := []string{NewInmemAddr(), NewInmemAddr(), NewInmemAddr()} + after := AddUniquePeer(peers, peers[2]) + if !reflect.DeepEqual(after, peers) { + t.Fatalf("unexpected append") + } + after = AddUniquePeer(peers, NewInmemAddr()) + if len(after) != 4 { + t.Fatalf("expected append") + } +} + +func TestEncodeDecodePeers(t *testing.T) { + peers := []string{NewInmemAddr(), NewInmemAddr(), NewInmemAddr()} + _, trans := NewInmemTransport() + + // Try to encode/decode + buf := encodePeers(peers, trans) + decoded := decodePeers(buf, trans) + + if !reflect.DeepEqual(peers, decoded) { + t.Fatalf("mismatch %v %v", peers, decoded) + } +} + +func TestBackoff(t *testing.T) { + b := backoff(10*time.Millisecond, 1, 8) + if b != 10*time.Millisecond { + t.Fatalf("bad: %v", b) + } + + b = backoff(20*time.Millisecond, 2, 8) + if b != 20*time.Millisecond { + t.Fatalf("bad: %v", b) + } + + b = backoff(10*time.Millisecond, 8, 8) + if b != 640*time.Millisecond { + t.Fatalf("bad: %v", b) + } + + b = backoff(10*time.Millisecond, 9, 8) + if b != 640*time.Millisecond { + t.Fatalf("bad: %v", b) + } +} diff --git a/Godeps/_workspace/src/github.com/influxdb/influxdb/client/README.md b/Godeps/_workspace/src/github.com/influxdb/influxdb/client/README.md new file mode 100644 index 0000000000000..883941e9007f5 --- /dev/null +++ b/Godeps/_workspace/src/github.com/influxdb/influxdb/client/README.md @@ -0,0 +1,207 @@ +# InfluxDB Client + +[![GoDoc](https://godoc.org/github.com/influxdb/influxdb?status.svg)](http://godoc.org/github.com/influxdb/influxdb/client) + +## Description + +A Go client library written and maintained by the **InfluxDB** team. +This package provides convenience functions to read and write time series data. +It uses the HTTP protocol to communicate with your **InfluxDB** cluster. + + +## Getting Started + +### Connecting To Your Database + +Connecting to an **InfluxDB** database is straightforward. You will need a host +name, a port and the cluster user credentials if applicable. The default port is 8086. +You can customize these settings to your specific installation via the +**InfluxDB** configuration file. + +Thought not necessary for experimentation, you may want to create a new user +and authenticate the connection to your database. + +For more information please check out the +[Cluster Admin Docs](http://influxdb.com/docs/v0.9/query_language/database_administration.html). + +For the impatient, you can create a new admin user _bubba_ by firing off the +[InfluxDB CLI](https://github.com/influxdb/influxdb/blob/master/cmd/influx/main.go). + +```shell +influx +> create user bubba with password 'bumblebeetuna' +> grant all privileges to bubba +``` + +And now for good measure set the credentials in you shell environment. +In the example below we will use $INFLUX_USER and $INFLUX_PWD + +Now with the administrivia out of the way, let's connect to our database. + +NOTE: If you've opted out of creating a user, you can omit Username and Password in +the configuration below. + +```go +package main + +import "github.com/influxdb/influxdb/client" +import "net/url" + +const ( + MyHost = "localhost" + MyPort = 8086 + MyDB = "square_holes" + MyMeasurement = "shapes" +) + +func main() { + u, err := url.Parse(fmt.Sprintf("http://%s:%d", MyHost, MyPort)) + if err != nil { + log.Fatal(err) + } + + conf := client.Config{ + URL: *u, + Username: os.Getenv("INFLUX_USER"), + Password: os.Getenv("INFLUX_PWD"), + } + + con, err := client.NewClient(conf) + if err != nil { + log.Fatal(err) + } + + dur, ver, err := con.Ping() + if err != nil { + log.Fatal(err) + } + log.Printf("Happy as a Hippo! %v, %s", dur, ver) +} + +``` + +### Inserting Data + +Time series data aka *points* are written to the database using batch inserts. +The mechanism is to create one or more points and then create a batch aka *batch points* +and write these to a given database and series. A series is a combination of a +measurement (time/values) and a set of tags. + +In this sample we will create a batch of a 1,000 points. Each point has a time and +a single value as well as 2 tags indicating a shape and color. We write these points +to a database called _square_holes_ using a measurement named _shapes_. + +NOTE: You can specify a RetentionPolicy as part of the batch points. If not +provided InfluxDB will use the database _default_ retention policy. By default, the _default_ +retention policy never deletes any data it contains. + +```go +func writePoints(con *client.Client) { + var ( + shapes = []string{"circle", "rectangle", "square", "triangle"} + colors = []string{"red", "blue", "green"} + sampleSize = 1000 + pts = make([]client.Point, sampleSize) + ) + + rand.Seed(42) + for i := 0; i < sampleSize; i++ { + pts[i] = client.Point{ + Measurement: "shapes", + Tags: map[string]string{ + "color": strconv.Itoa(rand.Intn(len(colors))), + "shape": strconv.Itoa(rand.Intn(len(shapes))), + }, + Fields: map[string]interface{}{ + "value": rand.Intn(sampleSize), + }, + Time: time.Now(), + Precision: "s", + } + } + + bps := client.BatchPoints{ + Points: pts, + Database: MyDB, + RetentionPolicy: "default", + } + _, err := con.Write(bps) + if err != nil { + log.Fatal(err) + } +} +``` + + +### Querying Data + +One nice advantage of using **InfluxDB** the ability to query your data using familiar +SQL constructs. In this example we can create a convenience function to query the database +as follows: + +```go +// queryDB convenience function to query the database +func queryDB(con *client.Client, cmd string) (res []client.Result, err error) { + q := client.Query{ + Command: cmd, + Database: MyDB, + } + if response, err := con.Query(q); err == nil { + if response.Error() != nil { + return res, response.Error() + } + res = response.Results + } + return +} +``` + +#### Creating a Database +```go +_, err := queryDB(con, fmt.Sprintf("create database %s", MyDB)) +if err != nil { + log.Fatal(err) +} +``` + +#### Count Records +```go +q := fmt.Sprintf("select count(%s) from %s", "value", MyMeasurement) +res, err := queryDB(con, q) +if err != nil { + log.Fatal(err) +} +count := res[0].Series[0].Values[0][1] +log.Printf("Found a total of `%v records", count) + +``` + +#### Find the last 10 _shapes_ records + +```go +q := fmt.Sprintf("select * from %s limit %d", MyMeasurement, 20) +res, err = queryDB(con, q) +if err != nil { + log.Fatal(err) +} + +for i, row := range res[0].Series[0].Values { + t, err := time.Parse(time.RFC3339, row[0].(string)) + if err != nil { + log.Fatal(err) + } + val, err := row[1].(json.Number).Int64() + log.Printf("[%2d] %s: %03d\n", i, t.Format(time.Stamp), val) +} +``` + +## Go Docs + +Please refer to +[http://godoc.org/github.com/influxdb/influxdb/client](http://godoc.org/github.com/influxdb/influxdb/client) +for documentation. + +## See Also + +You can also examine how the client library is used by the +[InfluxDB CLI](https://github.com/influxdb/influxdb/blob/master/cmd/influx/main.go). diff --git a/Godeps/_workspace/src/github.com/influxdb/influxdb/client/example_test.go b/Godeps/_workspace/src/github.com/influxdb/influxdb/client/example_test.go new file mode 100644 index 0000000000000..58805ceeaa3b3 --- /dev/null +++ b/Godeps/_workspace/src/github.com/influxdb/influxdb/client/example_test.go @@ -0,0 +1,113 @@ +package client_test + +import ( + "fmt" + "log" + "math/rand" + "net/url" + "os" + "strconv" + "time" + + "github.com/influxdb/influxdb/client" +) + +func ExampleNewClient() { + host, err := url.Parse(fmt.Sprintf("http://%s:%d", "localhost", 8086)) + if err != nil { + log.Fatal(err) + } + + // NOTE: this assumes you've setup a user and have setup shell env variables, + // namely INFLUX_USER/INFLUX_PWD. If not just ommit Username/Password below. + conf := client.Config{ + URL: *host, + Username: os.Getenv("INFLUX_USER"), + Password: os.Getenv("INFLUX_PWD"), + } + con, err := client.NewClient(conf) + if err != nil { + log.Fatal(err) + } + log.Println("Connection", con) +} + +func ExampleClient_Ping() { + host, err := url.Parse(fmt.Sprintf("http://%s:%d", "localhost", 8086)) + if err != nil { + log.Fatal(err) + } + con, err := client.NewClient(client.Config{URL: *host}) + if err != nil { + log.Fatal(err) + } + + dur, ver, err := con.Ping() + if err != nil { + log.Fatal(err) + } + log.Printf("Happy as a hippo! %v, %s", dur, ver) +} + +func ExampleClient_Query() { + host, err := url.Parse(fmt.Sprintf("http://%s:%d", "localhost", 8086)) + if err != nil { + log.Fatal(err) + } + con, err := client.NewClient(client.Config{URL: *host}) + if err != nil { + log.Fatal(err) + } + + q := client.Query{ + Command: "select count(value) from shapes", + Database: "square_holes", + } + if response, err := con.Query(q); err == nil && response.Error() == nil { + log.Println(response.Results) + } +} + +func ExampleClient_Write() { + host, err := url.Parse(fmt.Sprintf("http://%s:%d", "localhost", 8086)) + if err != nil { + log.Fatal(err) + } + con, err := client.NewClient(client.Config{URL: *host}) + if err != nil { + log.Fatal(err) + } + + var ( + shapes = []string{"circle", "rectangle", "square", "triangle"} + colors = []string{"red", "blue", "green"} + sampleSize = 1000 + pts = make([]client.Point, sampleSize) + ) + + rand.Seed(42) + for i := 0; i < sampleSize; i++ { + pts[i] = client.Point{ + Measurement: "shapes", + Tags: map[string]string{ + "color": strconv.Itoa(rand.Intn(len(colors))), + "shape": strconv.Itoa(rand.Intn(len(shapes))), + }, + Fields: map[string]interface{}{ + "value": rand.Intn(sampleSize), + }, + Time: time.Now(), + Precision: "s", + } + } + + bps := client.BatchPoints{ + Points: pts, + Database: "BumbeBeeTuna", + RetentionPolicy: "default", + } + _, err = con.Write(bps) + if err != nil { + log.Fatal(err) + } +} diff --git a/Godeps/_workspace/src/github.com/influxdb/influxdb/client/influxdb.go b/Godeps/_workspace/src/github.com/influxdb/influxdb/client/influxdb.go new file mode 100644 index 0000000000000..c4f34d84c5f19 --- /dev/null +++ b/Godeps/_workspace/src/github.com/influxdb/influxdb/client/influxdb.go @@ -0,0 +1,656 @@ +package client + +import ( + "bytes" + "encoding/json" + "errors" + "fmt" + "io/ioutil" + "net" + "net/http" + "net/url" + "strconv" + "strings" + "time" + + "github.com/influxdb/influxdb/influxql" + "github.com/influxdb/influxdb/tsdb" +) + +const ( + // DefaultHost is the default host used to connect to an InfluxDB instance + DefaultHost = "localhost" + + // DefaultPort is the default port used to connect to an InfluxDB instance + DefaultPort = 8086 + + // DefaultTimeout is the default connection timeout used to connect to an InfluxDB instance + DefaultTimeout = 0 +) + +// Query is used to send a command to the server. Both Command and Database are required. +type Query struct { + Command string + Database string +} + +// ParseConnectionString will parse a string to create a valid connection URL +func ParseConnectionString(path string, ssl bool) (url.URL, error) { + var host string + var port int + + if strings.Contains(path, ":") { + h := strings.Split(path, ":") + i, e := strconv.Atoi(h[1]) + if e != nil { + return url.URL{}, fmt.Errorf("invalid port number %q: %s\n", path, e) + } + port = i + if h[0] == "" { + host = DefaultHost + } else { + host = h[0] + } + } else { + host = path + // If they didn't specify a port, always use the default port + port = DefaultPort + } + + u := url.URL{ + Scheme: "http", + } + if ssl { + u.Scheme = "https" + } + u.Host = net.JoinHostPort(host, strconv.Itoa(port)) + + return u, nil +} + +// Config is used to specify what server to connect to. +// URL: The URL of the server connecting to. +// Username/Password are optional. They will be passed via basic auth if provided. +// UserAgent: If not provided, will default "InfluxDBClient", +// Timeout: If not provided, will default to 0 (no timeout) +type Config struct { + URL url.URL + Username string + Password string + UserAgent string + Timeout time.Duration +} + +// NewConfig will create a config to be used in connecting to the client +func NewConfig() Config { + return Config{ + Timeout: DefaultTimeout, + } +} + +// Client is used to make calls to the server. +type Client struct { + url url.URL + username string + password string + httpClient *http.Client + userAgent string +} + +const ( + ConsistencyOne = "one" + ConsistencyAll = "all" + ConsistencyQuorum = "quorum" + ConsistencyAny = "any" +) + +// NewClient will instantiate and return a connected client to issue commands to the server. +func NewClient(c Config) (*Client, error) { + client := Client{ + url: c.URL, + username: c.Username, + password: c.Password, + httpClient: &http.Client{Timeout: c.Timeout}, + userAgent: c.UserAgent, + } + if client.userAgent == "" { + client.userAgent = "InfluxDBClient" + } + return &client, nil +} + +// SetAuth will update the username and passwords +func (c *Client) SetAuth(u, p string) { + c.username = u + c.password = p +} + +// Query sends a command to the server and returns the Response +func (c *Client) Query(q Query) (*Response, error) { + u := c.url + + u.Path = "query" + values := u.Query() + values.Set("q", q.Command) + values.Set("db", q.Database) + u.RawQuery = values.Encode() + + req, err := http.NewRequest("GET", u.String(), nil) + if err != nil { + return nil, err + } + req.Header.Set("User-Agent", c.userAgent) + if c.username != "" { + req.SetBasicAuth(c.username, c.password) + } + + resp, err := c.httpClient.Do(req) + if err != nil { + return nil, err + } + defer resp.Body.Close() + + var response Response + dec := json.NewDecoder(resp.Body) + dec.UseNumber() + decErr := dec.Decode(&response) + + // ignore this error if we got an invalid status code + if decErr != nil && decErr.Error() == "EOF" && resp.StatusCode != http.StatusOK { + decErr = nil + } + // If we got a valid decode error, send that back + if decErr != nil { + return nil, decErr + } + // If we don't have an error in our json response, and didn't get statusOK, then send back an error + if resp.StatusCode != http.StatusOK && response.Error() == nil { + return &response, fmt.Errorf("received status code %d from server", resp.StatusCode) + } + return &response, nil +} + +// Write takes BatchPoints and allows for writing of multiple points with defaults +// If successful, error is nil and Response is nil +// If an error occurs, Response may contain additional information if populated. +func (c *Client) Write(bp BatchPoints) (*Response, error) { + u := c.url + u.Path = "write" + + var b bytes.Buffer + for _, p := range bp.Points { + if p.Raw != "" { + if _, err := b.WriteString(p.Raw); err != nil { + return nil, err + } + } else { + for k, v := range bp.Tags { + if p.Tags == nil { + p.Tags = make(map[string]string, len(bp.Tags)) + } + p.Tags[k] = v + } + + if _, err := b.WriteString(p.MarshalString()); err != nil { + return nil, err + } + } + + if err := b.WriteByte('\n'); err != nil { + return nil, err + } + } + + req, err := http.NewRequest("POST", u.String(), &b) + if err != nil { + return nil, err + } + req.Header.Set("Content-Type", "") + req.Header.Set("User-Agent", c.userAgent) + if c.username != "" { + req.SetBasicAuth(c.username, c.password) + } + params := req.URL.Query() + params.Set("db", bp.Database) + params.Set("rp", bp.RetentionPolicy) + params.Set("precision", bp.Precision) + params.Set("consistency", bp.WriteConsistency) + req.URL.RawQuery = params.Encode() + + resp, err := c.httpClient.Do(req) + if err != nil { + return nil, err + } + defer resp.Body.Close() + + var response Response + body, err := ioutil.ReadAll(resp.Body) + if err != nil { + return nil, err + } + + if resp.StatusCode != http.StatusNoContent && resp.StatusCode != http.StatusOK { + var err = fmt.Errorf(string(body)) + response.Err = err + return &response, err + } + + return nil, nil +} + +// WriteLineProtocol takes a string with line returns to delimit each write +// If successful, error is nil and Response is nil +// If an error occurs, Response may contain additional information if populated. +func (c *Client) WriteLineProtocol(data, database, retentionPolicy, precision, writeConsistency string) (*Response, error) { + u := c.url + u.Path = "write" + + r := strings.NewReader(data) + + req, err := http.NewRequest("POST", u.String(), r) + if err != nil { + return nil, err + } + req.Header.Set("Content-Type", "") + req.Header.Set("User-Agent", c.userAgent) + if c.username != "" { + req.SetBasicAuth(c.username, c.password) + } + params := req.URL.Query() + params.Set("db", database) + params.Set("rp", retentionPolicy) + params.Set("precision", precision) + params.Set("consistency", writeConsistency) + req.URL.RawQuery = params.Encode() + + resp, err := c.httpClient.Do(req) + if err != nil { + return nil, err + } + defer resp.Body.Close() + + var response Response + body, err := ioutil.ReadAll(resp.Body) + if err != nil { + return nil, err + } + + if resp.StatusCode != http.StatusNoContent && resp.StatusCode != http.StatusOK { + err := fmt.Errorf(string(body)) + response.Err = err + return &response, err + } + + return nil, nil +} + +// Ping will check to see if the server is up +// Ping returns how long the request took, the version of the server it connected to, and an error if one occurred. +func (c *Client) Ping() (time.Duration, string, error) { + now := time.Now() + u := c.url + u.Path = "ping" + + req, err := http.NewRequest("GET", u.String(), nil) + if err != nil { + return 0, "", err + } + req.Header.Set("User-Agent", c.userAgent) + if c.username != "" { + req.SetBasicAuth(c.username, c.password) + } + + resp, err := c.httpClient.Do(req) + if err != nil { + return 0, "", err + } + defer resp.Body.Close() + + version := resp.Header.Get("X-Influxdb-Version") + return time.Since(now), version, nil +} + +// Structs + +// Result represents a resultset returned from a single statement. +type Result struct { + Series []influxql.Row + Err error +} + +// MarshalJSON encodes the result into JSON. +func (r *Result) MarshalJSON() ([]byte, error) { + // Define a struct that outputs "error" as a string. + var o struct { + Series []influxql.Row `json:"series,omitempty"` + Err string `json:"error,omitempty"` + } + + // Copy fields to output struct. + o.Series = r.Series + if r.Err != nil { + o.Err = r.Err.Error() + } + + return json.Marshal(&o) +} + +// UnmarshalJSON decodes the data into the Result struct +func (r *Result) UnmarshalJSON(b []byte) error { + var o struct { + Series []influxql.Row `json:"series,omitempty"` + Err string `json:"error,omitempty"` + } + + dec := json.NewDecoder(bytes.NewBuffer(b)) + dec.UseNumber() + err := dec.Decode(&o) + if err != nil { + return err + } + r.Series = o.Series + if o.Err != "" { + r.Err = errors.New(o.Err) + } + return nil +} + +// Response represents a list of statement results. +type Response struct { + Results []Result + Err error +} + +// MarshalJSON encodes the response into JSON. +func (r *Response) MarshalJSON() ([]byte, error) { + // Define a struct that outputs "error" as a string. + var o struct { + Results []Result `json:"results,omitempty"` + Err string `json:"error,omitempty"` + } + + // Copy fields to output struct. + o.Results = r.Results + if r.Err != nil { + o.Err = r.Err.Error() + } + + return json.Marshal(&o) +} + +// UnmarshalJSON decodes the data into the Response struct +func (r *Response) UnmarshalJSON(b []byte) error { + var o struct { + Results []Result `json:"results,omitempty"` + Err string `json:"error,omitempty"` + } + + dec := json.NewDecoder(bytes.NewBuffer(b)) + dec.UseNumber() + err := dec.Decode(&o) + if err != nil { + return err + } + r.Results = o.Results + if o.Err != "" { + r.Err = errors.New(o.Err) + } + return nil +} + +// Error returns the first error from any statement. +// Returns nil if no errors occurred on any statements. +func (r Response) Error() error { + if r.Err != nil { + return r.Err + } + for _, result := range r.Results { + if result.Err != nil { + return result.Err + } + } + return nil +} + +// Point defines the fields that will be written to the database +// Measurement, Time, and Fields are required +// Precision can be specified if the time is in epoch format (integer). +// Valid values for Precision are n, u, ms, s, m, and h +type Point struct { + Measurement string + Tags map[string]string + Time time.Time + Fields map[string]interface{} + Precision string + Raw string +} + +// MarshalJSON will format the time in RFC3339Nano +// Precision is also ignored as it is only used for writing, not reading +// Or another way to say it is we always send back in nanosecond precision +func (p *Point) MarshalJSON() ([]byte, error) { + point := struct { + Measurement string `json:"measurement,omitempty"` + Tags map[string]string `json:"tags,omitempty"` + Time string `json:"time,omitempty"` + Fields map[string]interface{} `json:"fields,omitempty"` + Precision string `json:"precision,omitempty"` + }{ + Measurement: p.Measurement, + Tags: p.Tags, + Fields: p.Fields, + Precision: p.Precision, + } + // Let it omit empty if it's really zero + if !p.Time.IsZero() { + point.Time = p.Time.UTC().Format(time.RFC3339Nano) + } + return json.Marshal(&point) +} + +func (p *Point) MarshalString() string { + return tsdb.NewPoint(p.Measurement, p.Tags, p.Fields, p.Time).String() +} + +// UnmarshalJSON decodes the data into the Point struct +func (p *Point) UnmarshalJSON(b []byte) error { + var normal struct { + Measurement string `json:"measurement"` + Tags map[string]string `json:"tags"` + Time time.Time `json:"time"` + Precision string `json:"precision"` + Fields map[string]interface{} `json:"fields"` + } + var epoch struct { + Measurement string `json:"measurement"` + Tags map[string]string `json:"tags"` + Time *int64 `json:"time"` + Precision string `json:"precision"` + Fields map[string]interface{} `json:"fields"` + } + + if err := func() error { + var err error + dec := json.NewDecoder(bytes.NewBuffer(b)) + dec.UseNumber() + if err = dec.Decode(&epoch); err != nil { + return err + } + // Convert from epoch to time.Time, but only if Time + // was actually set. + var ts time.Time + if epoch.Time != nil { + ts, err = EpochToTime(*epoch.Time, epoch.Precision) + if err != nil { + return err + } + } + p.Measurement = epoch.Measurement + p.Tags = epoch.Tags + p.Time = ts + p.Precision = epoch.Precision + p.Fields = normalizeFields(epoch.Fields) + return nil + }(); err == nil { + return nil + } + + dec := json.NewDecoder(bytes.NewBuffer(b)) + dec.UseNumber() + if err := dec.Decode(&normal); err != nil { + return err + } + normal.Time = SetPrecision(normal.Time, normal.Precision) + p.Measurement = normal.Measurement + p.Tags = normal.Tags + p.Time = normal.Time + p.Precision = normal.Precision + p.Fields = normalizeFields(normal.Fields) + + return nil +} + +// Remove any notion of json.Number +func normalizeFields(fields map[string]interface{}) map[string]interface{} { + newFields := map[string]interface{}{} + + for k, v := range fields { + switch v := v.(type) { + case json.Number: + jv, e := v.Float64() + if e != nil { + panic(fmt.Sprintf("unable to convert json.Number to float64: %s", e)) + } + newFields[k] = jv + default: + newFields[k] = v + } + } + return newFields +} + +// BatchPoints is used to send batched data in a single write. +// Database and Points are required +// If no retention policy is specified, it will use the databases default retention policy. +// If tags are specified, they will be "merged" with all points. If a point already has that tag, it is ignored. +// If time is specified, it will be applied to any point with an empty time. +// Precision can be specified if the time is in epoch format (integer). +// Valid values for Precision are n, u, ms, s, m, and h +type BatchPoints struct { + Points []Point `json:"points,omitempty"` + Database string `json:"database,omitempty"` + RetentionPolicy string `json:"retentionPolicy,omitempty"` + Tags map[string]string `json:"tags,omitempty"` + Time time.Time `json:"time,omitempty"` + Precision string `json:"precision,omitempty"` + WriteConsistency string `json:"-"` +} + +// UnmarshalJSON decodes the data into the BatchPoints struct +func (bp *BatchPoints) UnmarshalJSON(b []byte) error { + var normal struct { + Points []Point `json:"points"` + Database string `json:"database"` + RetentionPolicy string `json:"retentionPolicy"` + Tags map[string]string `json:"tags"` + Time time.Time `json:"time"` + Precision string `json:"precision"` + } + var epoch struct { + Points []Point `json:"points"` + Database string `json:"database"` + RetentionPolicy string `json:"retentionPolicy"` + Tags map[string]string `json:"tags"` + Time *int64 `json:"time"` + Precision string `json:"precision"` + } + + if err := func() error { + var err error + if err = json.Unmarshal(b, &epoch); err != nil { + return err + } + // Convert from epoch to time.Time + var ts time.Time + if epoch.Time != nil { + ts, err = EpochToTime(*epoch.Time, epoch.Precision) + if err != nil { + return err + } + } + bp.Points = epoch.Points + bp.Database = epoch.Database + bp.RetentionPolicy = epoch.RetentionPolicy + bp.Tags = epoch.Tags + bp.Time = ts + bp.Precision = epoch.Precision + return nil + }(); err == nil { + return nil + } + + if err := json.Unmarshal(b, &normal); err != nil { + return err + } + normal.Time = SetPrecision(normal.Time, normal.Precision) + bp.Points = normal.Points + bp.Database = normal.Database + bp.RetentionPolicy = normal.RetentionPolicy + bp.Tags = normal.Tags + bp.Time = normal.Time + bp.Precision = normal.Precision + + return nil +} + +// utility functions + +// Addr provides the current url as a string of the server the client is connected to. +func (c *Client) Addr() string { + return c.url.String() +} + +// helper functions + +// EpochToTime takes a unix epoch time and uses precision to return back a time.Time +func EpochToTime(epoch int64, precision string) (time.Time, error) { + if precision == "" { + precision = "s" + } + var t time.Time + switch precision { + case "h": + t = time.Unix(0, epoch*int64(time.Hour)) + case "m": + t = time.Unix(0, epoch*int64(time.Minute)) + case "s": + t = time.Unix(0, epoch*int64(time.Second)) + case "ms": + t = time.Unix(0, epoch*int64(time.Millisecond)) + case "u": + t = time.Unix(0, epoch*int64(time.Microsecond)) + case "n": + t = time.Unix(0, epoch) + default: + return time.Time{}, fmt.Errorf("Unknown precision %q", precision) + } + return t, nil +} + +// SetPrecision will round a time to the specified precision +func SetPrecision(t time.Time, precision string) time.Time { + switch precision { + case "n": + case "u": + return t.Round(time.Microsecond) + case "ms": + return t.Round(time.Millisecond) + case "s": + return t.Round(time.Second) + case "m": + return t.Round(time.Minute) + case "h": + return t.Round(time.Hour) + } + return t +} diff --git a/Godeps/_workspace/src/github.com/influxdb/influxdb/client/influxdb_test.go b/Godeps/_workspace/src/github.com/influxdb/influxdb/client/influxdb_test.go new file mode 100644 index 0000000000000..0a6df042eeacb --- /dev/null +++ b/Godeps/_workspace/src/github.com/influxdb/influxdb/client/influxdb_test.go @@ -0,0 +1,530 @@ +package client_test + +import ( + "encoding/json" + "fmt" + "net/http" + "net/http/httptest" + "net/url" + "strings" + "testing" + "time" + + "github.com/influxdb/influxdb/client" +) + +func BenchmarkUnmarshalJSON2Tags(b *testing.B) { + var bp client.BatchPoints + data := []byte(` +{ + "database": "foo", + "retentionPolicy": "bar", + "points": [ + { + "name": "cpu", + "tags": { + "host": "server01", + "region": "us-east1" + }, + "time": 14244733039069373, + "precision": "n", + "fields": { + "value": 4541770385657154000 + } + } + ] +} +`) + + for i := 0; i < b.N; i++ { + if err := json.Unmarshal(data, &bp); err != nil { + b.Errorf("unable to unmarshal nanosecond data: %s", err.Error()) + } + b.SetBytes(int64(len(data))) + } +} + +func BenchmarkUnmarshalJSON10Tags(b *testing.B) { + var bp client.BatchPoints + data := []byte(` +{ + "database": "foo", + "retentionPolicy": "bar", + "points": [ + { + "name": "cpu", + "tags": { + "host": "server01", + "region": "us-east1", + "tag1": "value1", + "tag2": "value2", + "tag2": "value3", + "tag4": "value4", + "tag5": "value5", + "tag6": "value6", + "tag7": "value7", + "tag8": "value8" + }, + "time": 14244733039069373, + "precision": "n", + "fields": { + "value": 4541770385657154000 + } + } + ] +} +`) + + for i := 0; i < b.N; i++ { + if err := json.Unmarshal(data, &bp); err != nil { + b.Errorf("unable to unmarshal nanosecond data: %s", err.Error()) + } + b.SetBytes(int64(len(data))) + } +} + +func TestNewClient(t *testing.T) { + config := client.Config{} + _, err := client.NewClient(config) + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } +} + +func TestClient_Ping(t *testing.T) { + ts := emptyTestServer() + defer ts.Close() + + u, _ := url.Parse(ts.URL) + config := client.Config{URL: *u} + c, err := client.NewClient(config) + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } + d, version, err := c.Ping() + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } + if d == 0 { + t.Fatalf("expected a duration greater than zero. actual %v", d) + } + if version != "x.x" { + t.Fatalf("unexpected version. expected %s, actual %v", "x.x", version) + } +} + +func TestClient_Query(t *testing.T) { + ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + var data client.Response + w.WriteHeader(http.StatusOK) + _ = json.NewEncoder(w).Encode(data) + })) + defer ts.Close() + + u, _ := url.Parse(ts.URL) + config := client.Config{URL: *u} + c, err := client.NewClient(config) + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } + + query := client.Query{} + _, err = c.Query(query) + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } +} + +func TestClient_BasicAuth(t *testing.T) { + ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + u, p, ok := r.BasicAuth() + + if !ok { + t.Errorf("basic auth error") + } + if u != "username" { + t.Errorf("unexpected username, expected %q, actual %q", "username", u) + } + if p != "password" { + t.Errorf("unexpected password, expected %q, actual %q", "password", p) + } + w.WriteHeader(http.StatusNoContent) + })) + defer ts.Close() + + u, _ := url.Parse(ts.URL) + u.User = url.UserPassword("username", "password") + config := client.Config{URL: *u, Username: "username", Password: "password"} + c, err := client.NewClient(config) + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } + + _, _, err = c.Ping() + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } +} + +func TestClient_Write(t *testing.T) { + ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + var data client.Response + w.WriteHeader(http.StatusNoContent) + _ = json.NewEncoder(w).Encode(data) + })) + defer ts.Close() + + u, _ := url.Parse(ts.URL) + config := client.Config{URL: *u} + c, err := client.NewClient(config) + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } + + bp := client.BatchPoints{} + r, err := c.Write(bp) + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } + if r != nil { + t.Fatalf("unexpected response. expected %v, actual %v", nil, r) + } +} + +func TestClient_UserAgent(t *testing.T) { + receivedUserAgent := "" + ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + receivedUserAgent = r.UserAgent() + + var data client.Response + w.WriteHeader(http.StatusOK) + _ = json.NewEncoder(w).Encode(data) + })) + defer ts.Close() + + _, err := http.Get(ts.URL) + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } + + tests := []struct { + name string + userAgent string + expected string + }{ + { + name: "Empty user agent", + userAgent: "", + expected: "InfluxDBClient", + }, + { + name: "Custom user agent", + userAgent: "Test Influx Client", + expected: "Test Influx Client", + }, + } + + for _, test := range tests { + u, _ := url.Parse(ts.URL) + config := client.Config{URL: *u, UserAgent: test.userAgent} + c, err := client.NewClient(config) + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } + + receivedUserAgent = "" + query := client.Query{} + _, err = c.Query(query) + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } + if !strings.HasPrefix(receivedUserAgent, test.expected) { + t.Fatalf("Unexpected user agent. expected %v, actual %v", test.expected, receivedUserAgent) + } + + receivedUserAgent = "" + bp := client.BatchPoints{} + _, err = c.Write(bp) + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } + if !strings.HasPrefix(receivedUserAgent, test.expected) { + t.Fatalf("Unexpected user agent. expected %v, actual %v", test.expected, receivedUserAgent) + } + + receivedUserAgent = "" + _, _, err = c.Ping() + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } + if receivedUserAgent != test.expected { + t.Fatalf("Unexpected user agent. expected %v, actual %v", test.expected, receivedUserAgent) + } + } +} + +func TestPoint_UnmarshalEpoch(t *testing.T) { + now := time.Now() + tests := []struct { + name string + epoch int64 + precision string + expected time.Time + }{ + { + name: "nanoseconds", + epoch: now.UnixNano(), + precision: "n", + expected: now, + }, + { + name: "microseconds", + epoch: now.Round(time.Microsecond).UnixNano() / int64(time.Microsecond), + precision: "u", + expected: now.Round(time.Microsecond), + }, + { + name: "milliseconds", + epoch: now.Round(time.Millisecond).UnixNano() / int64(time.Millisecond), + precision: "ms", + expected: now.Round(time.Millisecond), + }, + { + name: "seconds", + epoch: now.Round(time.Second).UnixNano() / int64(time.Second), + precision: "s", + expected: now.Round(time.Second), + }, + { + name: "minutes", + epoch: now.Round(time.Minute).UnixNano() / int64(time.Minute), + precision: "m", + expected: now.Round(time.Minute), + }, + { + name: "hours", + epoch: now.Round(time.Hour).UnixNano() / int64(time.Hour), + precision: "h", + expected: now.Round(time.Hour), + }, + { + name: "max int64", + epoch: 9223372036854775807, + precision: "n", + expected: time.Unix(0, 9223372036854775807), + }, + { + name: "100 years from now", + epoch: now.Add(time.Hour * 24 * 365 * 100).UnixNano(), + precision: "n", + expected: now.Add(time.Hour * 24 * 365 * 100), + }, + } + + for _, test := range tests { + t.Logf("testing %q\n", test.name) + data := []byte(fmt.Sprintf(`{"time": %d, "precision":"%s"}`, test.epoch, test.precision)) + t.Logf("json: %s", string(data)) + var p client.Point + err := json.Unmarshal(data, &p) + if err != nil { + t.Fatalf("unexpected error. exptected: %v, actual: %v", nil, err) + } + if !p.Time.Equal(test.expected) { + t.Fatalf("Unexpected time. expected: %v, actual: %v", test.expected, p.Time) + } + } +} + +func TestPoint_UnmarshalRFC(t *testing.T) { + now := time.Now().UTC() + tests := []struct { + name string + rfc string + now time.Time + expected time.Time + }{ + { + name: "RFC3339Nano", + rfc: time.RFC3339Nano, + now: now, + expected: now, + }, + { + name: "RFC3339", + rfc: time.RFC3339, + now: now.Round(time.Second), + expected: now.Round(time.Second), + }, + } + + for _, test := range tests { + t.Logf("testing %q\n", test.name) + ts := test.now.Format(test.rfc) + data := []byte(fmt.Sprintf(`{"time": %q}`, ts)) + t.Logf("json: %s", string(data)) + var p client.Point + err := json.Unmarshal(data, &p) + if err != nil { + t.Fatalf("unexpected error. exptected: %v, actual: %v", nil, err) + } + if !p.Time.Equal(test.expected) { + t.Fatalf("Unexpected time. expected: %v, actual: %v", test.expected, p.Time) + } + } +} + +func TestPoint_MarshalOmitempty(t *testing.T) { + now := time.Now().UTC() + tests := []struct { + name string + point client.Point + now time.Time + expected string + }{ + { + name: "all empty", + point: client.Point{Measurement: "cpu", Fields: map[string]interface{}{"value": 1.1}}, + now: now, + expected: `{"measurement":"cpu","fields":{"value":1.1}}`, + }, + { + name: "with time", + point: client.Point{Measurement: "cpu", Fields: map[string]interface{}{"value": 1.1}, Time: now}, + now: now, + expected: fmt.Sprintf(`{"measurement":"cpu","time":"%s","fields":{"value":1.1}}`, now.Format(time.RFC3339Nano)), + }, + { + name: "with tags", + point: client.Point{Measurement: "cpu", Fields: map[string]interface{}{"value": 1.1}, Tags: map[string]string{"foo": "bar"}}, + now: now, + expected: `{"measurement":"cpu","tags":{"foo":"bar"},"fields":{"value":1.1}}`, + }, + { + name: "with precision", + point: client.Point{Measurement: "cpu", Fields: map[string]interface{}{"value": 1.1}, Precision: "ms"}, + now: now, + expected: `{"measurement":"cpu","fields":{"value":1.1},"precision":"ms"}`, + }, + } + + for _, test := range tests { + t.Logf("testing %q\n", test.name) + b, err := json.Marshal(&test.point) + if err != nil { + t.Fatalf("unexpected error. exptected: %v, actual: %v", nil, err) + } + if test.expected != string(b) { + t.Fatalf("Unexpected result. expected: %v, actual: %v", test.expected, string(b)) + } + } +} + +func TestEpochToTime(t *testing.T) { + now := time.Now() + + tests := []struct { + name string + epoch int64 + precision string + expected time.Time + }{ + {name: "nanoseconds", epoch: now.UnixNano(), precision: "n", expected: now}, + {name: "microseconds", epoch: now.Round(time.Microsecond).UnixNano() / int64(time.Microsecond), precision: "u", expected: now.Round(time.Microsecond)}, + {name: "milliseconds", epoch: now.Round(time.Millisecond).UnixNano() / int64(time.Millisecond), precision: "ms", expected: now.Round(time.Millisecond)}, + {name: "seconds", epoch: now.Round(time.Second).UnixNano() / int64(time.Second), precision: "s", expected: now.Round(time.Second)}, + {name: "minutes", epoch: now.Round(time.Minute).UnixNano() / int64(time.Minute), precision: "m", expected: now.Round(time.Minute)}, + {name: "hours", epoch: now.Round(time.Hour).UnixNano() / int64(time.Hour), precision: "h", expected: now.Round(time.Hour)}, + } + + for _, test := range tests { + t.Logf("testing %q\n", test.name) + tm, e := client.EpochToTime(test.epoch, test.precision) + if e != nil { + t.Fatalf("unexpected error: expected %v, actual: %v", nil, e) + } + if tm != test.expected { + t.Fatalf("unexpected time: expected %v, actual %v", test.expected, tm) + } + } +} + +// helper functions + +func emptyTestServer() *httptest.Server { + return httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + w.Header().Set("X-Influxdb-Version", "x.x") + return + })) +} + +// Ensure that data with epoch times can be decoded. +func TestBatchPoints_Normal(t *testing.T) { + var bp client.BatchPoints + data := []byte(` +{ + "database": "foo", + "retentionPolicy": "bar", + "points": [ + { + "name": "cpu", + "tags": { + "host": "server01" + }, + "time": 14244733039069373, + "precision": "n", + "values": { + "value": 4541770385657154000 + } + }, + { + "name": "cpu", + "tags": { + "host": "server01" + }, + "time": 14244733039069380, + "precision": "n", + "values": { + "value": 7199311900554737000 + } + } + ] +} +`) + + if err := json.Unmarshal(data, &bp); err != nil { + t.Errorf("unable to unmarshal nanosecond data: %s", err.Error()) + } +} + +func TestClient_Timeout(t *testing.T) { + ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + time.Sleep(1 * time.Second) + var data client.Response + w.WriteHeader(http.StatusOK) + _ = json.NewEncoder(w).Encode(data) + })) + defer ts.Close() + + u, _ := url.Parse(ts.URL) + config := client.Config{URL: *u, Timeout: 500 * time.Millisecond} + c, err := client.NewClient(config) + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } + + query := client.Query{} + _, err = c.Query(query) + if err == nil { + t.Fatalf("unexpected success. expected timeout error") + } else if !strings.Contains(err.Error(), "use of closed network connection") { + t.Fatalf("unexpected error. expected 'use of closed network connection' error, got %v", err) + } + + confignotimeout := client.Config{URL: *u} + cnotimeout, err := client.NewClient(confignotimeout) + _, err = cnotimeout.Query(query) + if err != nil { + t.Fatalf("unexpected error. expected %v, actual %v", nil, err) + } +} diff --git a/Godeps/_workspace/src/github.com/influxdb/influxdb/influxql/INFLUXQL.md b/Godeps/_workspace/src/github.com/influxdb/influxdb/influxql/INFLUXQL.md new file mode 100644 index 0000000000000..3ef272f4177ec --- /dev/null +++ b/Godeps/_workspace/src/github.com/influxdb/influxdb/influxql/INFLUXQL.md @@ -0,0 +1,650 @@ +# The Influx Query Language Specification + +## Introduction + +This is a reference for the Influx Query Language ("InfluxQL"). + +InfluxQL is a SQL-like query language for interacting with InfluxDB. It has been lovingly crafted to feel familiar to those coming from other SQL or SQL-like environments while providing features specific to storing and analyzing time series data. + +## Notation + +The syntax is specified using Extended Backus-Naur Form ("EBNF"). EBNF is the same notation used in the [Go](http://golang.org) programming language specification, which can be found [here](https://golang.org/ref/spec). Not so coincidentally, InfluxDB is written in Go. + +``` +Production = production_name "=" [ Expression ] "." . +Expression = Alternative { "|" Alternative } . +Alternative = Term { Term } . +Term = production_name | token [ "…" token ] | Group | Option | Repetition . +Group = "(" Expression ")" . +Option = "[" Expression "]" . +Repetition = "{" Expression "}" . +``` + +Notation operators in order of increasing precedence: + +``` +| alternation +() grouping +[] option (0 or 1 times) +{} repetition (0 to n times) +``` + +## Query representation + +### Characters + +InfluxQL is Unicode text encoded in [UTF-8](http://en.wikipedia.org/wiki/UTF-8). + +``` +newline = /* the Unicode code point U+000A */ . +unicode_char = /* an arbitrary Unicode code point except newline */ . +``` + +## Letters and digits + +Letters are the set of ASCII characters plus the underscore character _ (U+005F) is considered a letter. + +Only decimal digits are supported. + +``` +letter = ascii_letter | "_" . +ascii_letter = "A" … "Z" | "a" … "z" . +digit = "0" … "9" . +``` + +## Identifiers + +Identifiers are tokens which refer to database names, retention policy names, user names, measurement names, tag keys, and field names. + +The rules: + +- double quoted identifiers can contain any unicode character other than a new line +- double quoted identifiers can contain escaped `"` characters (i.e., `\"`) +- unquoted identifiers must start with an upper or lowercase ASCII character or "_" +- unquoted identifiers may contain only ASCII letters, decimal digits, and "_" + +``` +identifier = unquoted_identifier | quoted_identifier . +unquoted_identifier = ( letter ) { letter | digit } . +quoted_identifier = `"` unicode_char { unicode_char } `"` . +``` + +#### Examples: + +``` +cpu +_cpu_stats +"1h" +"anything really" +"1_Crazy-1337.identifier>NAME👍" +``` + +## Keywords + +``` +ALL ALTER AS ASC BEGIN BY +CREATE CONTINUOUS DATABASE DATABASES DEFAULT DELETE +DESC DROP DURATION END EXISTS EXPLAIN +FIELD FROM GRANT GROUP IF IN +INNER INSERT INTO KEY KEYS LIMIT +SHOW MEASUREMENT MEASUREMENTS OFFSET ON ORDER +PASSWORD POLICY POLICIES PRIVILEGES QUERIES QUERY +READ REPLICATION RETENTION REVOKE SELECT SERIES +SLIMIT SOFFSET TAG TO USER USERS +VALUES WHERE WITH WRITE +``` + +## Literals + +### Integers + +InfluxQL supports decimal integer literals. Hexadecimal and octal literals are not currently supported. + +``` +int_lit = ( "1" … "9" ) { digit } . +``` + +### Floats + +InfluxQL supports floating-point literals. Exponents are not currently supported. + +``` +float_lit = int_lit "." int_lit . +``` + +### Strings + +String literals must be surrounded by single quotes. Strings may contain `'` characters as long as they are escaped (i.e., `\'`). + +``` +string_lit = `'` { unicode_char } `'`' . +``` + +### Durations + +Duration literals specify a length of time. An integer literal followed immediately (with no spaces) by a duration unit listed below is interpreted as a duration literal. + +``` +Duration unit definitions +------------------------- +| Units | Meaning | +|--------|-----------------------------------------| +| u or µ | microseconds (1 millionth of a second) | +| ms | milliseconds (1 thousandth of a second) | +| s | second | +| m | minute | +| h | hour | +| d | day | +| w | week | +``` + +``` +duration_lit = int_lit duration_unit . +duration_unit = "u" | "µ" | "s" | "h" | "d" | "w" | "ms" . +``` + +### Dates & Times + +The date and time literal format is not specified in EBNF like the rest of this document. It is specified using Go's date / time parsing format, which is a reference date written in the format required by InfluxQL. The reference date time is: + +InfluxQL reference date time: January 2nd, 2006 at 3:04:05 PM + +``` +time_lit = "2006-01-02 15:04:05.999999" | "2006-01-02" +``` + +### Booleans + +``` +bool_lit = TRUE | FALSE . +``` + +### Regular Expressions + +``` +regex_lit = "/" { unicode_char } "/" . +``` + +## Queries + +A query is composed of one or more statements separated by a semicolon. + +``` +query = statement { ; statement } . + +statement = alter_retention_policy_stmt | + create_continuous_query_stmt | + create_database_stmt | + create_retention_policy_stmt | + create_user_stmt | + delete_stmt | + drop_continuous_query_stmt | + drop_database_stmt | + drop_measurement_stmt | + drop_retention_policy_stmt | + drop_series_stmt | + drop_user_stmt | + grant_stmt | + show_continuous_queries_stmt | + show_databases_stmt | + show_field_keys_stmt | + show_measurements_stmt | + show_retention_policies | + show_series_stmt | + show_tag_keys_stmt | + show_tag_values_stmt | + show_users_stmt | + revoke_stmt | + select_stmt . +``` + +## Statements + +### ALTER RETENTION POLICY + +``` +alter_retention_policy_stmt = "ALTER RETENTION POLICY" policy_name "ON" + db_name retention_policy_option + [ retention_policy_option ] + [ retention_policy_option ] . + +db_name = identifier . + +policy_name = identifier . + +retention_policy_option = retention_policy_duration | + retention_policy_replication | + "DEFAULT" . + +retention_policy_duration = "DURATION" duration_lit . +retention_policy_replication = "REPLICATION" int_lit +``` + +#### Examples: + +```sql +-- Set default retention policy for mydb to 1h.cpu. +ALTER RETENTION POLICY "1h.cpu" ON mydb DEFAULT; + +-- Change duration and replication factor. +ALTER RETENTION POLICY policy1 ON somedb DURATION 1h REPLICATION 4 +``` + +### CREATE CONTINUOUS QUERY + +``` +create_continuous_query_stmt = "CREATE CONTINUOUS QUERY" query_name "ON" db_name + "BEGIN" select_stmt "END" . + +query_name = identifier . +``` + +#### Examples: + +```sql +-- selects from default retention policy and writes into 6_months retention policy +CREATE CONTINUOUS QUERY "10m_event_count" +ON db_name +BEGIN + SELECT count(value) + INTO "6_months".events + FROM events + GROUP BY time(10m) +END; + +-- this selects from the output of one continuous query in one retention policy and outputs to another series in another retention policy +CREATE CONTINUOUS QUERY "1h_event_count" +ON db_name +BEGIN + SELECT sum(count) as count + INTO "2_years".events + FROM "6_months".events + GROUP BY time(1h) +END; +``` + +### CREATE DATABASE + +``` +create_database_stmt = "CREATE DATABASE" db_name +``` + +#### Example: + +```sql +CREATE DATABASE foo +``` + +### CREATE RETENTION POLICY + +``` +create_retention_policy_stmt = "CREATE RETENTION POLICY" policy_name "ON" + db_name retention_policy_duration + retention_policy_replication + [ "DEFAULT" ] . +``` + +#### Examples + +```sql +-- Create a retention policy. +CREATE RETENTION POLICY "10m.events" ON somedb DURATION 10m REPLICATION 2; + +-- Create a retention policy and set it as the default. +CREATE RETENTION POLICY "10m.events" ON somedb DURATION 10m REPLICATION 2 DEFAULT; +``` + +### CREATE USER + +``` +create_user_stmt = "CREATE USER" user_name "WITH PASSWORD" password + [ "WITH ALL PRIVILEGES" ] . +``` + +#### Examples: + +```sql +-- Create a normal database user. +CREATE USER jdoe WITH PASSWORD '1337password'; + +-- Create a cluster admin. +-- Note: Unlike the GRANT statement, the "PRIVILEGES" keyword is required here. +CREATE USER jdoe WITH PASSWORD '1337password' WITH ALL PRIVILEGES; +``` + +### DELETE + +``` +delete_stmt = "DELETE" from_clause where_clause . +``` + +#### Example: + +```sql +-- delete data points from the cpu measurement where the region tag +-- equals 'uswest' +DELETE FROM cpu WHERE region = 'uswest'; +``` + +### DROP CONTINUOUS QUERY + +drop_continuous_query_stmt = "DROP CONTINUOUS QUERY" query_name . + +#### Example: + +```sql +DROP CONTINUOUS QUERY myquery; +``` + +### DROP DATABASE + +drop_database_stmt = "DROP DATABASE" db_name . + +#### Example: + +```sql +DROP DATABASE mydb; +``` + +### DROP MEASUREMENT + +``` +drop_measurement_stmt = "DROP MEASUREMENT" measurement . +``` + +#### Examples: + +```sql +-- drop the cpu measurement +DROP MEASUREMENT cpu; +``` + +### DROP RETENTION POLICY + +``` +drop_retention_policy_stmt = "DROP RETENTION POLICY" policy_name "ON" db_name . +``` + +#### Example: + +```sql +-- drop the retention policy named 1h.cpu from mydb +DROP RETENTION POLICY "1h.cpu" ON mydb; +``` + +### DROP SERIES + +``` +drop_series_stmt = "DROP SERIES" [ from_clause ] [ where_clause ] +``` + +#### Example: + +```sql + +``` + +### DROP USER + +``` +drop_user_stmt = "DROP USER" user_name . +``` + +#### Example: + +```sql +DROP USER jdoe; + +``` + +### GRANT + +NOTE: Users can be granted privileges on databases that do not exist. + +``` +grant_stmt = "GRANT" privilege [ on_clause ] to_clause +``` + +#### Examples: + +```sql +-- grant cluster admin privileges +GRANT ALL TO jdoe; + +-- grant read access to a database +GRANT READ ON mydb TO jdoe; +``` + +### SHOW CONTINUOUS QUERIES + +show_continuous_queries_stmt = "SHOW CONTINUOUS QUERIES" + +#### Example: + +```sql +-- show all continuous queries +SHOW CONTINUOUS QUERIES; +``` + +### SHOW DATABASES + +``` +show_databases_stmt = "SHOW DATABASES" . +``` + +#### Example: + +```sql +-- show all databases +SHOW DATABASES; +``` + +### SHOW FIELD + +show_field_keys_stmt = "SHOW FIELD KEYS" [ from_clause ] . + +#### Examples: + +```sql +-- show field keys from all measurements +SHOW FIELD KEYS; + +-- show field keys from specified measurement +SHOW FIELD KEYS FROM cpu; +``` + +### SHOW MEASUREMENTS + +show_measurements_stmt = [ where_clause ] [ group_by_clause ] [ limit_clause ] + [ offset_clause ] . + +```sql +-- show all measurements +SHOW MEASUREMENTS; + +-- show measurements where region tag = 'uswest' AND host tag = 'serverA' +SHOW MEASUREMENTS WHERE region = 'uswest' AND host = 'serverA'; +``` + +### SHOW RETENTION POLICIES + +``` +show_retention_policies = "SHOW RETENTION POLICIES ON" db_name . +``` + +#### Example: + +```sql +-- show all retention policies on a database +SHOW RETENTION POLICIES ON mydb; +``` + +### SHOW SERIES + +``` +show_series_stmt = [ from_clause ] [ where_clause ] [ group_by_clause ] + [ limit_clause ] [ offset_clause ] . +``` + +#### Example: + +```sql + +``` + +### SHOW TAG KEYS + +``` +show_tag_keys_stmt = [ from_clause ] [ where_clause ] [ group_by_clause ] + [ limit_clause ] [ offset_clause ] . +``` + +#### Examples: + +```sql +-- show all tag keys +SHOW TAG KEYS; + +-- show all tag keys from the cpu measurement +SHOW TAG KEYS FROM cpu; + +-- show all tag keys from the cpu measurement where the region key = 'uswest' +SHOW TAG KEYS FROM cpu WHERE region = 'uswest'; + +-- show all tag keys where the host key = 'serverA' +SHOW TAG KEYS WHERE host = 'serverA'; +``` + +### SHOW TAG VALUES + +``` +show_tag_values_stmt = [ from_clause ] with_tag_clause [ where_clause ] + [ group_by_clause ] [ limit_clause ] [ offset_clause ] . +``` + +#### Examples: + +```sql +-- show all tag values across all measurements for the region tag +SHOW TAG VALUES WITH TAG = 'region'; + +-- show tag values from the cpu measurement for the region tag +SHOW TAG VALUES FROM cpu WITH TAG = 'region'; + +-- show tag values from the cpu measurement for region & host tag keys where service = 'redis' +SHOW TAG VALUES FROM cpu WITH TAG IN (region, host) WHERE service = 'redis'; +``` + +### SHOW USERS + +``` +show_users_stmt = "SHOW USERS" . +``` + +#### Example: + +```sql +-- show all users +SHOW USERS; +``` + +### REVOKE + +``` +revoke_stmt = privilege [ "ON" db_name ] "FROM" user_name +``` + +#### Examples: + +```sql +-- revoke cluster admin from jdoe +REVOKE ALL PRIVILEGES FROM jdoe; + +-- revoke read privileges from jdoe on mydb +REVOKE READ ON mydb FROM jdoe; +``` + +### SELECT + +``` +select_stmt = fields from_clause [ into_clause ] [ where_clause ] + [ group_by_clause ] [ order_by_clause ] [ limit_clause ] + [ offset_clause ] [ slimit_clause ] [ soffset_clause ]. +``` + +#### Examples: + +```sql +-- select mean value from the cpu measurement where region = 'uswest' grouped by 10 minute intervals +SELECT mean(value) FROM cpu WHERE region = 'uswest' GROUP BY time(10m) fill(0); +``` + +## Clauses + +``` +from_clause = "FROM" measurements . + +group_by_clause = "GROUP BY" dimensions fill(