From adb569e5b95cdf79917dfedba3f1329ab1f5d7b5 Mon Sep 17 00:00:00 2001 From: Alex Dadgar Date: Mon, 27 Jun 2016 22:49:53 -0700 Subject: [PATCH] update columnize to include optimized code --- .../github.com/ryanuber/columnize/.travis.yml | 3 - .../ryanuber/columnize/columnize.go | 55 ++++++++++++++----- vendor/vendor.json | 4 +- 3 files changed, 45 insertions(+), 17 deletions(-) delete mode 100644 vendor/github.com/ryanuber/columnize/.travis.yml diff --git a/vendor/github.com/ryanuber/columnize/.travis.yml b/vendor/github.com/ryanuber/columnize/.travis.yml deleted file mode 100644 index 1a0bbea6c77..00000000000 --- a/vendor/github.com/ryanuber/columnize/.travis.yml +++ /dev/null @@ -1,3 +0,0 @@ -language: go -go: - - tip diff --git a/vendor/github.com/ryanuber/columnize/columnize.go b/vendor/github.com/ryanuber/columnize/columnize.go index d87785940c6..9f0fe72944c 100644 --- a/vendor/github.com/ryanuber/columnize/columnize.go +++ b/vendor/github.com/ryanuber/columnize/columnize.go @@ -1,6 +1,7 @@ package columnize import ( + "bytes" "fmt" "strings" ) @@ -31,13 +32,14 @@ func DefaultConfig() *Config { // Returns a list of elements, each representing a single item which will // belong to a column of output. func getElementsFromLine(config *Config, line string) []interface{} { - elements := make([]interface{}, 0) - for _, field := range strings.Split(line, config.Delim) { + seperated := strings.Split(line, config.Delim) + elements := make([]interface{}, len(seperated)) + for i, field := range seperated { value := strings.TrimSpace(field) if value == "" && config.Empty != "" { value = config.Empty } - elements = append(elements, value) + elements[i] = value } return elements } @@ -45,7 +47,7 @@ func getElementsFromLine(config *Config, line string) []interface{} { // Examines a list of strings and determines how wide each column should be // considering all of the elements that need to be printed within it. func getWidthsFromLines(config *Config, lines []string) []int { - var widths []int + widths := make([]int, 0, 8) for _, line := range lines { elems := getElementsFromLine(config, line) @@ -65,18 +67,22 @@ func getWidthsFromLines(config *Config, lines []string) []int { // returns a sprintf-style format string which can be used to print output // aligned properly with other lines using the same widths set. func (c *Config) getStringFormat(widths []int, columns int) string { - // Start with the prefix, if any was given. - stringfmt := c.Prefix + // Create the buffer with an estimate of the length + buf := bytes.NewBuffer(make([]byte, 0, (6+len(c.Glue))*columns)) + + // Start with the prefix, if any was given. The buffer will not return an + // error so it does not need to be handled + buf.WriteString(c.Prefix) // Create the format string from the discovered widths for i := 0; i < columns && i < len(widths); i++ { if i == columns-1 { - stringfmt += "%s\n" + buf.WriteString("%s\n") } else { - stringfmt += fmt.Sprintf("%%-%ds%s", widths[i], c.Glue) + fmt.Fprintf(buf, "%%-%ds%s", widths[i], c.Glue) } } - return stringfmt + return buf.String() } // MergeConfig merges two config objects together and returns the resulting @@ -108,18 +114,41 @@ func MergeConfig(a, b *Config) *Config { // Format is the public-facing interface that takes either a plain string // or a list of strings and returns nicely aligned output. func Format(lines []string, config *Config) string { - var result string - conf := MergeConfig(DefaultConfig(), config) widths := getWidthsFromLines(conf, lines) + // Estimate the buffer size + glueSize := len(conf.Glue) + var size int + for _, w := range widths { + size += w + glueSize + } + size *= len(lines) + + // Create the buffer + buf := bytes.NewBuffer(make([]byte, 0, size)) + + // Create a cache for the string formats + fmtCache := make(map[int]string, 16) + // Create the formatted output using the format string for _, line := range lines { elems := getElementsFromLine(conf, line) - stringfmt := conf.getStringFormat(widths, len(elems)) - result += fmt.Sprintf(stringfmt, elems...) + + // Get the string format using cache + numElems := len(elems) + stringfmt, ok := fmtCache[numElems] + if !ok { + stringfmt = conf.getStringFormat(widths, numElems) + fmtCache[numElems] = stringfmt + } + + fmt.Fprintf(buf, stringfmt, elems...) } + // Get the string result + result := buf.String() + // Remove trailing newline without removing leading/trailing space if n := len(result); n > 0 && result[n-1] == '\n' { result = result[:n-1] diff --git a/vendor/vendor.json b/vendor/vendor.json index 2dc2e511c24..0d930c40b6d 100644 --- a/vendor/vendor.json +++ b/vendor/vendor.json @@ -661,9 +661,11 @@ "revision": "89ab7f2ccc1e45ddf6485eaa802c35dcf321dfc8" }, { + "checksumSHA1": "1SC2ACq72a+yfN6CYp5s5woKsR4=", "comment": "v2.0.1-8-g983d3a5", "path": "github.com/ryanuber/columnize", - "revision": "983d3a5fab1bf04d1b412465d2d9f8430e2e917e" + "revision": "50b9b539927dfe231b8be1caf81b2c81d5940276", + "revisionTime": "2016-06-28T05:25:20Z" }, { "checksumSHA1": "G1oy2AGv5pCnNL0hRfg+mlX4Uq4=",