mirror of
https://codeberg.org/scip/tablizer.git
synced 2025-12-19 13:31:02 +01:00
Compare commits
1 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 35de2fea2f |
32
.github/workflows/release.yaml
vendored
32
.github/workflows/release.yaml
vendored
@@ -1,32 +0,0 @@
|
|||||||
name: build-and-test
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
tags:
|
|
||||||
- "*"
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
release:
|
|
||||||
name: Build Release Assets
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Checkout code
|
|
||||||
uses: actions/checkout@v2
|
|
||||||
|
|
||||||
- name: Set up Go
|
|
||||||
uses: actions/setup-go@v1
|
|
||||||
with:
|
|
||||||
go-version: 1.22.11
|
|
||||||
|
|
||||||
- name: Build the executables
|
|
||||||
run: ./mkrel.sh tablizer ${{ github.ref_name}}
|
|
||||||
|
|
||||||
- name: List the executables
|
|
||||||
run: ls -l ./releases
|
|
||||||
|
|
||||||
- name: Upload the binaries
|
|
||||||
uses: svenstaro/upload-release-action@v2
|
|
||||||
with:
|
|
||||||
repo_token: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
tag: ${{ github.ref_name }}
|
|
||||||
file: ./releases/*
|
|
||||||
file_glob: true
|
|
||||||
3
Makefile
3
Makefile
@@ -53,7 +53,8 @@ buildlocal:
|
|||||||
go build -ldflags "-X 'github.com/tlinden/tablizer/cfg.VERSION=$(VERSION)'"
|
go build -ldflags "-X 'github.com/tlinden/tablizer/cfg.VERSION=$(VERSION)'"
|
||||||
|
|
||||||
release:
|
release:
|
||||||
gh release create $(version) --generate-notes
|
./mkrel.sh $(tool) $(version)
|
||||||
|
gh release create $(version) --generate-notes releases/*
|
||||||
|
|
||||||
install: buildlocal
|
install: buildlocal
|
||||||
install -d -o $(UID) -g $(GID) $(PREFIX)/bin
|
install -d -o $(UID) -g $(GID) $(PREFIX)/bin
|
||||||
|
|||||||
@@ -22,7 +22,7 @@ Operational Flags:
|
|||||||
-s, --separator string Custom field separator
|
-s, --separator string Custom field separator
|
||||||
-k, --sort-by int Sort by column (default: 1)
|
-k, --sort-by int Sort by column (default: 1)
|
||||||
-z, --fuzzy Use fuzzy search [experimental]
|
-z, --fuzzy Use fuzzy search [experimental]
|
||||||
-F, --filter field[!]=reg Filter given field with regex, can be used multiple times
|
-F, --filter field=reg Filter given field with regex, can be used multiple times
|
||||||
-T, --transpose-columns string Transpose the speficied columns (separated by ,)
|
-T, --transpose-columns string Transpose the speficied columns (separated by ,)
|
||||||
-R, --regex-transposer /from/to/ Apply /search/replace/ regexp to fields given in -T
|
-R, --regex-transposer /from/to/ Apply /search/replace/ regexp to fields given in -T
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
Copyright © 2022-2025 Thomas von Dein
|
Copyright © 2022-2024 Thomas von Dein
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
This program is free software: you can redistribute it and/or modify
|
||||||
it under the terms of the GNU General Public License as published by
|
it under the terms of the GNU General Public License as published by
|
||||||
@@ -28,7 +28,7 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
const DefaultSeparator string = `(\s\s+|\t)`
|
const DefaultSeparator string = `(\s\s+|\t)`
|
||||||
const Version string = "v1.3.2"
|
const Version string = "v1.3.1"
|
||||||
const MAXPARTS = 2
|
const MAXPARTS = 2
|
||||||
|
|
||||||
var DefaultConfigfile = os.Getenv("HOME") + "/.config/tablizer/config"
|
var DefaultConfigfile = os.Getenv("HOME") + "/.config/tablizer/config"
|
||||||
@@ -52,17 +52,6 @@ type Transposer struct {
|
|||||||
Replace string
|
Replace string
|
||||||
}
|
}
|
||||||
|
|
||||||
type Pattern struct {
|
|
||||||
Pattern string
|
|
||||||
PatternRe *regexp.Regexp
|
|
||||||
Negate bool
|
|
||||||
}
|
|
||||||
|
|
||||||
type Filter struct {
|
|
||||||
Regex *regexp.Regexp
|
|
||||||
Negate bool
|
|
||||||
}
|
|
||||||
|
|
||||||
// internal config
|
// internal config
|
||||||
type Config struct {
|
type Config struct {
|
||||||
Debug bool
|
Debug bool
|
||||||
@@ -73,7 +62,8 @@ type Config struct {
|
|||||||
Separator string
|
Separator string
|
||||||
OutputMode int
|
OutputMode int
|
||||||
InvertMatch bool
|
InvertMatch bool
|
||||||
Patterns []*Pattern
|
Pattern string
|
||||||
|
PatternR *regexp.Regexp
|
||||||
UseFuzzySearch bool
|
UseFuzzySearch bool
|
||||||
UseHighlight bool
|
UseHighlight bool
|
||||||
|
|
||||||
@@ -105,7 +95,7 @@ type Config struct {
|
|||||||
|
|
||||||
// used for field filtering
|
// used for field filtering
|
||||||
Rawfilters []string
|
Rawfilters []string
|
||||||
Filters map[string]Filter //map[string]*regexp.Regexp
|
Filters map[string]*regexp.Regexp
|
||||||
|
|
||||||
// -r <file>
|
// -r <file>
|
||||||
InputFile string
|
InputFile string
|
||||||
@@ -275,20 +265,12 @@ func (conf *Config) PrepareModeFlags(flag Modeflag) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (conf *Config) PrepareFilters() error {
|
func (conf *Config) PrepareFilters() error {
|
||||||
conf.Filters = make(map[string]Filter, len(conf.Rawfilters))
|
conf.Filters = make(map[string]*regexp.Regexp, len(conf.Rawfilters))
|
||||||
|
|
||||||
for _, rawfilter := range conf.Rawfilters {
|
for _, filter := range conf.Rawfilters {
|
||||||
filter := Filter{}
|
parts := strings.Split(filter, "=")
|
||||||
|
|
||||||
parts := strings.Split(rawfilter, "!=")
|
|
||||||
if len(parts) != MAXPARTS {
|
if len(parts) != MAXPARTS {
|
||||||
parts = strings.Split(rawfilter, "=")
|
return errors.New("filter field and value must be separated by =")
|
||||||
|
|
||||||
if len(parts) != MAXPARTS {
|
|
||||||
return errors.New("filter field and value must be separated by '=' or '!='")
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
filter.Negate = true
|
|
||||||
}
|
}
|
||||||
|
|
||||||
reg, err := regexp.Compile(parts[1])
|
reg, err := regexp.Compile(parts[1])
|
||||||
@@ -297,8 +279,7 @@ func (conf *Config) PrepareFilters() error {
|
|||||||
parts[0], err)
|
parts[0], err)
|
||||||
}
|
}
|
||||||
|
|
||||||
filter.Regex = reg
|
conf.Filters[strings.ToLower(strings.ToLower(parts[0]))] = reg
|
||||||
conf.Filters[strings.ToLower(parts[0])] = filter
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
@@ -352,37 +333,15 @@ func (conf *Config) ApplyDefaults() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (conf *Config) PreparePattern(patterns []*Pattern) error {
|
func (conf *Config) PreparePattern(pattern string) error {
|
||||||
// regex checks if a pattern looks like /$pattern/[i!]
|
PatternR, err := regexp.Compile(pattern)
|
||||||
flagre := regexp.MustCompile(`^/(.*)/([i!]*)$`)
|
|
||||||
|
|
||||||
for _, pattern := range patterns {
|
|
||||||
matches := flagre.FindAllStringSubmatch(pattern.Pattern, -1)
|
|
||||||
|
|
||||||
// we have a regex with flags
|
|
||||||
for _, match := range matches {
|
|
||||||
pattern.Pattern = match[1] // the inner part is our actual pattern
|
|
||||||
flags := match[2] // the flags
|
|
||||||
|
|
||||||
for _, flag := range flags {
|
|
||||||
switch flag {
|
|
||||||
case 'i':
|
|
||||||
pattern.Pattern = `(?i)` + pattern.Pattern
|
|
||||||
case '!':
|
|
||||||
pattern.Negate = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
PatternRe, err := regexp.Compile(pattern.Pattern)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("regexp pattern %s is invalid: %w", pattern.Pattern, err)
|
return fmt.Errorf("regexp pattern %s is invalid: %w", conf.Pattern, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
pattern.PatternRe = PatternRe
|
conf.PatternR = PatternR
|
||||||
}
|
conf.Pattern = pattern
|
||||||
|
|
||||||
conf.Patterns = patterns
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -79,55 +79,20 @@ func TestPrepareSortFlags(t *testing.T) {
|
|||||||
|
|
||||||
func TestPreparePattern(t *testing.T) {
|
func TestPreparePattern(t *testing.T) {
|
||||||
var tests = []struct {
|
var tests = []struct {
|
||||||
patterns []*Pattern
|
pattern string
|
||||||
name string
|
|
||||||
wanterr bool
|
wanterr bool
|
||||||
wanticase bool
|
|
||||||
wantneg bool
|
|
||||||
}{
|
}{
|
||||||
{
|
{"[A-Z]+", false},
|
||||||
[]*Pattern{{Pattern: "[A-Z]+"}},
|
{"[a-z", true},
|
||||||
"simple",
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
[]*Pattern{{Pattern: "[a-z"}},
|
|
||||||
"regfail",
|
|
||||||
true,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
[]*Pattern{{Pattern: "/[A-Z]+/i"}},
|
|
||||||
"icase",
|
|
||||||
false,
|
|
||||||
true,
|
|
||||||
false,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
[]*Pattern{{Pattern: "/[A-Z]+/!"}},
|
|
||||||
"negate",
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
[]*Pattern{{Pattern: "/[A-Z]+/!i"}},
|
|
||||||
"negicase",
|
|
||||||
false,
|
|
||||||
true,
|
|
||||||
true,
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, testdata := range tests {
|
for _, testdata := range tests {
|
||||||
testname := fmt.Sprintf("PreparePattern-pattern-%s-wanterr-%t", testdata.name, testdata.wanterr)
|
testname := fmt.Sprintf("PreparePattern-pattern-%s-wanterr-%t",
|
||||||
|
testdata.pattern, testdata.wanterr)
|
||||||
t.Run(testname, func(t *testing.T) {
|
t.Run(testname, func(t *testing.T) {
|
||||||
conf := Config{}
|
conf := Config{}
|
||||||
|
|
||||||
err := conf.PreparePattern(testdata.patterns)
|
err := conf.PreparePattern(testdata.pattern)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if !testdata.wanterr {
|
if !testdata.wanterr {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
Copyright © 2022-2025 Thomas von Dein
|
Copyright © 2022-2024 Thomas von Dein
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
This program is free software: you can redistribute it and/or modify
|
||||||
it under the terms of the GNU General Public License as published by
|
it under the terms of the GNU General Public License as published by
|
||||||
@@ -190,7 +190,7 @@ func Execute() {
|
|||||||
|
|
||||||
// filters
|
// filters
|
||||||
rootCmd.PersistentFlags().StringArrayVarP(&conf.Rawfilters,
|
rootCmd.PersistentFlags().StringArrayVarP(&conf.Rawfilters,
|
||||||
"filter", "F", nil, "Filter by field (field=regexp || field!=regexp)")
|
"filter", "F", nil, "Filter by field (field=regexp)")
|
||||||
rootCmd.PersistentFlags().StringArrayVarP(&conf.Transposers,
|
rootCmd.PersistentFlags().StringArrayVarP(&conf.Transposers,
|
||||||
"regex-transposer", "R", nil, "apply /search/replace/ regexp to fields given in -T")
|
"regex-transposer", "R", nil, "apply /search/replace/ regexp to fields given in -T")
|
||||||
|
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ NAME
|
|||||||
|
|
||||||
SYNOPSIS
|
SYNOPSIS
|
||||||
Usage:
|
Usage:
|
||||||
tablizer [regex,...] [file, ...] [flags]
|
tablizer [regex] [file, ...] [flags]
|
||||||
|
|
||||||
Operational Flags:
|
Operational Flags:
|
||||||
-c, --columns string Only show the speficied columns (separated by ,)
|
-c, --columns string Only show the speficied columns (separated by ,)
|
||||||
@@ -17,7 +17,7 @@ SYNOPSIS
|
|||||||
-s, --separator string Custom field separator
|
-s, --separator string Custom field separator
|
||||||
-k, --sort-by int|name Sort by column (default: 1)
|
-k, --sort-by int|name Sort by column (default: 1)
|
||||||
-z, --fuzzy Use fuzzy search [experimental]
|
-z, --fuzzy Use fuzzy search [experimental]
|
||||||
-F, --filter field[!]=reg Filter given field with regex, can be used multiple times
|
-F, --filter field=reg Filter given field with regex, can be used multiple times
|
||||||
-T, --transpose-columns string Transpose the speficied columns (separated by ,)
|
-T, --transpose-columns string Transpose the speficied columns (separated by ,)
|
||||||
-R, --regex-transposer /from/to/ Apply /search/replace/ regexp to fields given in -T
|
-R, --regex-transposer /from/to/ Apply /search/replace/ regexp to fields given in -T
|
||||||
|
|
||||||
@@ -130,43 +130,30 @@ DESCRIPTION
|
|||||||
for the developer.
|
for the developer.
|
||||||
|
|
||||||
PATTERNS AND FILTERING
|
PATTERNS AND FILTERING
|
||||||
You can reduce the rows being displayed by using one or more regular
|
You can reduce the rows being displayed by using a regular expression
|
||||||
expression patterns. The regexp language being used is the one of
|
pattern. The regexp is PCRE compatible, refer to the syntax cheat sheet
|
||||||
GOLANG, refer to the syntax cheat sheet here:
|
here: <https://github.com/google/re2/wiki/Syntax>. If you want to read a
|
||||||
<https://pkg.go.dev/regexp/syntax>.
|
more comprehensive documentation about the topic and have perl installed
|
||||||
|
you can read it with:
|
||||||
If you want to read a more comprehensive documentation about the topic
|
|
||||||
and have perl installed you can read it with:
|
|
||||||
|
|
||||||
perldoc perlre
|
perldoc perlre
|
||||||
|
|
||||||
Or read it online: <https://perldoc.perl.org/perlre>. But please note
|
Or read it online: <https://perldoc.perl.org/perlre>.
|
||||||
that the GO regexp engine does NOT support all perl regex terms,
|
|
||||||
especially look-ahead and look-behind.
|
|
||||||
|
|
||||||
If you want to supply flags to a regex, then surround it with slashes
|
A note on modifiers: the regexp engine used in tablizer uses another
|
||||||
and append the flag. The following flags are supported:
|
modifier syntax:
|
||||||
|
|
||||||
i => case insensitive
|
(?MODIFIER)
|
||||||
! => negative match
|
|
||||||
|
The most important modifiers are:
|
||||||
|
|
||||||
|
"i" ignore case "m" multiline mode "s" single line mode
|
||||||
|
|
||||||
Example for a case insensitive search:
|
Example for a case insensitive search:
|
||||||
|
|
||||||
kubectl get pods -A | tablizer "/account/i"
|
kubectl get pods -A | tablizer "(?i)account"
|
||||||
|
|
||||||
If you use the "!" flag, then the regex match will be negated, that is,
|
You can use the experimental fuzzy search feature by providing the
|
||||||
if a line in the input matches the given regex, but "!" is supplied,
|
|
||||||
tablizer will NOT include it in the output.
|
|
||||||
|
|
||||||
For example, here we want to get all lines matching "foo" but not "bar":
|
|
||||||
|
|
||||||
cat table | tablizer foo '/bar/!'
|
|
||||||
|
|
||||||
This would match a line "foo zorro" but not "foo bar".
|
|
||||||
|
|
||||||
The flags can also be combined.
|
|
||||||
|
|
||||||
You can also use the experimental fuzzy search feature by providing the
|
|
||||||
option -z, in which case the pattern is regarded as a fuzzy search term,
|
option -z, in which case the pattern is regarded as a fuzzy search term,
|
||||||
not a regexp.
|
not a regexp.
|
||||||
|
|
||||||
@@ -181,10 +168,6 @@ DESCRIPTION
|
|||||||
If you specify more than one filter, both filters have to match (AND
|
If you specify more than one filter, both filters have to match (AND
|
||||||
operation).
|
operation).
|
||||||
|
|
||||||
These field filters can also be negated:
|
|
||||||
|
|
||||||
fieldname!=regexp
|
|
||||||
|
|
||||||
If the option -v is specified, the filtering is inverted.
|
If the option -v is specified, the filtering is inverted.
|
||||||
|
|
||||||
COLUMNS
|
COLUMNS
|
||||||
@@ -409,7 +392,7 @@ AUTHORS
|
|||||||
var usage = `
|
var usage = `
|
||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
tablizer [regex,...] [file, ...] [flags]
|
tablizer [regex] [file, ...] [flags]
|
||||||
|
|
||||||
Operational Flags:
|
Operational Flags:
|
||||||
-c, --columns string Only show the speficied columns (separated by ,)
|
-c, --columns string Only show the speficied columns (separated by ,)
|
||||||
@@ -420,7 +403,7 @@ Operational Flags:
|
|||||||
-s, --separator string Custom field separator
|
-s, --separator string Custom field separator
|
||||||
-k, --sort-by int|name Sort by column (default: 1)
|
-k, --sort-by int|name Sort by column (default: 1)
|
||||||
-z, --fuzzy Use fuzzy search [experimental]
|
-z, --fuzzy Use fuzzy search [experimental]
|
||||||
-F, --filter field[!]=reg Filter given field with regex, can be used multiple times
|
-F, --filter field=reg Filter given field with regex, can be used multiple times
|
||||||
-T, --transpose-columns string Transpose the speficied columns (separated by ,)
|
-T, --transpose-columns string Transpose the speficied columns (separated by ,)
|
||||||
-R, --regex-transposer /from/to/ Apply /search/replace/ regexp to fields given in -T
|
-R, --regex-transposer /from/to/ Apply /search/replace/ regexp to fields given in -T
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
Copyright © 2022-2025 Thomas von Dein
|
Copyright © 2022-2024 Thomas von Dein
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
This program is free software: you can redistribute it and/or modify
|
||||||
it under the terms of the GNU General Public License as published by
|
it under the terms of the GNU General Public License as published by
|
||||||
@@ -29,44 +29,13 @@ import (
|
|||||||
/*
|
/*
|
||||||
* [!]Match a line, use fuzzy search for normal pattern strings and
|
* [!]Match a line, use fuzzy search for normal pattern strings and
|
||||||
* regexp otherwise.
|
* regexp otherwise.
|
||||||
|
|
||||||
'foo bar' foo, /bar/! => false => line contains foo and not (not bar)
|
|
||||||
'foo nix' foo, /bar/! => ture => line contains foo and (not bar)
|
|
||||||
'foo bar' foo, /bar/ => true => line contains both foo and bar
|
|
||||||
'foo nix' foo, /bar/ => false => line does not contain bar
|
|
||||||
'foo bar' foo, /nix/ => false => line does not contain nix
|
|
||||||
*/
|
*/
|
||||||
func matchPattern(conf cfg.Config, line string) bool {
|
func matchPattern(conf cfg.Config, line string) bool {
|
||||||
if len(conf.Patterns) == 0 {
|
|
||||||
// any line always matches ""
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
if conf.UseFuzzySearch {
|
if conf.UseFuzzySearch {
|
||||||
// fuzzy search only considers the 1st pattern
|
return fuzzy.MatchFold(conf.Pattern, line)
|
||||||
return fuzzy.MatchFold(conf.Patterns[0].Pattern, line)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var match int
|
return conf.PatternR.MatchString(line)
|
||||||
|
|
||||||
//fmt.Printf("<%s>\n", line)
|
|
||||||
for _, re := range conf.Patterns {
|
|
||||||
patmatch := re.PatternRe.MatchString(line)
|
|
||||||
if re.Negate {
|
|
||||||
// toggle the meaning of match
|
|
||||||
patmatch = !patmatch
|
|
||||||
}
|
|
||||||
|
|
||||||
if patmatch {
|
|
||||||
match++
|
|
||||||
}
|
|
||||||
|
|
||||||
//fmt.Printf("patmatch: %t, match: %d, pattern: %s, negate: %t\n", patmatch, match, re.Pattern, re.Negate)
|
|
||||||
}
|
|
||||||
|
|
||||||
// fmt.Printf("result: %t\n", match == len(conf.Patterns))
|
|
||||||
//fmt.Println()
|
|
||||||
return match == len(conf.Patterns)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -86,19 +55,15 @@ func FilterByFields(conf cfg.Config, data *Tabdata) (*Tabdata, bool, error) {
|
|||||||
keep := true
|
keep := true
|
||||||
|
|
||||||
for idx, header := range data.headers {
|
for idx, header := range data.headers {
|
||||||
lcheader := strings.ToLower(header)
|
if !Exists(conf.Filters, strings.ToLower(header)) {
|
||||||
if !Exists(conf.Filters, lcheader) {
|
|
||||||
// do not filter by unspecified field
|
// do not filter by unspecified field
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
match := conf.Filters[lcheader].Regex.MatchString(row[idx])
|
if !conf.Filters[strings.ToLower(header)].MatchString(row[idx]) {
|
||||||
if conf.Filters[lcheader].Negate {
|
// there IS a filter, but it doesn't match
|
||||||
match = !match
|
|
||||||
}
|
|
||||||
|
|
||||||
if !match {
|
|
||||||
keep = false
|
keep = false
|
||||||
|
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -158,11 +123,8 @@ func Exists[K comparable, V any](m map[K]V, v K) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Filters the whole input lines, returns filtered lines
|
|
||||||
*/
|
|
||||||
func FilterByPattern(conf cfg.Config, input io.Reader) (io.Reader, error) {
|
func FilterByPattern(conf cfg.Config, input io.Reader) (io.Reader, error) {
|
||||||
if len(conf.Patterns) == 0 {
|
if conf.Pattern == "" {
|
||||||
return input, nil
|
return input, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -174,7 +136,7 @@ func FilterByPattern(conf cfg.Config, input io.Reader) (io.Reader, error) {
|
|||||||
line := strings.TrimSpace(scanner.Text())
|
line := strings.TrimSpace(scanner.Text())
|
||||||
if hadFirst {
|
if hadFirst {
|
||||||
// don't match 1st line, it's the header
|
// don't match 1st line, it's the header
|
||||||
if matchPattern(conf, line) == conf.InvertMatch {
|
if conf.Pattern != "" && matchPattern(conf, line) == conf.InvertMatch {
|
||||||
// by default -v is false, so if a line does NOT
|
// by default -v is false, so if a line does NOT
|
||||||
// match the pattern, we will ignore it. However,
|
// match the pattern, we will ignore it. However,
|
||||||
// if the user specified -v, the matching is inverted,
|
// if the user specified -v, the matching is inverted,
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
Copyright © 2024-2025 Thomas von Dein
|
Copyright © 2024 Thomas von Dein
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
This program is free software: you can redistribute it and/or modify
|
||||||
it under the terms of the GNU General Public License as published by
|
it under the terms of the GNU General Public License as published by
|
||||||
@@ -29,17 +29,17 @@ func TestMatchPattern(t *testing.T) {
|
|||||||
var input = []struct {
|
var input = []struct {
|
||||||
name string
|
name string
|
||||||
fuzzy bool
|
fuzzy bool
|
||||||
patterns []*cfg.Pattern
|
pattern string
|
||||||
line string
|
line string
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
name: "normal",
|
name: "normal",
|
||||||
patterns: []*cfg.Pattern{{Pattern: "haus"}},
|
pattern: "haus",
|
||||||
line: "hausparty",
|
line: "hausparty",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "fuzzy",
|
name: "fuzzy",
|
||||||
patterns: []*cfg.Pattern{{Pattern: "hpt"}},
|
pattern: "hpt",
|
||||||
line: "haus-party-termin",
|
line: "haus-party-termin",
|
||||||
fuzzy: true,
|
fuzzy: true,
|
||||||
},
|
},
|
||||||
@@ -55,7 +55,7 @@ func TestMatchPattern(t *testing.T) {
|
|||||||
conf.UseFuzzySearch = true
|
conf.UseFuzzySearch = true
|
||||||
}
|
}
|
||||||
|
|
||||||
err := conf.PreparePattern(inputdata.patterns)
|
err := conf.PreparePattern(inputdata.pattern)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Errorf("PreparePattern returned error: %s", err)
|
t.Errorf("PreparePattern returned error: %s", err)
|
||||||
}
|
}
|
||||||
@@ -98,20 +98,6 @@ func TestFilterByFields(t *testing.T) {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
|
||||||
name: "one-field-negative",
|
|
||||||
filter: []string{"one!=asd"},
|
|
||||||
expect: Tabdata{
|
|
||||||
headers: []string{
|
|
||||||
"ONE", "TWO", "THREE",
|
|
||||||
},
|
|
||||||
entries: [][]string{
|
|
||||||
{"19191", "EDD 1", "x"},
|
|
||||||
{"8d8", "AN 1", "y"},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
{
|
||||||
name: "one-field-inverted",
|
name: "one-field-inverted",
|
||||||
filter: []string{"one=19"},
|
filter: []string{"one=19"},
|
||||||
|
|||||||
@@ -293,20 +293,12 @@ func colorizeData(conf cfg.Config, output string) string {
|
|||||||
|
|
||||||
return colorized
|
return colorized
|
||||||
|
|
||||||
case len(conf.Patterns) > 0 && !conf.NoColor && color.IsConsole(os.Stdout):
|
case len(conf.Pattern) > 0 && !conf.NoColor && color.IsConsole(os.Stdout):
|
||||||
out := output
|
r := regexp.MustCompile("(" + conf.Pattern + ")")
|
||||||
|
|
||||||
for _, re := range conf.Patterns {
|
return r.ReplaceAllStringFunc(output, func(in string) string {
|
||||||
if !re.Negate {
|
|
||||||
r := regexp.MustCompile("(" + re.Pattern + ")")
|
|
||||||
|
|
||||||
out = r.ReplaceAllStringFunc(out, func(in string) string {
|
|
||||||
return conf.ColorStyle.Sprint(in)
|
return conf.ColorStyle.Sprint(in)
|
||||||
})
|
})
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return out
|
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return output
|
return output
|
||||||
|
|||||||
20
lib/io.go
20
lib/io.go
@@ -29,13 +29,13 @@ import (
|
|||||||
const RWRR = 0755
|
const RWRR = 0755
|
||||||
|
|
||||||
func ProcessFiles(conf *cfg.Config, args []string) error {
|
func ProcessFiles(conf *cfg.Config, args []string) error {
|
||||||
fd, patterns, err := determineIO(conf, args)
|
fd, pattern, err := determineIO(conf, args)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := conf.PreparePattern(patterns); err != nil {
|
if err := conf.PreparePattern(pattern); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -63,9 +63,9 @@ func ProcessFiles(conf *cfg.Config, args []string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func determineIO(conf *cfg.Config, args []string) (io.Reader, []*cfg.Pattern, error) {
|
func determineIO(conf *cfg.Config, args []string) (io.Reader, string, error) {
|
||||||
var filehandle io.Reader
|
var filehandle io.Reader
|
||||||
var patterns []*cfg.Pattern
|
var pattern string
|
||||||
var haveio bool
|
var haveio bool
|
||||||
|
|
||||||
switch {
|
switch {
|
||||||
@@ -76,7 +76,7 @@ func determineIO(conf *cfg.Config, args []string) (io.Reader, []*cfg.Pattern, er
|
|||||||
fd, err := os.OpenFile(conf.InputFile, os.O_RDONLY, RWRR)
|
fd, err := os.OpenFile(conf.InputFile, os.O_RDONLY, RWRR)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, fmt.Errorf("failed to read input file %s: %w", conf.InputFile, err)
|
return nil, "", fmt.Errorf("failed to read input file %s: %w", conf.InputFile, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
filehandle = fd
|
filehandle = fd
|
||||||
@@ -93,15 +93,13 @@ func determineIO(conf *cfg.Config, args []string) (io.Reader, []*cfg.Pattern, er
|
|||||||
}
|
}
|
||||||
|
|
||||||
if len(args) > 0 {
|
if len(args) > 0 {
|
||||||
patterns = make([]*cfg.Pattern, len(args))
|
pattern = args[0]
|
||||||
for i, arg := range args {
|
conf.Pattern = args[0]
|
||||||
patterns[i] = &cfg.Pattern{Pattern: arg}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if !haveio {
|
if !haveio {
|
||||||
return nil, nil, errors.New("no file specified and nothing to read on stdin")
|
return nil, "", errors.New("no file specified and nothing to read on stdin")
|
||||||
}
|
}
|
||||||
|
|
||||||
return filehandle, patterns, nil
|
return filehandle, pattern, nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -137,7 +137,7 @@ func parseTabular(conf cfg.Config, input io.Reader) (Tabdata, error) {
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// data processing
|
// data processing
|
||||||
if matchPattern(conf, line) == conf.InvertMatch {
|
if conf.Pattern != "" && matchPattern(conf, line) == conf.InvertMatch {
|
||||||
// by default -v is false, so if a line does NOT
|
// by default -v is false, so if a line does NOT
|
||||||
// match the pattern, we will ignore it. However,
|
// match the pattern, we will ignore it. However,
|
||||||
// if the user specified -v, the matching is inverted,
|
// if the user specified -v, the matching is inverted,
|
||||||
|
|||||||
@@ -83,26 +83,23 @@ func TestParser(t *testing.T) {
|
|||||||
|
|
||||||
func TestParserPatternmatching(t *testing.T) {
|
func TestParserPatternmatching(t *testing.T) {
|
||||||
var tests = []struct {
|
var tests = []struct {
|
||||||
name string
|
|
||||||
entries [][]string
|
entries [][]string
|
||||||
patterns []*cfg.Pattern
|
pattern string
|
||||||
invert bool
|
invert bool
|
||||||
want bool
|
want bool
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
name: "match",
|
|
||||||
entries: [][]string{
|
entries: [][]string{
|
||||||
{"asd", "igig", "cxxxncnc"},
|
{"asd", "igig", "cxxxncnc"},
|
||||||
},
|
},
|
||||||
patterns: []*cfg.Pattern{{Pattern: "ig"}},
|
pattern: "ig",
|
||||||
invert: false,
|
invert: false,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "invert",
|
|
||||||
entries: [][]string{
|
entries: [][]string{
|
||||||
{"19191", "EDD 1", "X"},
|
{"19191", "EDD 1", "X"},
|
||||||
},
|
},
|
||||||
patterns: []*cfg.Pattern{{Pattern: "ig"}},
|
pattern: "ig",
|
||||||
invert: true,
|
invert: true,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@@ -110,15 +107,12 @@ func TestParserPatternmatching(t *testing.T) {
|
|||||||
for _, inputdata := range input {
|
for _, inputdata := range input {
|
||||||
for _, testdata := range tests {
|
for _, testdata := range tests {
|
||||||
testname := fmt.Sprintf("parse-%s-with-pattern-%s-inverted-%t",
|
testname := fmt.Sprintf("parse-%s-with-pattern-%s-inverted-%t",
|
||||||
inputdata.name, testdata.name, testdata.invert)
|
inputdata.name, testdata.pattern, testdata.invert)
|
||||||
t.Run(testname, func(t *testing.T) {
|
t.Run(testname, func(t *testing.T) {
|
||||||
conf := cfg.Config{
|
conf := cfg.Config{InvertMatch: testdata.invert, Pattern: testdata.pattern,
|
||||||
InvertMatch: testdata.invert,
|
Separator: inputdata.separator}
|
||||||
Patterns: testdata.patterns,
|
|
||||||
Separator: inputdata.separator,
|
|
||||||
}
|
|
||||||
|
|
||||||
_ = conf.PreparePattern(testdata.patterns)
|
_ = conf.PreparePattern(testdata.pattern)
|
||||||
|
|
||||||
readFd := strings.NewReader(strings.TrimSpace(inputdata.text))
|
readFd := strings.NewReader(strings.TrimSpace(inputdata.text))
|
||||||
gotdata, err := Parse(conf, readFd)
|
gotdata, err := Parse(conf, readFd)
|
||||||
@@ -131,7 +125,7 @@ func TestParserPatternmatching(t *testing.T) {
|
|||||||
} else {
|
} else {
|
||||||
if !reflect.DeepEqual(testdata.entries, gotdata.entries) {
|
if !reflect.DeepEqual(testdata.entries, gotdata.entries) {
|
||||||
t.Errorf("Parser returned invalid data (pattern: %s, invert: %t)\nExp: %+v\nGot: %+v\n",
|
t.Errorf("Parser returned invalid data (pattern: %s, invert: %t)\nExp: %+v\nGot: %+v\n",
|
||||||
testdata.name, testdata.invert, testdata.entries, gotdata.entries)
|
testdata.pattern, testdata.invert, testdata.entries, gotdata.entries)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,46 +0,0 @@
|
|||||||
# filtering
|
|
||||||
|
|
||||||
# a AND b
|
|
||||||
exec tablizer -r testtable.txt -H -cspecies invasive imperium
|
|
||||||
stdout 'namak'
|
|
||||||
! stdout human
|
|
||||||
|
|
||||||
# a AND !b
|
|
||||||
exec tablizer -r testtable.txt -H -cspecies invasive '/imperium/!'
|
|
||||||
stdout 'human'
|
|
||||||
! stdout namak
|
|
||||||
|
|
||||||
# a AND !b AND c
|
|
||||||
exec tablizer -r testtable.txt -H -cspecies peaceful '/imperium/!' planetary
|
|
||||||
stdout 'kenaha'
|
|
||||||
! stdout 'namak|heduu|riedl'
|
|
||||||
|
|
||||||
# case insensitive
|
|
||||||
exec tablizer -r testtable.txt -H -cspecies '/REGIONAL/i'
|
|
||||||
stdout namak
|
|
||||||
! stdout 'human|riedl|heduu|kenaa'
|
|
||||||
|
|
||||||
# case insensitive negated
|
|
||||||
exec tablizer -r testtable.txt -H -cspecies '/REGIONAL/!i'
|
|
||||||
stdout 'human|riedl|heduu|kenaa'
|
|
||||||
! stdout namak
|
|
||||||
|
|
||||||
# !a AND !b
|
|
||||||
exec tablizer -r testtable.txt -H -cspecies '/galactic/!' '/planetary/!'
|
|
||||||
stdout namak
|
|
||||||
! stdout 'human|riedl|heduu|kenaa'
|
|
||||||
|
|
||||||
# same case insensitive
|
|
||||||
exec tablizer -r testtable.txt -H -cspecies '/GALACTIC/i!' '/PLANETARY/!i'
|
|
||||||
stdout namak
|
|
||||||
! stdout 'human|riedl|heduu|kenaa'
|
|
||||||
|
|
||||||
# will be automatically created in work dir
|
|
||||||
-- testtable.txt --
|
|
||||||
SPECIES TYPE HOME STAGE SPREAD
|
|
||||||
human invasive earth brink planetary
|
|
||||||
riedl peaceful keauna civilized pangalactic
|
|
||||||
namak invasive namak imperium regional
|
|
||||||
heduu peaceful iu imperium galactic
|
|
||||||
kenaha peaceful kohi hunter-gatherer planetary
|
|
||||||
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
SPECIES TYPE HOME STAGE
|
|
||||||
human invasive earth brink
|
|
||||||
riedl peaceful keauna civilized
|
|
||||||
namak invasive namak imperium
|
|
||||||
heduu peaceful iu imperium
|
|
||||||
kenaha peaceful kohi hunter-gatherer
|
|
||||||
62
tablizer.1
62
tablizer.1
@@ -133,7 +133,7 @@
|
|||||||
.\" ========================================================================
|
.\" ========================================================================
|
||||||
.\"
|
.\"
|
||||||
.IX Title "TABLIZER 1"
|
.IX Title "TABLIZER 1"
|
||||||
.TH TABLIZER 1 "2025-01-30" "1" "User Commands"
|
.TH TABLIZER 1 "2025-01-15" "1" "User Commands"
|
||||||
.\" For nroff, turn off justification. Always turn off hyphenation; it makes
|
.\" For nroff, turn off justification. Always turn off hyphenation; it makes
|
||||||
.\" way too many mistakes in technical documents.
|
.\" way too many mistakes in technical documents.
|
||||||
.if n .ad l
|
.if n .ad l
|
||||||
@@ -144,7 +144,7 @@ tablizer \- Manipulate tabular output of other programs
|
|||||||
.IX Header "SYNOPSIS"
|
.IX Header "SYNOPSIS"
|
||||||
.Vb 2
|
.Vb 2
|
||||||
\& Usage:
|
\& Usage:
|
||||||
\& tablizer [regex,...] [file, ...] [flags]
|
\& tablizer [regex] [file, ...] [flags]
|
||||||
\&
|
\&
|
||||||
\& Operational Flags:
|
\& Operational Flags:
|
||||||
\& \-c, \-\-columns string Only show the speficied columns (separated by ,)
|
\& \-c, \-\-columns string Only show the speficied columns (separated by ,)
|
||||||
@@ -155,7 +155,7 @@ tablizer \- Manipulate tabular output of other programs
|
|||||||
\& \-s, \-\-separator string Custom field separator
|
\& \-s, \-\-separator string Custom field separator
|
||||||
\& \-k, \-\-sort\-by int|name Sort by column (default: 1)
|
\& \-k, \-\-sort\-by int|name Sort by column (default: 1)
|
||||||
\& \-z, \-\-fuzzy Use fuzzy search [experimental]
|
\& \-z, \-\-fuzzy Use fuzzy search [experimental]
|
||||||
\& \-F, \-\-filter field[!]=reg Filter given field with regex, can be used multiple times
|
\& \-F, \-\-filter field=reg Filter given field with regex, can be used multiple times
|
||||||
\& \-T, \-\-transpose\-columns string Transpose the speficied columns (separated by ,)
|
\& \-T, \-\-transpose\-columns string Transpose the speficied columns (separated by ,)
|
||||||
\& \-R, \-\-regex\-transposer /from/to/ Apply /search/replace/ regexp to fields given in \-T
|
\& \-R, \-\-regex\-transposer /from/to/ Apply /search/replace/ regexp to fields given in \-T
|
||||||
\&
|
\&
|
||||||
@@ -278,52 +278,38 @@ Finally the \fB\-d\fR option enables debugging output which is mostly
|
|||||||
useful for the developer.
|
useful for the developer.
|
||||||
.SS "\s-1PATTERNS AND FILTERING\s0"
|
.SS "\s-1PATTERNS AND FILTERING\s0"
|
||||||
.IX Subsection "PATTERNS AND FILTERING"
|
.IX Subsection "PATTERNS AND FILTERING"
|
||||||
You can reduce the rows being displayed by using one or more regular
|
You can reduce the rows being displayed by using a regular expression
|
||||||
expression patterns. The regexp language being used is the one of
|
pattern. The regexp is \s-1PCRE\s0 compatible, refer to the syntax cheat
|
||||||
\&\s-1GOLANG,\s0 refer to the syntax cheat sheet here:
|
sheet here: <https://github.com/google/re2/wiki/Syntax>. If you want
|
||||||
<https://pkg.go.dev/regexp/syntax>.
|
to read a more comprehensive documentation about the topic and have
|
||||||
.PP
|
perl installed you can read it with:
|
||||||
If you want to read a more comprehensive documentation about the
|
|
||||||
topic and have perl installed you can read it with:
|
|
||||||
.PP
|
.PP
|
||||||
.Vb 1
|
.Vb 1
|
||||||
\& perldoc perlre
|
\& perldoc perlre
|
||||||
.Ve
|
.Ve
|
||||||
.PP
|
.PP
|
||||||
Or read it online: <https://perldoc.perl.org/perlre>. But please note
|
Or read it online: <https://perldoc.perl.org/perlre>.
|
||||||
that the \s-1GO\s0 regexp engine does \s-1NOT\s0 support all perl regex terms,
|
|
||||||
especially look-ahead and look-behind.
|
|
||||||
.PP
|
.PP
|
||||||
If you want to supply flags to a regex, then surround it with slashes
|
A note on modifiers: the regexp engine used in tablizer uses another
|
||||||
and append the flag. The following flags are supported:
|
modifier syntax:
|
||||||
.PP
|
.PP
|
||||||
.Vb 2
|
.Vb 1
|
||||||
\& i => case insensitive
|
\& (?MODIFIER)
|
||||||
\& ! => negative match
|
|
||||||
.Ve
|
.Ve
|
||||||
.PP
|
.PP
|
||||||
|
The most important modifiers are:
|
||||||
|
.PP
|
||||||
|
\&\f(CW\*(C`i\*(C'\fR ignore case
|
||||||
|
\&\f(CW\*(C`m\*(C'\fR multiline mode
|
||||||
|
\&\f(CW\*(C`s\*(C'\fR single line mode
|
||||||
|
.PP
|
||||||
Example for a case insensitive search:
|
Example for a case insensitive search:
|
||||||
.PP
|
.PP
|
||||||
.Vb 1
|
.Vb 1
|
||||||
\& kubectl get pods \-A | tablizer "/account/i"
|
\& kubectl get pods \-A | tablizer "(?i)account"
|
||||||
.Ve
|
.Ve
|
||||||
.PP
|
.PP
|
||||||
If you use the \f(CW\*(C`!\*(C'\fR flag, then the regex match will be negated, that
|
You can use the experimental fuzzy search feature by providing the
|
||||||
is, if a line in the input matches the given regex, but \f(CW\*(C`!\*(C'\fR is
|
|
||||||
supplied, tablizer will \s-1NOT\s0 include it in the output.
|
|
||||||
.PP
|
|
||||||
For example, here we want to get all lines matching \*(L"foo\*(R" but not
|
|
||||||
\&\*(L"bar\*(R":
|
|
||||||
.PP
|
|
||||||
.Vb 1
|
|
||||||
\& cat table | tablizer foo \*(Aq/bar/!\*(Aq
|
|
||||||
.Ve
|
|
||||||
.PP
|
|
||||||
This would match a line \*(L"foo zorro\*(R" but not \*(L"foo bar\*(R".
|
|
||||||
.PP
|
|
||||||
The flags can also be combined.
|
|
||||||
.PP
|
|
||||||
You can also use the experimental fuzzy search feature by providing the
|
|
||||||
option \fB\-z\fR, in which case the pattern is regarded as a fuzzy search
|
option \fB\-z\fR, in which case the pattern is regarded as a fuzzy search
|
||||||
term, not a regexp.
|
term, not a regexp.
|
||||||
.PP
|
.PP
|
||||||
@@ -340,12 +326,6 @@ Fieldnames (== columns headers) are case insensitive.
|
|||||||
If you specify more than one filter, both filters have to match (\s-1AND\s0
|
If you specify more than one filter, both filters have to match (\s-1AND\s0
|
||||||
operation).
|
operation).
|
||||||
.PP
|
.PP
|
||||||
These field filters can also be negated:
|
|
||||||
.PP
|
|
||||||
.Vb 1
|
|
||||||
\& fieldname!=regexp
|
|
||||||
.Ve
|
|
||||||
.PP
|
|
||||||
If the option \fB\-v\fR is specified, the filtering is inverted.
|
If the option \fB\-v\fR is specified, the filtering is inverted.
|
||||||
.SS "\s-1COLUMNS\s0"
|
.SS "\s-1COLUMNS\s0"
|
||||||
.IX Subsection "COLUMNS"
|
.IX Subsection "COLUMNS"
|
||||||
|
|||||||
54
tablizer.pod
54
tablizer.pod
@@ -5,7 +5,7 @@ tablizer - Manipulate tabular output of other programs
|
|||||||
=head1 SYNOPSIS
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
Usage:
|
Usage:
|
||||||
tablizer [regex,...] [file, ...] [flags]
|
tablizer [regex] [file, ...] [flags]
|
||||||
|
|
||||||
Operational Flags:
|
Operational Flags:
|
||||||
-c, --columns string Only show the speficied columns (separated by ,)
|
-c, --columns string Only show the speficied columns (separated by ,)
|
||||||
@@ -16,7 +16,7 @@ tablizer - Manipulate tabular output of other programs
|
|||||||
-s, --separator string Custom field separator
|
-s, --separator string Custom field separator
|
||||||
-k, --sort-by int|name Sort by column (default: 1)
|
-k, --sort-by int|name Sort by column (default: 1)
|
||||||
-z, --fuzzy Use fuzzy search [experimental]
|
-z, --fuzzy Use fuzzy search [experimental]
|
||||||
-F, --filter field[!]=reg Filter given field with regex, can be used multiple times
|
-F, --filter field=reg Filter given field with regex, can be used multiple times
|
||||||
-T, --transpose-columns string Transpose the speficied columns (separated by ,)
|
-T, --transpose-columns string Transpose the speficied columns (separated by ,)
|
||||||
-R, --regex-transposer /from/to/ Apply /search/replace/ regexp to fields given in -T
|
-R, --regex-transposer /from/to/ Apply /search/replace/ regexp to fields given in -T
|
||||||
|
|
||||||
@@ -142,44 +142,32 @@ useful for the developer.
|
|||||||
|
|
||||||
=head2 PATTERNS AND FILTERING
|
=head2 PATTERNS AND FILTERING
|
||||||
|
|
||||||
You can reduce the rows being displayed by using one or more regular
|
You can reduce the rows being displayed by using a regular expression
|
||||||
expression patterns. The regexp language being used is the one of
|
pattern. The regexp is PCRE compatible, refer to the syntax cheat
|
||||||
GOLANG, refer to the syntax cheat sheet here:
|
sheet here: L<https://github.com/google/re2/wiki/Syntax>. If you want
|
||||||
L<https://pkg.go.dev/regexp/syntax>.
|
to read a more comprehensive documentation about the topic and have
|
||||||
|
perl installed you can read it with:
|
||||||
If you want to read a more comprehensive documentation about the
|
|
||||||
topic and have perl installed you can read it with:
|
|
||||||
|
|
||||||
perldoc perlre
|
perldoc perlre
|
||||||
|
|
||||||
Or read it online: L<https://perldoc.perl.org/perlre>. But please note
|
Or read it online: L<https://perldoc.perl.org/perlre>.
|
||||||
that the GO regexp engine does NOT support all perl regex terms,
|
|
||||||
especially look-ahead and look-behind.
|
|
||||||
|
|
||||||
If you want to supply flags to a regex, then surround it with slashes
|
A note on modifiers: the regexp engine used in tablizer uses another
|
||||||
and append the flag. The following flags are supported:
|
modifier syntax:
|
||||||
|
|
||||||
i => case insensitive
|
(?MODIFIER)
|
||||||
! => negative match
|
|
||||||
|
The most important modifiers are:
|
||||||
|
|
||||||
|
C<i> ignore case
|
||||||
|
C<m> multiline mode
|
||||||
|
C<s> single line mode
|
||||||
|
|
||||||
Example for a case insensitive search:
|
Example for a case insensitive search:
|
||||||
|
|
||||||
kubectl get pods -A | tablizer "/account/i"
|
kubectl get pods -A | tablizer "(?i)account"
|
||||||
|
|
||||||
If you use the C<!> flag, then the regex match will be negated, that
|
You can use the experimental fuzzy search feature by providing the
|
||||||
is, if a line in the input matches the given regex, but C<!> is
|
|
||||||
supplied, tablizer will NOT include it in the output.
|
|
||||||
|
|
||||||
For example, here we want to get all lines matching "foo" but not
|
|
||||||
"bar":
|
|
||||||
|
|
||||||
cat table | tablizer foo '/bar/!'
|
|
||||||
|
|
||||||
This would match a line "foo zorro" but not "foo bar".
|
|
||||||
|
|
||||||
The flags can also be combined.
|
|
||||||
|
|
||||||
You can also use the experimental fuzzy search feature by providing the
|
|
||||||
option B<-z>, in which case the pattern is regarded as a fuzzy search
|
option B<-z>, in which case the pattern is regarded as a fuzzy search
|
||||||
term, not a regexp.
|
term, not a regexp.
|
||||||
|
|
||||||
@@ -194,10 +182,6 @@ Fieldnames (== columns headers) are case insensitive.
|
|||||||
If you specify more than one filter, both filters have to match (AND
|
If you specify more than one filter, both filters have to match (AND
|
||||||
operation).
|
operation).
|
||||||
|
|
||||||
These field filters can also be negated:
|
|
||||||
|
|
||||||
fieldname!=regexp
|
|
||||||
|
|
||||||
If the option B<-v> is specified, the filtering is inverted.
|
If the option B<-v> is specified, the filtering is inverted.
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user