Skip to content

Commit

Permalink
Move log tokenizer to lib/parseutils package
Browse files Browse the repository at this point in the history
Up until now the tokenizer implementation was only used to parse the log
output option configuration. With our current intention of using this
format for other options, such as traces output, it becomes evident that
this implementation should be moved away from the log package and into a
more generic one.
  • Loading branch information
ka3de committed Nov 8, 2023
1 parent e44a08f commit 154a7f3
Show file tree
Hide file tree
Showing 5 changed files with 38 additions and 31 deletions.
2 changes: 2 additions & 0 deletions lib/parseutils/doc.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
// Package parseutils provides text parsing utilities.
package parseutils
25 changes: 14 additions & 11 deletions log/tokenizer.go → lib/parseutils/tokenizer.go
Original file line number Diff line number Diff line change
@@ -1,9 +1,10 @@
package log
package parseutils

import "fmt"

type token struct {
key, value string
// Token represents a key-value token.
type Token struct {
Key, Value string
inside rune // shows whether it's inside a given collection, currently [ means it's an array
}

Expand Down Expand Up @@ -64,8 +65,10 @@ func (t *tokenizer) readArray() (string, error) {
return t.s[start:], fmt.Errorf("array value for key `%s` didn't end", t.currentKey)
}

func tokenize(s string) ([]token, error) {
result := []token{}
// Tokenize parses the input string into key-value tokens according to format:
// key1=value1,key2=value2,...,keyN=valueN.
func Tokenize(s string) ([]Token, error) {
result := []Token{}
t := &tokenizer{s: s}

var err error
Expand All @@ -79,19 +82,19 @@ func tokenize(s string) ([]token, error) {
t.i++
value, err = t.readArray()

result = append(result, token{
key: t.currentKey,
value: value,
result = append(result, Token{
Key: t.currentKey,
Value: value,
inside: '[',
})
if err != nil {
return result, err
}
} else {
value = t.readValue()
result = append(result, token{
key: t.currentKey,
value: value,
result = append(result, Token{
Key: t.currentKey,
Value: value,
})
}
}
Expand Down
22 changes: 11 additions & 11 deletions log/tokenizer_test.go → lib/parseutils/tokenizer_test.go
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
package log
package parseutils

import (
"testing"
Expand All @@ -11,17 +11,17 @@ func TestTokenizer(t *testing.T) {
t.Parallel()

input := "loki=something,s.e=2231,s=12,12=3,a=[1,2,3],b=[1],s=c"
tokens, err := tokenize(input)
tokens, err := Tokenize(input)
require.NoError(t, err)

expected := []token{
{key: "loki", value: "something"},
{key: "s.e", value: "2231"},
{key: "s", value: "12"},
{key: "12", value: "3"},
{key: "a", value: "1,2,3", inside: '['},
{key: "b", value: "1", inside: '['},
{key: "s", value: "c"},
expected := []Token{
{Key: "loki", Value: "something"},
{Key: "s.e", Value: "2231"},
{Key: "s", Value: "12"},
{Key: "12", Value: "3"},
{Key: "a", Value: "1,2,3", inside: '['},
{Key: "b", Value: "1", inside: '['},
{Key: "s", Value: "c"},
}
assert.Equal(t, expected, tokens)
}
Expand Down Expand Up @@ -61,7 +61,7 @@ func TestTokenizerInvalid(t *testing.T) {
t.Run(test.name, func(t *testing.T) {
t.Parallel()

_, err := tokenize(test.input)
_, err := Tokenize(test.input)
require.EqualError(t, err, test.errorMsg)
})
}
Expand Down
13 changes: 7 additions & 6 deletions log/file.go
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ import (

"github.com/sirupsen/logrus"
"go.k6.io/k6/lib/fsext"
"go.k6.io/k6/lib/parseutils"
)

// fileHookBufferSize is a default size for the fileHook's loglines channel.
Expand Down Expand Up @@ -55,25 +56,25 @@ func FileHookFromConfigLine(
}

func (h *fileHook) parseArgs(line string) error {
tokens, err := tokenize(line)
tokens, err := parseutils.Tokenize(line)
if err != nil {
return fmt.Errorf("error while parsing logfile configuration %w", err)
}

for _, token := range tokens {
switch token.key {
switch token.Key {
case "file":
if token.value == "" {
if token.Value == "" {
return fmt.Errorf("filepath must not be empty")
}
h.path = token.value
h.path = token.Value
case "level":
h.levels, err = parseLevels(token.value)
h.levels, err = parseLevels(token.Value)
if err != nil {
return err
}
default:
return fmt.Errorf("unknown logfile config key %s", token.key)
return fmt.Errorf("unknown logfile config key %s", token.Key)
}
}

Expand Down
7 changes: 4 additions & 3 deletions log/loki.go
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ import (
"time"

"github.com/sirupsen/logrus"
"go.k6.io/k6/lib/parseutils"
)

// lokiHook is a Logrus hook for flushing to Loki.
Expand Down Expand Up @@ -77,14 +78,14 @@ func LokiFromConfigLine(fallbackLogger logrus.FieldLogger, line string) (AsyncHo
}

func (h *lokiHook) parseArgs(line string) error {
tokens, err := tokenize(line)
tokens, err := parseutils.Tokenize(line)
if err != nil {
return fmt.Errorf("error while parsing loki configuration %w", err)
}

for _, token := range tokens {
key := token.key
value := token.value
key := token.Key
value := token.Value

var err error
switch key {
Expand Down

0 comments on commit 154a7f3

Please sign in to comment.