mirror of
https://codeberg.org/scip/tablizer.git
synced 2025-12-18 21:11:03 +01:00
Compare commits
31 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
46fde289f5 | ||
| d331f0b2e6 | |||
|
|
fc0352efa9 | ||
| 2bb0cdb0af | |||
| fad8a59023 | |||
|
|
bd5ee90324 | ||
| bc717baa3f | |||
| c34f030914 | |||
|
|
f1aa9d0000 | ||
| 736dd37f16 | |||
| e0dc6bb845 | |||
|
|
8bdb3db105 | ||
| 4ce6c30f54 | |||
|
|
ec0b210167 | ||
| 253ef8262e | |||
| da48994744 | |||
| 39f06fddc8 | |||
|
|
50a9378d92 | ||
|
|
35b726fee4 | ||
|
|
8c87da34f2 | ||
|
|
6f0f5afb27 | ||
|
|
62b606e7da | ||
|
|
567d23b175 | ||
|
|
14f24533f0 | ||
|
|
4e413c02b5 | ||
|
|
6d8c0c0936 | ||
|
|
21b607af7c | ||
|
|
06a5d74fb6 | ||
|
|
5f3f7c417c | ||
|
|
687f4b7bb2 | ||
| 24b66b8a6b |
38
.github/workflows/ci.yaml
vendored
38
.github/workflows/ci.yaml
vendored
@@ -1,38 +0,0 @@
|
|||||||
name: build-and-test-tablizer
|
|
||||||
on: [push, pull_request]
|
|
||||||
jobs:
|
|
||||||
build:
|
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
version: ['1.23']
|
|
||||||
os: [ubuntu-latest, macos-latest, windows-latest]
|
|
||||||
name: Build
|
|
||||||
runs-on: ${{ matrix.os }}
|
|
||||||
steps:
|
|
||||||
- name: Set up Go ${{ matrix.version }}
|
|
||||||
uses: actions/setup-go@v5
|
|
||||||
with:
|
|
||||||
go-version: '${{ matrix.version }}'
|
|
||||||
id: go
|
|
||||||
|
|
||||||
- name: checkout
|
|
||||||
uses: actions/checkout@v5
|
|
||||||
|
|
||||||
- name: build
|
|
||||||
run: make
|
|
||||||
|
|
||||||
- name: test
|
|
||||||
run: make test
|
|
||||||
|
|
||||||
golangci:
|
|
||||||
name: lint
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- uses: actions/setup-go@v5
|
|
||||||
with:
|
|
||||||
go-version: 1.23
|
|
||||||
- uses: actions/checkout@v5
|
|
||||||
- name: golangci-lint
|
|
||||||
uses: golangci/golangci-lint-action@v6
|
|
||||||
with:
|
|
||||||
skip-cache: true
|
|
||||||
87
.github/workflows/release.yaml
vendored
87
.github/workflows/release.yaml
vendored
@@ -1,87 +0,0 @@
|
|||||||
name: build-release
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
tags:
|
|
||||||
- "v*.*.*"
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
release:
|
|
||||||
name: Build Release Assets
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Checkout code
|
|
||||||
uses: actions/checkout@v5
|
|
||||||
|
|
||||||
- name: Set up Go
|
|
||||||
uses: actions/setup-go@v5
|
|
||||||
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
|
|
||||||
|
|
||||||
- name: Build Changelog
|
|
||||||
id: github_release
|
|
||||||
uses: mikepenz/release-changelog-builder-action@v5
|
|
||||||
env:
|
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
with:
|
|
||||||
mode: "PR"
|
|
||||||
configurationJson: |
|
|
||||||
{
|
|
||||||
"template": "#{{CHANGELOG}}\n\n**Full Changelog**: #{{RELEASE_DIFF}}",
|
|
||||||
"pr_template": "- #{{TITLE}} (##{{NUMBER}}) by #{{AUTHOR}}\n#{{BODY}}",
|
|
||||||
"empty_template": "- no changes",
|
|
||||||
"categories": [
|
|
||||||
{
|
|
||||||
"title": "## New Features",
|
|
||||||
"labels": ["add", "feature"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"title": "## Bug Fixes",
|
|
||||||
"labels": ["fix", "bug", "revert"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"title": "## Documentation Enhancements",
|
|
||||||
"labels": ["doc"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"title": "## Refactoring Efforts",
|
|
||||||
"labels": ["refactor"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"title": "## Miscellaneus Changes",
|
|
||||||
"labels": []
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"ignore_labels": [
|
|
||||||
"duplicate", "good first issue", "help wanted", "invalid", "question", "wontfix"
|
|
||||||
],
|
|
||||||
"label_extractor": [
|
|
||||||
{
|
|
||||||
"pattern": "(.) (.+)",
|
|
||||||
"target": "$1"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pattern": "(.) (.+)",
|
|
||||||
"target": "$1",
|
|
||||||
"on_property": "title"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
- name: Create Release
|
|
||||||
uses: softprops/action-gh-release@v2
|
|
||||||
with:
|
|
||||||
body: ${{steps.github_release.outputs.changelog}}
|
|
||||||
69
.goreleaser.yaml
Normal file
69
.goreleaser.yaml
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
# vim: set ts=2 sw=2 tw=0 fo=cnqoj
|
||||||
|
|
||||||
|
version: 2
|
||||||
|
|
||||||
|
before:
|
||||||
|
hooks:
|
||||||
|
- go mod tidy
|
||||||
|
|
||||||
|
gitea_urls:
|
||||||
|
api: https://codeberg.org/api/v1
|
||||||
|
download: https://codeberg.org
|
||||||
|
|
||||||
|
builds:
|
||||||
|
- env:
|
||||||
|
- CGO_ENABLED=0
|
||||||
|
goos:
|
||||||
|
- linux
|
||||||
|
- windows
|
||||||
|
- darwin
|
||||||
|
- freebsd
|
||||||
|
|
||||||
|
archives:
|
||||||
|
- formats: [tar.gz]
|
||||||
|
# this name template makes the OS and Arch compatible with the results of `uname`.
|
||||||
|
name_template: >-
|
||||||
|
{{ .ProjectName }}_
|
||||||
|
{{- title .Os }}_
|
||||||
|
{{- if eq .Arch "amd64" }}x86_64
|
||||||
|
{{- else if eq .Arch "386" }}i386
|
||||||
|
{{- else }}{{ .Arch }}{{ end }}
|
||||||
|
{{- if .Arm }}v{{ .Arm }}{{ end }}_{{ .Tag }}
|
||||||
|
# use zip for windows archives
|
||||||
|
format_overrides:
|
||||||
|
- goos: windows
|
||||||
|
formats: [zip]
|
||||||
|
- goos: linux
|
||||||
|
formats: [tar.gz,binary]
|
||||||
|
files:
|
||||||
|
- src: "*.md"
|
||||||
|
strip_parent: true
|
||||||
|
- src: "docs/*"
|
||||||
|
strip_parent: true
|
||||||
|
- src: Makefile.dist
|
||||||
|
dst: Makefile
|
||||||
|
wrap_in_directory: true
|
||||||
|
|
||||||
|
changelog:
|
||||||
|
sort: asc
|
||||||
|
filters:
|
||||||
|
exclude:
|
||||||
|
- "^docs:"
|
||||||
|
- "^test:"
|
||||||
|
groups:
|
||||||
|
- title: Improved
|
||||||
|
regexp: '^.*?(feat|add|new)(\([[:word:]]+\))??!?:.+$'
|
||||||
|
order: 0
|
||||||
|
- title: Fixed
|
||||||
|
regexp: '^.*?(bug|fix)(\([[:word:]]+\))??!?:.+$'
|
||||||
|
order: 1
|
||||||
|
- title: Changed
|
||||||
|
order: 999
|
||||||
|
|
||||||
|
release:
|
||||||
|
header: "# Release Notes"
|
||||||
|
footer: >-
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
Full Changelog: [{{ .PreviousTag }}...{{ .Tag }}](https://codeberg.org/scip/tablizer/compare/{{ .PreviousTag }}...{{ .Tag }})
|
||||||
36
.woodpecker/build.yaml
Normal file
36
.woodpecker/build.yaml
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
matrix:
|
||||||
|
platform:
|
||||||
|
- linux/amd64
|
||||||
|
goversion:
|
||||||
|
- 1.24
|
||||||
|
|
||||||
|
labels:
|
||||||
|
platform: ${platform}
|
||||||
|
|
||||||
|
steps:
|
||||||
|
build:
|
||||||
|
when:
|
||||||
|
event: [push]
|
||||||
|
image: golang:${goversion}
|
||||||
|
commands:
|
||||||
|
- go get
|
||||||
|
- go build
|
||||||
|
|
||||||
|
linter:
|
||||||
|
when:
|
||||||
|
event: [push]
|
||||||
|
image: golang:${goversion}
|
||||||
|
commands:
|
||||||
|
- curl -sSfL https://raw.githubusercontent.com/golangci/golangci-lint/HEAD/install.sh | sh -s -- -b $(go env GOPATH)/bin v2.5.0
|
||||||
|
- golangci-lint --version
|
||||||
|
- golangci-lint run ./...
|
||||||
|
depends_on: [build]
|
||||||
|
|
||||||
|
test:
|
||||||
|
when:
|
||||||
|
event: [push]
|
||||||
|
image: golang:${goversion}
|
||||||
|
commands:
|
||||||
|
- go get
|
||||||
|
- go test -v -cover
|
||||||
|
depends_on: [build,linter]
|
||||||
15
.woodpecker/release.yaml
Normal file
15
.woodpecker/release.yaml
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
# build release
|
||||||
|
|
||||||
|
labels:
|
||||||
|
platform: linux/amd64
|
||||||
|
|
||||||
|
steps:
|
||||||
|
goreleaser:
|
||||||
|
image: goreleaser/goreleaser
|
||||||
|
when:
|
||||||
|
event: [tag]
|
||||||
|
environment:
|
||||||
|
GITEA_TOKEN:
|
||||||
|
from_secret: DEPLOY_TOKEN
|
||||||
|
commands:
|
||||||
|
- goreleaser release --clean --verbose
|
||||||
62
CHANGELOG.md
62
CHANGELOG.md
@@ -4,9 +4,9 @@ All notable changes to this project will be documented in this file.
|
|||||||
|
|
||||||
The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) and this project adheres to [Semantic Versioning](http://semver.org).
|
The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) and this project adheres to [Semantic Versioning](http://semver.org).
|
||||||
|
|
||||||
## [v1.0.14](https://github.com/TLINDEN/tablizer/tree/v1.0.14) - 2023-01-23
|
## [v1.0.14](https://codeberg.org/scip/tablizer/tree/v1.0.14) - 2023-01-23
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/v1.0.13...v1.0.14)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/v1.0.13...v1.0.14)
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
@@ -28,9 +28,9 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) a
|
|||||||
- updated dependencies (go module versions)
|
- updated dependencies (go module versions)
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.13](https://github.com/TLINDEN/tablizer/tree/v1.0.13) - 2022-11-03
|
## [v1.0.13](https://codeberg.org/scip/tablizer/tree/v1.0.13) - 2022-11-03
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/v1.0.12...v1.0.13)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/v1.0.12...v1.0.13)
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
@@ -47,9 +47,9 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) a
|
|||||||
manpage.
|
manpage.
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.12](https://github.com/TLINDEN/tablizer/tree/v1.0.12) - 2022-10-25
|
## [v1.0.12](https://codeberg.org/scip/tablizer/tree/v1.0.12) - 2022-10-25
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/v1.0.11...v1.0.12)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/v1.0.11...v1.0.12)
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
@@ -72,14 +72,14 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) a
|
|||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
- Fixed [Bug #5](https://github.com/TLINDEN/tablizer/issues/5), where
|
- Fixed [Bug #5](https://codeberg.org/scip/tablizer/issues/5), where
|
||||||
matches have not been highlighted correctly in some rare cases.
|
matches have not been highlighted correctly in some rare cases.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.11](https://github.com/TLINDEN/tablizer/tree/v1.0.11) - 2022-10-19
|
## [v1.0.11](https://codeberg.org/scip/tablizer/tree/v1.0.11) - 2022-10-19
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/v1.0.10...v1.0.11)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/v1.0.10...v1.0.11)
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
@@ -101,9 +101,9 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) a
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.10](https://github.com/TLINDEN/tablizer/tree/v1.0.10) - 2022-10-15
|
## [v1.0.10](https://codeberg.org/scip/tablizer/tree/v1.0.10) - 2022-10-15
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/v1.0.9...v1.0.10)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/v1.0.9...v1.0.10)
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
@@ -123,9 +123,9 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) a
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.9](https://github.com/TLINDEN/tablizer/tree/v1.0.9) - 2022-10-14
|
## [v1.0.9](https://codeberg.org/scip/tablizer/tree/v1.0.9) - 2022-10-14
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/v1.0.8...v1.0.9)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/v1.0.8...v1.0.9)
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
@@ -137,9 +137,9 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) a
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.8](https://github.com/TLINDEN/tablizer/tree/v1.0.8) - 2022-10-13
|
## [v1.0.8](https://codeberg.org/scip/tablizer/tree/v1.0.8) - 2022-10-13
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/v1.0.7...v1.0.8)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/v1.0.7...v1.0.8)
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
@@ -147,9 +147,9 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) a
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.7](https://github.com/TLINDEN/tablizer/tree/v1.0.7) - 2022-10-11
|
## [v1.0.7](https://codeberg.org/scip/tablizer/tree/v1.0.7) - 2022-10-11
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/v1.0.6...v1.0.7)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/v1.0.6...v1.0.7)
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
@@ -165,9 +165,9 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) a
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.6](https://github.com/TLINDEN/tablizer/tree/v1.0.6) - 2022-10-05
|
## [v1.0.6](https://codeberg.org/scip/tablizer/tree/v1.0.6) - 2022-10-05
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/v1.0.5...v1.0.6)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/v1.0.5...v1.0.6)
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
@@ -183,9 +183,9 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) a
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.5](https://github.com/TLINDEN/tablizer/tree/v1.0.5) - 2022-10-05
|
## [v1.0.5](https://codeberg.org/scip/tablizer/tree/v1.0.5) - 2022-10-05
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/v1.0.4...v1.0.5)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/v1.0.4...v1.0.5)
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
@@ -203,9 +203,9 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) a
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.4](https://github.com/TLINDEN/tablizer/tree/v1.0.4) - 2022-10-04
|
## [v1.0.4](https://codeberg.org/scip/tablizer/tree/v1.0.4) - 2022-10-04
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/v1.0.3...v1.0.4)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/v1.0.3...v1.0.4)
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
@@ -221,9 +221,9 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) a
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.3](https://github.com/TLINDEN/tablizer/tree/v1.0.3) - 2022-10-03
|
## [v1.0.3](https://codeberg.org/scip/tablizer/tree/v1.0.3) - 2022-10-03
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/v1.0.2...v1.0.3)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/v1.0.2...v1.0.3)
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
@@ -237,9 +237,9 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) a
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.2](https://github.com/TLINDEN/tablizer/tree/v1.0.2) - 2022-10-02
|
## [v1.0.2](https://codeberg.org/scip/tablizer/tree/v1.0.2) - 2022-10-02
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/v1.0.1...v1.0.2)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/v1.0.1...v1.0.2)
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
@@ -255,9 +255,9 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) a
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.1](https://github.com/TLINDEN/tablizer/tree/v1.0.1) - 2022-09-30
|
## [v1.0.1](https://codeberg.org/scip/tablizer/tree/v1.0.1) - 2022-09-30
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/v1.0.0...v1.0.1)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/v1.0.0...v1.0.1)
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
@@ -271,8 +271,8 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) a
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.0](https://github.com/TLINDEN/tablizer/tree/v1.0.0) - 2022-09-28
|
## [v1.0.0](https://codeberg.org/scip/tablizer/tree/v1.0.0) - 2022-09-28
|
||||||
|
|
||||||
[Full Changelog](https://github.com/TLINDEN/tablizer/compare/02a64a5c3fe4220df2c791ff1421d16ebd428c19...v1.0.0)
|
[Full Changelog](https://codeberg.org/scip/tablizer/compare/02a64a5c3fe4220df2c791ff1421d16ebd428c19...v1.0.0)
|
||||||
|
|
||||||
Initial release.
|
Initial release.
|
||||||
|
|||||||
6
Makefile
6
Makefile
@@ -50,7 +50,7 @@ ifdef HAVE_POD
|
|||||||
endif
|
endif
|
||||||
|
|
||||||
buildlocal:
|
buildlocal:
|
||||||
go build -ldflags "-X 'github.com/tlinden/tablizer/cfg.VERSION=$(VERSION)'"
|
go build -ldflags "-X 'codeberg.org/scip/tablizer/cfg.VERSION=$(VERSION)'"
|
||||||
|
|
||||||
release:
|
release:
|
||||||
gh release create $(version) --generate-notes
|
gh release create $(version) --generate-notes
|
||||||
@@ -65,11 +65,11 @@ clean:
|
|||||||
rm -rf $(tool) releases coverage.out
|
rm -rf $(tool) releases coverage.out
|
||||||
|
|
||||||
test: clean
|
test: clean
|
||||||
go test ./... $(OPTS)
|
go test -count=1 -cover ./... $(OPTS)
|
||||||
|
|
||||||
singletest:
|
singletest:
|
||||||
@echo "Call like this: 'make singletest TEST=TestPrepareColumns MOD=lib'"
|
@echo "Call like this: 'make singletest TEST=TestPrepareColumns MOD=lib'"
|
||||||
go test -run $(TEST) github.com/tlinden/tablizer/$(MOD) $(OPTS)
|
go test -run $(TEST) codeberg.org/scip/tablizer/$(MOD) $(OPTS)
|
||||||
|
|
||||||
cover-report:
|
cover-report:
|
||||||
go test ./... -cover -coverprofile=coverage.out
|
go test ./... -cover -coverprofile=coverage.out
|
||||||
|
|||||||
20
Makefile.dist
Normal file
20
Makefile.dist
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
# -*-make-*-
|
||||||
|
|
||||||
|
.PHONY: install all
|
||||||
|
|
||||||
|
tool = rpn
|
||||||
|
PREFIX = /usr/local
|
||||||
|
UID = root
|
||||||
|
GID = 0
|
||||||
|
|
||||||
|
all:
|
||||||
|
@echo "Type 'sudo make install' to install the tool."
|
||||||
|
@echo "To change prefix, type 'sudo make install PREFIX=/opt'"
|
||||||
|
|
||||||
|
install:
|
||||||
|
install -d -o $(UID) -g $(GID) $(PREFIX)/bin
|
||||||
|
install -d -o $(UID) -g $(GID) $(PREFIX)/man/man1
|
||||||
|
install -d -o $(UID) -g $(GID) $(PREFIX)/share/doc
|
||||||
|
install -o $(UID) -g $(GID) -m 555 $(tool) $(PREFIX)/sbin/
|
||||||
|
install -o $(UID) -g $(GID) -m 444 $(tool).1 $(PREFIX)/man/man1/
|
||||||
|
install -o $(UID) -g $(GID) -m 444 *.md $(PREFIX)/share/doc/
|
||||||
136
README.md
136
README.md
@@ -1,6 +1,6 @@
|
|||||||
[](https://github.com/tlinden/tablizer/actions)
|
[](https://ci.codeberg.org/repos/15519)
|
||||||
[](https://github.com/tlinden/tablizer/blob/master/LICENSE)
|
[](https://codeberg.org/scip/tablizer/blob/master/LICENSE)
|
||||||
[](https://goreportcard.com/report/github.com/tlinden/tablizer)
|
[](https://goreportcard.com/report/codeberg.org/scip/tablizer)
|
||||||
|
|
||||||
## tablizer - Manipulate tabular output of other programs
|
## tablizer - Manipulate tabular output of other programs
|
||||||
|
|
||||||
@@ -11,50 +11,79 @@ ignore certain column[s] by regex, name or number. It can output the
|
|||||||
tabular data in a range of formats (see below). There's even an
|
tabular data in a range of formats (see below). There's even an
|
||||||
interactive filter/selection tool available.
|
interactive filter/selection tool available.
|
||||||
|
|
||||||
Usage:
|
## FEATURES
|
||||||
|
|
||||||
|
- supports csv, json or ascii format input from files or stdin
|
||||||
|
- split any tabular input data by character or regular expression into columns
|
||||||
|
- add headers if input data doesn't contain them (automatically or manually)
|
||||||
|
- print tabular data as ascii table, org-mode, markdown, csv, shell-evaluable or yaml format
|
||||||
|
- filter rows by regular expression (saves a call to `| grep ...`)
|
||||||
|
- filter rows by column filter
|
||||||
|
- filters may also be negations eg `-Fname!=cow.*` or `-v`
|
||||||
|
- modify cells wih regular expressions
|
||||||
|
- reduce columns by specifying which columns to show, with regex support
|
||||||
|
- color support
|
||||||
|
- sort by any field[s], multiple sort modes are supported
|
||||||
|
- shell completion for options
|
||||||
|
- regular used options can be put into a config file
|
||||||
|
- filter TUI where where you can interactively sort and filter rows
|
||||||
|
|
||||||
|
## Demo
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
```default
|
```default
|
||||||
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 ,)
|
||||||
-v, --invert-match select non-matching rows
|
-v, --invert-match select non-matching rows
|
||||||
-n, --numbering Enable header numbering
|
-n, --numbering Enable header numbering
|
||||||
-N, --no-color Disable pattern highlighting
|
-N, --no-color Disable pattern highlighting
|
||||||
-H, --no-headers Disable headers display
|
-H, --no-headers Disable headers display
|
||||||
-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
|
||||||
-I, --interactive Interactively filter and select rows
|
-j, --json Read JSON input (must be array of hashes)
|
||||||
|
-I, --interactive Interactively filter and select rows
|
||||||
|
--auto-headers Generate headers if there are none present in input
|
||||||
|
--custom-headers a,b,... Use custom headers, separated by comma
|
||||||
|
|
||||||
|
|
||||||
Output Flags (mutually exclusive):
|
Output Flags (mutually exclusive):
|
||||||
-X, --extended Enable extended output
|
-X, --extended Enable extended output
|
||||||
-M, --markdown Enable markdown table output
|
-M, --markdown Enable markdown table output
|
||||||
-O, --orgtbl Enable org-mode table output
|
-O, --orgtbl Enable org-mode table output
|
||||||
-S, --shell Enable shell evaluable output
|
-S, --shell Enable shell evaluable output
|
||||||
-Y, --yaml Enable yaml output
|
-Y, --yaml Enable yaml output
|
||||||
-C, --csv Enable CSV output
|
-C, --csv Enable CSV output
|
||||||
-A, --ascii Default output mode, ascii tabular
|
-A, --ascii Default output mode, ascii tabular
|
||||||
-L, --hightlight-lines Use alternating background colors for tables
|
-L, --hightlight-lines Use alternating background colors for tables
|
||||||
-y, --yank-columns Yank specified columns (separated by ,) to clipboard,
|
-y, --yank-columns Yank specified columns (separated by ,) to clipboard,
|
||||||
space separated
|
space separated
|
||||||
|
--ofs <char> Output field separator, used by -A and -C.
|
||||||
|
|
||||||
Sort Mode Flags (mutually exclusive):
|
Sort Mode Flags (mutually exclusive):
|
||||||
-a, --sort-age sort according to age (duration) string
|
-a, --sort-age sort according to age (duration) string
|
||||||
-D, --sort-desc Sort in descending order (default: ascending)
|
-D, --sort-desc Sort in descending order (default: ascending)
|
||||||
-i, --sort-numeric sort according to string numerical value
|
-i, --sort-numeric sort according to string numerical value
|
||||||
-t, --sort-time sort according to time string
|
-t, --sort-time sort according to time string
|
||||||
|
|
||||||
Other Flags:
|
Other Flags:
|
||||||
--completion <shell> Generate the autocompletion script for <shell>
|
-r --read-file <file> Use <file> as input instead of STDIN
|
||||||
-f, --config <file> Configuration file (default: ~/.config/tablizer/config)
|
--completion <shell> Generate the autocompletion script for <shell>
|
||||||
-d, --debug Enable debugging
|
-f, --config <file> Configuration file (default: ~/.config/tablizer/config)
|
||||||
-h, --help help for tablizer
|
-d, --debug Enable debugging
|
||||||
-m, --man Display manual page
|
-h, --help help for tablizer
|
||||||
-V, --version Print program version
|
-m, --man Display manual page
|
||||||
|
-V, --version Print program version
|
||||||
```
|
```
|
||||||
|
|
||||||
Let's take this output:
|
Let's take this output:
|
||||||
@@ -71,13 +100,13 @@ to do this with tablizer:
|
|||||||
|
|
||||||
```
|
```
|
||||||
% kubectl get pods | tablizer
|
% kubectl get pods | tablizer
|
||||||
NAME(1) READY(2) STATUS(3) RESTARTS(4) AGE(5)
|
NAME READY STATUS RESTARTS AGE
|
||||||
repldepl-7bcd8d5b64-7zq4l 1/1 Running 1 (69m ago) 5h26m
|
repldepl-7bcd8d5b64-7zq4l 1/1 Running 1 (69m ago) 5h26m
|
||||||
repldepl-7bcd8d5b64-m48n8 1/1 Running 1 (69m ago) 5h26m
|
repldepl-7bcd8d5b64-m48n8 1/1 Running 1 (69m ago) 5h26m
|
||||||
repldepl-7bcd8d5b64-q2bf4 1/1 Running 1 (69m ago) 5h26m
|
repldepl-7bcd8d5b64-q2bf4 1/1 Running 1 (69m ago) 5h26m
|
||||||
|
|
||||||
% kubectl get pods | tablizer -c 1,3
|
% kubectl get pods | tablizer -c 1,3
|
||||||
NAME(1) STATUS(3)
|
NAME STATUS
|
||||||
repldepl-7bcd8d5b64-7zq4l Running
|
repldepl-7bcd8d5b64-7zq4l Running
|
||||||
repldepl-7bcd8d5b64-m48n8 Running
|
repldepl-7bcd8d5b64-m48n8 Running
|
||||||
repldepl-7bcd8d5b64-q2bf4 Running
|
repldepl-7bcd8d5b64-q2bf4 Running
|
||||||
@@ -115,14 +144,14 @@ You can also specify a regex pattern to reduce the output:
|
|||||||
|
|
||||||
```
|
```
|
||||||
% kubectl get pods | tablizer q2bf4
|
% kubectl get pods | tablizer q2bf4
|
||||||
NAME(1) READY(2) STATUS(3) RESTARTS(4) AGE(5)
|
NAME READY STATUS RESTARTS AGE
|
||||||
repldepl-7bcd8d5b64-q2bf4 1/1 Running 1 (69m ago) 5h26m
|
repldepl-7bcd8d5b64-q2bf4 1/1 Running 1 (69m ago) 5h26m
|
||||||
```
|
```
|
||||||
|
|
||||||
Sometimes a filter regex is to broad and you wish to filter only on a
|
Sometimes a filter regex is to broad and you wish to filter only on a
|
||||||
particular column. This is possible using `-F`:
|
particular column. This is possible using `-F`:
|
||||||
```
|
```
|
||||||
% kubectl get pods | tablizer -n -Fname=2
|
% kubectl get pods | tablizer -Fname=2
|
||||||
NAME READY STATUS RESTARTS AGE
|
NAME READY STATUS RESTARTS AGE
|
||||||
repldepl-7bcd8d5b64-q2bf4 1/1 Running 1 (69m ago) 5h26m
|
repldepl-7bcd8d5b64-q2bf4 1/1 Running 1 (69m ago) 5h26m
|
||||||
```
|
```
|
||||||
@@ -136,7 +165,7 @@ You can also use it to modify certain cells using regular expression
|
|||||||
matching. For example:
|
matching. For example:
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
kubectl get pods | tablizer -n -T4 -R '/ /-/'
|
kubectl get pods | tablizer -T4 -R '/ /-/'
|
||||||
NAME READY STATUS RESTARTS AGE
|
NAME READY STATUS RESTARTS AGE
|
||||||
repldepl-7bcd8d5b64-7zq4l 1/1 Running 1-(69m-ago) 5h26m
|
repldepl-7bcd8d5b64-7zq4l 1/1 Running 1-(69m-ago) 5h26m
|
||||||
repldepl-7bcd8d5b64-m48n8 1/1 Running 1-(69m-ago) 5h26m
|
repldepl-7bcd8d5b64-m48n8 1/1 Running 1-(69m-ago) 5h26m
|
||||||
@@ -147,17 +176,23 @@ Here, we modified the 4th column (`-T4`) by replacing every space with
|
|||||||
a dash. If you need to work with `/` characters, you can also use any
|
a dash. If you need to work with `/` characters, you can also use any
|
||||||
other separator, for instance: `-R '| |-|'`.
|
other separator, for instance: `-R '| |-|'`.
|
||||||
|
|
||||||
|
There's also an interactive mode, invoked with the option B<-I>, where
|
||||||
|
you can interactively filter and select rows:
|
||||||
|
|
||||||
|
<img width="937" height="293" alt="interactive" src="https://github.com/user-attachments/assets/0d4d65e2-d156-43ed-8021-39047c7939ed" />
|
||||||
|
|
||||||
|
|
||||||
## Demo
|
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
There are multiple ways to install **tablizer**:
|
There are multiple ways to install **tablizer**:
|
||||||
|
|
||||||
- Go to the [latest release page](https://github.com/tlinden/tablizer/releases/latest),
|
- You can use [stew](https://github.com/marwanhawari/stew) to install tablizer:
|
||||||
|
```default
|
||||||
|
stew install tlinden/tablizer
|
||||||
|
```
|
||||||
|
|
||||||
|
- Go to the [latest release page](https://codeberg.org/scip/tablizer/releases),
|
||||||
locate the binary for your operating system and platform.
|
locate the binary for your operating system and platform.
|
||||||
|
|
||||||
Download it and put it into some directory within your `$PATH` variable.
|
Download it and put it into some directory within your `$PATH` variable.
|
||||||
@@ -170,7 +205,7 @@ There are multiple ways to install **tablizer**:
|
|||||||
|
|
||||||
- You can also install from source. Issue the following commands in your shell:
|
- You can also install from source. Issue the following commands in your shell:
|
||||||
```
|
```
|
||||||
git clone https://github.com/TLINDEN/tablizer.git
|
git clone https://codeberg.org/scip/tablizer.git
|
||||||
cd tablizer
|
cd tablizer
|
||||||
make
|
make
|
||||||
sudo make install
|
sudo make install
|
||||||
@@ -182,10 +217,9 @@ hesitate to ask me about it, I'll add it.
|
|||||||
## Documentation
|
## Documentation
|
||||||
|
|
||||||
The documentation is provided as a unix man-page. It will be
|
The documentation is provided as a unix man-page. It will be
|
||||||
automatically installed if you install from source. However, you can
|
automatically installed if you install from source.
|
||||||
read the man-page online:
|
|
||||||
|
|
||||||
https://github.com/TLINDEN/tablizer/blob/main/tablizer.pod
|
[However, you can read the man-page online](https://codeberg.org/scip/tablizer/raw/branch/main/tablizer.pod).
|
||||||
|
|
||||||
Or if you cloned the repository you can read it this way (perl needs
|
Or if you cloned the repository you can read it this way (perl needs
|
||||||
to be installed though): `perldoc tablizer.pod`.
|
to be installed though): `perldoc tablizer.pod`.
|
||||||
@@ -202,7 +236,7 @@ that's the best way for me to forget to do something.
|
|||||||
|
|
||||||
In order to report a bug, unexpected behavior, feature requests
|
In order to report a bug, unexpected behavior, feature requests
|
||||||
or to submit a patch, please open an issue on github:
|
or to submit a patch, please open an issue on github:
|
||||||
https://github.com/TLINDEN/tablizer/issues.
|
https://codeberg.org/scip/tablizer/issues.
|
||||||
|
|
||||||
## Prior Art
|
## Prior Art
|
||||||
|
|
||||||
@@ -249,4 +283,4 @@ T.v.Dein <tom AT vondein DOT org>
|
|||||||
|
|
||||||
## Project homepage
|
## Project homepage
|
||||||
|
|
||||||
https://github.com/TLINDEN/tablizer
|
https://codeberg.org/scip/tablizer
|
||||||
|
|||||||
@@ -27,13 +27,26 @@ import (
|
|||||||
"github.com/hashicorp/hcl/v2/hclsimple"
|
"github.com/hashicorp/hcl/v2/hclsimple"
|
||||||
)
|
)
|
||||||
|
|
||||||
const DefaultSeparator string = `(\s\s+|\t)`
|
const (
|
||||||
const Version string = "v1.5.3"
|
Version = "v1.5.12"
|
||||||
const MAXPARTS = 2
|
MAXPARTS = 2
|
||||||
|
)
|
||||||
|
|
||||||
var DefaultConfigfile = os.Getenv("HOME") + "/.config/tablizer/config"
|
var (
|
||||||
|
DefaultConfigfile = os.Getenv("HOME") + "/.config/tablizer/config"
|
||||||
|
VERSION string // maintained by -x
|
||||||
|
|
||||||
var VERSION string // maintained by -x
|
SeparatorTemplates = map[string]string{
|
||||||
|
":tab:": `\s*\t\s*`, // tab but eats spaces around
|
||||||
|
":spaces:": `\s{2,}`, // 2 or more spaces
|
||||||
|
":pipe:": `\s*\|\s*`, // one pipe eating spaces around
|
||||||
|
":default:": `(\s\s+|\t)`, // 2 or more spaces or tab
|
||||||
|
":nonword:": `\W`, // word boundary
|
||||||
|
":nondigit:": `\D`, // same for numbers
|
||||||
|
":special:": `[\*\+\-_\(\)\[\]\{\}?\\/<>=&$§"':,\^]+`, // match any special char
|
||||||
|
":nonprint:": `[[:^print:]]+`, // non printables
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
// public config, set via config file or using defaults
|
// public config, set via config file or using defaults
|
||||||
type Settings struct {
|
type Settings struct {
|
||||||
@@ -79,6 +92,10 @@ type Config struct {
|
|||||||
UseFuzzySearch bool
|
UseFuzzySearch bool
|
||||||
UseHighlight bool
|
UseHighlight bool
|
||||||
Interactive bool
|
Interactive bool
|
||||||
|
InputJSON bool
|
||||||
|
AutoHeaders bool
|
||||||
|
CustomHeaders []string
|
||||||
|
Template string
|
||||||
|
|
||||||
SortMode string
|
SortMode string
|
||||||
SortDescending bool
|
SortDescending bool
|
||||||
@@ -112,6 +129,8 @@ type Config struct {
|
|||||||
|
|
||||||
// -r <file>
|
// -r <file>
|
||||||
InputFile string
|
InputFile string
|
||||||
|
|
||||||
|
OFS string
|
||||||
}
|
}
|
||||||
|
|
||||||
// maps outputmode short flags to output mode, ie. -O => -o orgtbl
|
// maps outputmode short flags to output mode, ie. -O => -o orgtbl
|
||||||
@@ -123,6 +142,8 @@ type Modeflag struct {
|
|||||||
Y bool
|
Y bool
|
||||||
A bool
|
A bool
|
||||||
C bool
|
C bool
|
||||||
|
J bool
|
||||||
|
P bool // template
|
||||||
}
|
}
|
||||||
|
|
||||||
// used for switching printers
|
// used for switching printers
|
||||||
@@ -134,6 +155,8 @@ const (
|
|||||||
Yaml
|
Yaml
|
||||||
CSV
|
CSV
|
||||||
ASCII
|
ASCII
|
||||||
|
Json
|
||||||
|
Template
|
||||||
)
|
)
|
||||||
|
|
||||||
// various sort types
|
// various sort types
|
||||||
@@ -272,6 +295,10 @@ func (conf *Config) PrepareModeFlags(flag Modeflag) {
|
|||||||
conf.OutputMode = Yaml
|
conf.OutputMode = Yaml
|
||||||
case flag.C:
|
case flag.C:
|
||||||
conf.OutputMode = CSV
|
conf.OutputMode = CSV
|
||||||
|
case flag.J:
|
||||||
|
conf.OutputMode = Json
|
||||||
|
case conf.Template != "":
|
||||||
|
conf.OutputMode = Template
|
||||||
default:
|
default:
|
||||||
conf.OutputMode = ASCII
|
conf.OutputMode = ASCII
|
||||||
}
|
}
|
||||||
@@ -353,6 +380,13 @@ func (conf *Config) ApplyDefaults() {
|
|||||||
if conf.OutputMode == Yaml || conf.OutputMode == CSV {
|
if conf.OutputMode == Yaml || conf.OutputMode == CSV {
|
||||||
conf.Numbering = false
|
conf.Numbering = false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if conf.Separator[0] == ':' && conf.Separator[len(conf.Separator)-1] == ':' {
|
||||||
|
separator, ok := SeparatorTemplates[conf.Separator]
|
||||||
|
if ok {
|
||||||
|
conf.Separator = separator
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (conf *Config) PreparePattern(patterns []*Pattern) error {
|
func (conf *Config) PreparePattern(patterns []*Pattern) error {
|
||||||
@@ -390,6 +424,12 @@ func (conf *Config) PreparePattern(patterns []*Pattern) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (conf *Config) PrepareCustomHeaders(custom string) {
|
||||||
|
if len(custom) > 0 {
|
||||||
|
conf.CustomHeaders = strings.Split(custom, ",")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Parse config file. Ignore if the file doesn't exist but return an
|
// Parse config file. Ignore if the file doesn't exist but return an
|
||||||
// error if it exists but fails to read or parse
|
// error if it exists but fails to read or parse
|
||||||
func (conf *Config) ParseConfigfile() error {
|
func (conf *Config) ParseConfigfile() error {
|
||||||
|
|||||||
@@ -21,6 +21,8 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
// "reflect"
|
// "reflect"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestPrepareModeFlags(t *testing.T) {
|
func TestPrepareModeFlags(t *testing.T) {
|
||||||
@@ -44,9 +46,8 @@ func TestPrepareModeFlags(t *testing.T) {
|
|||||||
conf := Config{}
|
conf := Config{}
|
||||||
|
|
||||||
conf.PrepareModeFlags(testdata.flag)
|
conf.PrepareModeFlags(testdata.flag)
|
||||||
if conf.OutputMode != testdata.expect {
|
|
||||||
t.Errorf("got: %d, expect: %d", conf.OutputMode, testdata.expect)
|
assert.EqualValues(t, testdata.expect, conf.OutputMode)
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -70,9 +71,7 @@ func TestPrepareSortFlags(t *testing.T) {
|
|||||||
|
|
||||||
conf.PrepareSortFlags(testdata.flag)
|
conf.PrepareSortFlags(testdata.flag)
|
||||||
|
|
||||||
if conf.SortMode != testdata.expect {
|
assert.EqualValues(t, testdata.expect, conf.SortMode)
|
||||||
t.Errorf("got: %s, expect: %s", conf.SortMode, testdata.expect)
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -81,7 +80,7 @@ func TestPreparePattern(t *testing.T) {
|
|||||||
var tests = []struct {
|
var tests = []struct {
|
||||||
patterns []*Pattern
|
patterns []*Pattern
|
||||||
name string
|
name string
|
||||||
wanterr bool
|
wanterror bool
|
||||||
wanticase bool
|
wanticase bool
|
||||||
wantneg bool
|
wantneg bool
|
||||||
}{
|
}{
|
||||||
@@ -123,16 +122,16 @@ func TestPreparePattern(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
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.name, testdata.wanterror)
|
||||||
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.patterns)
|
||||||
|
|
||||||
if err != nil {
|
if testdata.wanterror {
|
||||||
if !testdata.wanterr {
|
assert.Error(t, err)
|
||||||
t.Errorf("PreparePattern returned error: %s", err)
|
} else {
|
||||||
}
|
assert.NoError(t, err)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|||||||
28
cmd/root.go
28
cmd/root.go
@@ -20,11 +20,12 @@ import (
|
|||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
|
"slices"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
|
"codeberg.org/scip/tablizer/lib"
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
"github.com/tlinden/tablizer/cfg"
|
|
||||||
"github.com/tlinden/tablizer/lib"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func completion(cmd *cobra.Command, mode string) error {
|
func completion(cmd *cobra.Command, mode string) error {
|
||||||
@@ -58,6 +59,7 @@ func Execute() {
|
|||||||
ShowCompletion string
|
ShowCompletion string
|
||||||
modeflag cfg.Modeflag
|
modeflag cfg.Modeflag
|
||||||
sortmode cfg.Sortmode
|
sortmode cfg.Sortmode
|
||||||
|
headers string
|
||||||
)
|
)
|
||||||
|
|
||||||
var rootCmd = &cobra.Command{
|
var rootCmd = &cobra.Command{
|
||||||
@@ -90,6 +92,7 @@ func Execute() {
|
|||||||
conf.CheckEnv()
|
conf.CheckEnv()
|
||||||
conf.PrepareModeFlags(modeflag)
|
conf.PrepareModeFlags(modeflag)
|
||||||
conf.PrepareSortFlags(sortmode)
|
conf.PrepareSortFlags(sortmode)
|
||||||
|
conf.PrepareCustomHeaders(headers)
|
||||||
|
|
||||||
wrapE(conf.PrepareFilters())
|
wrapE(conf.PrepareFilters())
|
||||||
|
|
||||||
@@ -122,7 +125,7 @@ func Execute() {
|
|||||||
"Use alternating background colors")
|
"Use alternating background colors")
|
||||||
rootCmd.PersistentFlags().StringVarP(&ShowCompletion, "completion", "", "",
|
rootCmd.PersistentFlags().StringVarP(&ShowCompletion, "completion", "", "",
|
||||||
"Display completion code")
|
"Display completion code")
|
||||||
rootCmd.PersistentFlags().StringVarP(&conf.Separator, "separator", "s", cfg.DefaultSeparator,
|
rootCmd.PersistentFlags().StringVarP(&conf.Separator, "separator", "s", cfg.SeparatorTemplates[":default:"],
|
||||||
"Custom field separator")
|
"Custom field separator")
|
||||||
rootCmd.PersistentFlags().StringVarP(&conf.Columns, "columns", "c", "",
|
rootCmd.PersistentFlags().StringVarP(&conf.Columns, "columns", "c", "",
|
||||||
"Only show the speficied columns (separated by ,)")
|
"Only show the speficied columns (separated by ,)")
|
||||||
@@ -131,7 +134,15 @@ func Execute() {
|
|||||||
rootCmd.PersistentFlags().StringVarP(&conf.TransposeColumns, "transpose-columns", "T", "",
|
rootCmd.PersistentFlags().StringVarP(&conf.TransposeColumns, "transpose-columns", "T", "",
|
||||||
"Transpose the speficied columns (separated by ,)")
|
"Transpose the speficied columns (separated by ,)")
|
||||||
rootCmd.PersistentFlags().BoolVarP(&conf.Interactive, "interactive", "I", false,
|
rootCmd.PersistentFlags().BoolVarP(&conf.Interactive, "interactive", "I", false,
|
||||||
"interactive mode (experimental)")
|
"interactive mode")
|
||||||
|
rootCmd.PersistentFlags().StringVarP(&conf.OFS, "ofs", "o", "",
|
||||||
|
"Output field separator (' ' for ascii table, ',' for CSV)")
|
||||||
|
rootCmd.PersistentFlags().BoolVarP(&conf.InputJSON, "json", "j", false,
|
||||||
|
"JSON input mode")
|
||||||
|
rootCmd.PersistentFlags().BoolVarP(&conf.AutoHeaders, "auto-headers", "g", false,
|
||||||
|
"Generate headers automatically")
|
||||||
|
rootCmd.PersistentFlags().StringVarP(&headers, "custom-headers", "x", "",
|
||||||
|
"Custom headers")
|
||||||
|
|
||||||
// sort options
|
// sort options
|
||||||
rootCmd.PersistentFlags().StringVarP(&conf.SortByColumn, "sort-by", "k", "",
|
rootCmd.PersistentFlags().StringVarP(&conf.SortByColumn, "sort-by", "k", "",
|
||||||
@@ -160,12 +171,16 @@ func Execute() {
|
|||||||
"Enable shell mode output")
|
"Enable shell mode output")
|
||||||
rootCmd.PersistentFlags().BoolVarP(&modeflag.Y, "yaml", "Y", false,
|
rootCmd.PersistentFlags().BoolVarP(&modeflag.Y, "yaml", "Y", false,
|
||||||
"Enable yaml output")
|
"Enable yaml output")
|
||||||
|
rootCmd.PersistentFlags().BoolVarP(&modeflag.J, "jsonout", "J", false,
|
||||||
|
"Enable json output")
|
||||||
rootCmd.PersistentFlags().BoolVarP(&modeflag.C, "csv", "C", false,
|
rootCmd.PersistentFlags().BoolVarP(&modeflag.C, "csv", "C", false,
|
||||||
"Enable CSV output")
|
"Enable CSV output")
|
||||||
rootCmd.PersistentFlags().BoolVarP(&modeflag.A, "ascii", "A", false,
|
rootCmd.PersistentFlags().BoolVarP(&modeflag.A, "ascii", "A", false,
|
||||||
"Enable ASCII output (default)")
|
"Enable ASCII output (default)")
|
||||||
rootCmd.MarkFlagsMutuallyExclusive("extended", "markdown", "orgtbl",
|
rootCmd.MarkFlagsMutuallyExclusive("extended", "markdown", "orgtbl",
|
||||||
"shell", "yaml", "csv")
|
"shell", "yaml", "csv")
|
||||||
|
rootCmd.PersistentFlags().StringVarP(&conf.Template, "template", "P", "",
|
||||||
|
"template for template output mode")
|
||||||
|
|
||||||
// config file
|
// config file
|
||||||
rootCmd.PersistentFlags().StringVarP(&conf.Configfile, "config", "f", cfg.DefaultConfigfile,
|
rootCmd.PersistentFlags().StringVarP(&conf.Configfile, "config", "f", cfg.DefaultConfigfile,
|
||||||
@@ -183,6 +198,11 @@ func Execute() {
|
|||||||
|
|
||||||
rootCmd.SetUsageTemplate(strings.TrimSpace(usage) + "\n")
|
rootCmd.SetUsageTemplate(strings.TrimSpace(usage) + "\n")
|
||||||
|
|
||||||
|
if slices.Contains(os.Args, "-h") {
|
||||||
|
fmt.Println(shortusage)
|
||||||
|
os.Exit(0)
|
||||||
|
}
|
||||||
|
|
||||||
err := rootCmd.Execute()
|
err := rootCmd.Execute()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
|
|||||||
18
cmd/shortusage.go
Normal file
18
cmd/shortusage.go
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
package cmd
|
||||||
|
|
||||||
|
const shortusage = `tablizer [regex,...] [-r file] [flags]
|
||||||
|
-c col,... show specified columns -L highlight matching lines
|
||||||
|
-k col,... sort by specified columns -j read JSON input
|
||||||
|
-F col=reg filter field with regexp -v invert match
|
||||||
|
-T col,... transpose specified columns -n numberize columns
|
||||||
|
-R /from/to/ apply replacement to columns in -T -N do not use colors
|
||||||
|
-y col,... yank columns to clipboard -H do not show headers
|
||||||
|
--ofs char output field separator -s specify field separator
|
||||||
|
-r file read input from file -z use fuzzy search
|
||||||
|
-f file read config from file -I interactive filter mode
|
||||||
|
-x col,... use custom headers -d debug
|
||||||
|
-o char use char as output separator -g auto generate headers
|
||||||
|
|
||||||
|
-O org -C CSV -M md -X ext -S shell -Y yaml -J json -P template
|
||||||
|
-a sort by age -i sort numerically -t sort by time -D sort descending order
|
||||||
|
-m show manual -v show version --help show detailed help`
|
||||||
243
cmd/tablizer.go
243
cmd/tablizer.go
@@ -9,45 +9,51 @@ SYNOPSIS
|
|||||||
tablizer [regex,...] [-r file] [flags]
|
tablizer [regex,...] [-r 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 ,)
|
||||||
-v, --invert-match select non-matching rows
|
-v, --invert-match select non-matching rows
|
||||||
-n, --numbering Enable header numbering
|
-n, --numbering Enable header numbering
|
||||||
-N, --no-color Disable pattern highlighting
|
-N, --no-color Disable pattern highlighting
|
||||||
-H, --no-headers Disable headers display
|
-H, --no-headers Disable headers display
|
||||||
-s, --separator string Custom field separator
|
-s, --separator <string> Custom field separator (maybe char, string or :class:)
|
||||||
-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
|
||||||
-I, --interactive Interactively filter and select rows
|
-j, --json Read JSON input (must be array of hashes)
|
||||||
|
-I, --interactive Interactively filter and select rows
|
||||||
|
-g, --auto-headers Generate headers if there are none present in input
|
||||||
|
-x, --custom-headers a,b,... Use custom headers, separated by comma
|
||||||
|
|
||||||
Output Flags (mutually exclusive):
|
Output Flags (mutually exclusive):
|
||||||
-X, --extended Enable extended output
|
-X, --extended Enable extended output
|
||||||
-M, --markdown Enable markdown table output
|
-M, --markdown Enable markdown table output
|
||||||
-O, --orgtbl Enable org-mode table output
|
-O, --orgtbl Enable org-mode table output
|
||||||
-S, --shell Enable shell evaluable output
|
-S, --shell Enable shell evaluable output
|
||||||
-Y, --yaml Enable yaml output
|
-Y, --yaml Enable yaml output
|
||||||
-C, --csv Enable CSV output
|
-J, --jsonout Enable JSON output
|
||||||
-A, --ascii Default output mode, ascii tabular
|
-C, --csv Enable CSV output
|
||||||
-L, --hightlight-lines Use alternating background colors for tables
|
-A, --ascii Default output mode, ascii tabular
|
||||||
-y, --yank-columns Yank specified columns (separated by ,) to clipboard,
|
-P, --template <tpl> Enable template mode with template <tpl>
|
||||||
space separated
|
-L, --hightlight-lines Use alternating background colors for tables
|
||||||
|
-o, --ofs <char> Output field separator, used by -A and -C.
|
||||||
|
-y, --yank-columns Yank specified columns (separated by ,) to clipboard,
|
||||||
|
space separated
|
||||||
|
|
||||||
Sort Mode Flags (mutually exclusive):
|
Sort Mode Flags (mutually exclusive):
|
||||||
-a, --sort-age sort according to age (duration) string
|
-a, --sort-age sort according to age (duration) string
|
||||||
-D, --sort-desc Sort in descending order (default: ascending)
|
-D, --sort-desc Sort in descending order (default: ascending)
|
||||||
-i, --sort-numeric sort according to string numerical value
|
-i, --sort-numeric sort according to string numerical value
|
||||||
-t, --sort-time sort according to time string
|
-t, --sort-time sort according to time string
|
||||||
|
|
||||||
Other Flags:
|
Other Flags:
|
||||||
-r --read-file <file> Use <file> as input instead of STDIN
|
-r --read-file <file> Use <file> as input instead of STDIN
|
||||||
--completion <shell> Generate the autocompletion script for <shell>
|
--completion <shell> Generate the autocompletion script for <shell>
|
||||||
-f, --config <file> Configuration file (default: ~/.config/tablizer/config)
|
-f, --config <file> Configuration file (default: ~/.config/tablizer/config)
|
||||||
-d, --debug Enable debugging
|
-d, --debug Enable debugging
|
||||||
-h, --help help for tablizer
|
-h, --help help for tablizer
|
||||||
-m, --man Display manual page
|
-m, --man Display manual page
|
||||||
-V, --version Print program version
|
-V, --version Print program version
|
||||||
|
|
||||||
DESCRIPTION
|
DESCRIPTION
|
||||||
Many programs generate tabular output. But sometimes you need to
|
Many programs generate tabular output. But sometimes you need to
|
||||||
@@ -72,31 +78,37 @@ DESCRIPTION
|
|||||||
use the -v option to exclude all rows which match the pattern. Hence:
|
use the -v option to exclude all rows which match the pattern. Hence:
|
||||||
|
|
||||||
# read from STDIN
|
# read from STDIN
|
||||||
kubectl get pods | tablizer
|
> kubectl get pods | tablizer
|
||||||
|
|
||||||
# read a file
|
# read a file
|
||||||
tablizer filename
|
> tablizer -r filename
|
||||||
|
|
||||||
# search for pattern in a file (works like grep)
|
# search for pattern in a file (works like grep)
|
||||||
tablizer regex filename
|
> tablizer regex -r filename
|
||||||
|
|
||||||
# search for pattern in STDIN
|
# search for pattern in STDIN
|
||||||
kubectl get pods | tablizer regex
|
> kubectl get pods | tablizer regex
|
||||||
|
|
||||||
The output looks like the original one but every header field will have
|
The output looks like the original one. You can add the option -n, then
|
||||||
a numer associated with it, e.g.:
|
every header field will have a numer associated with it, e.g.:
|
||||||
|
|
||||||
NAME(1) READY(2) STATUS(3) RESTARTS(4) AGE(5)
|
NAME(1) READY(2) STATUS(3) RESTARTS(4) AGE(5)
|
||||||
|
|
||||||
These numbers denote the column and you can use them to specify which
|
These numbers denote the column and you can use them to specify which
|
||||||
columns you want to have in your output (see COLUMNS:
|
columns you want to have in your output (see COLUMNS:
|
||||||
|
|
||||||
kubectl get pods | tablizer -c1,3
|
> kubectl get pods | tablizer -c1,3
|
||||||
|
|
||||||
You can specify the numbers in any order but output will always follow
|
You can specify the numbers in any order but output will always follow
|
||||||
the original order.
|
the original order.
|
||||||
|
|
||||||
The numbering can be suppressed by using the -n option.
|
However, you may also just use the header names instead of numbers, eg:
|
||||||
|
|
||||||
|
> kubectl get pods | tablizer -cname,status
|
||||||
|
|
||||||
|
You can also use regular expressions with -c, eg:
|
||||||
|
|
||||||
|
> kubectl get pods | tablizer -c '[ae]'
|
||||||
|
|
||||||
By default tablizer shows a header containing the names of each column.
|
By default tablizer shows a header containing the names of each column.
|
||||||
This can be disabled using the -H option. Be aware that this only
|
This can be disabled using the -H option. Be aware that this only
|
||||||
@@ -133,6 +145,57 @@ DESCRIPTION
|
|||||||
Finally the -d option enables debugging output which is mostly useful
|
Finally the -d option enables debugging output which is mostly useful
|
||||||
for the developer.
|
for the developer.
|
||||||
|
|
||||||
|
SEPARATOR
|
||||||
|
The option -s can be a single character, in which case the CSV parser
|
||||||
|
will be invoked. You can also specify a string as separator. The string
|
||||||
|
will be interpreted as literal string unless it is a valid go regular
|
||||||
|
expression. For example:
|
||||||
|
|
||||||
|
-s '\t{2,}\'
|
||||||
|
|
||||||
|
is being used as a regexp and will match two or more consecutive tabs.
|
||||||
|
|
||||||
|
-s 'foo'
|
||||||
|
|
||||||
|
on the other hand is no regular expression and will be used literally.
|
||||||
|
|
||||||
|
To make live easier, there are a couple of predefined regular
|
||||||
|
expressions, which you can specify as classes:
|
||||||
|
|
||||||
|
* :tab:
|
||||||
|
|
||||||
|
Matches a tab and eats spaces around it.
|
||||||
|
|
||||||
|
* :spaces:
|
||||||
|
|
||||||
|
Matches 2 or more spaces.
|
||||||
|
|
||||||
|
* :pipe:
|
||||||
|
|
||||||
|
Matches a pipe character and eats spaces around it.
|
||||||
|
|
||||||
|
* :default:
|
||||||
|
|
||||||
|
Matches 2 or more spaces or tab. This is the default separator if
|
||||||
|
none is specified.
|
||||||
|
|
||||||
|
* :nonword:
|
||||||
|
|
||||||
|
Matches a non-word character.
|
||||||
|
|
||||||
|
* :nondigit:
|
||||||
|
|
||||||
|
Matches a non-digit character.
|
||||||
|
|
||||||
|
* :special:
|
||||||
|
|
||||||
|
Matches one or more special chars like brackets, dollar sign,
|
||||||
|
slashes etc.
|
||||||
|
|
||||||
|
* :nonprint:
|
||||||
|
|
||||||
|
Matches one or more non-printable characters.
|
||||||
|
|
||||||
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 one or more regular
|
||||||
expression patterns. The regexp language being used is the one of
|
expression patterns. The regexp language being used is the one of
|
||||||
@@ -156,7 +219,7 @@ DESCRIPTION
|
|||||||
|
|
||||||
Example for a case insensitive search:
|
Example for a case insensitive search:
|
||||||
|
|
||||||
kubectl get pods -A | tablizer "/account/i"
|
> kubectl get pods -A | tablizer "/account/i"
|
||||||
|
|
||||||
If you use the "!" flag, then the regex match will be negated, that is,
|
If you use the "!" flag, then the regex match will be negated, that is,
|
||||||
if a line in the input matches the given regex, but "!" is supplied,
|
if a line in the input matches the given regex, but "!" is supplied,
|
||||||
@@ -222,7 +285,7 @@ DESCRIPTION
|
|||||||
|
|
||||||
We want to see only the CMD column and use a regex for this:
|
We want to see only the CMD column and use a regex for this:
|
||||||
|
|
||||||
ps | tablizer -s '\s+' -c C
|
> ps | tablizer -s '\s+' -c C
|
||||||
CMD(4)
|
CMD(4)
|
||||||
bash
|
bash
|
||||||
ps
|
ps
|
||||||
@@ -253,7 +316,7 @@ DESCRIPTION
|
|||||||
|
|
||||||
Example:
|
Example:
|
||||||
|
|
||||||
cat t/testtable2
|
> cat t/testtable2
|
||||||
NAME DURATION
|
NAME DURATION
|
||||||
x 10
|
x 10
|
||||||
a 100
|
a 100
|
||||||
@@ -261,7 +324,7 @@ DESCRIPTION
|
|||||||
u 4
|
u 4
|
||||||
k 6
|
k 6
|
||||||
|
|
||||||
cat t/testtable2 | tablizer -T2 -R '/^\d/4/' -n
|
> cat t/testtable2 | tablizer -T2 -R '/^\d/4/' -n
|
||||||
NAME DURATION
|
NAME DURATION
|
||||||
x 40
|
x 40
|
||||||
a 400
|
a 400
|
||||||
@@ -277,7 +340,7 @@ DESCRIPTION
|
|||||||
header left, value right, aligned by the field widths. Here's an
|
header left, value right, aligned by the field widths. Here's an
|
||||||
example:
|
example:
|
||||||
|
|
||||||
kubectl get pods | ./tablizer -o extended
|
> kubectl get pods | ./tablizer -o extended
|
||||||
NAME: repldepl-7bcd8d5b64-7zq4l
|
NAME: repldepl-7bcd8d5b64-7zq4l
|
||||||
READY: 1/1
|
READY: 1/1
|
||||||
STATUS: Running
|
STATUS: Running
|
||||||
@@ -290,7 +353,7 @@ DESCRIPTION
|
|||||||
The option -o shell can be used if the output has to be processed by the
|
The option -o shell can be used if the output has to be processed by the
|
||||||
shell, it prints variable assignments for each cell, one line per row:
|
shell, it prints variable assignments for each cell, one line per row:
|
||||||
|
|
||||||
kubectl get pods | ./tablizer -o extended ./tablizer -o shell
|
> kubectl get pods | ./tablizer -o extended ./tablizer -o shell
|
||||||
NAME="repldepl-7bcd8d5b64-7zq4l" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
NAME="repldepl-7bcd8d5b64-7zq4l" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
||||||
NAME="repldepl-7bcd8d5b64-m48n8" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
NAME="repldepl-7bcd8d5b64-m48n8" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
||||||
NAME="repldepl-7bcd8d5b64-q2bf4" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
NAME="repldepl-7bcd8d5b64-q2bf4" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
||||||
@@ -302,6 +365,18 @@ DESCRIPTION
|
|||||||
markdown which prints a Markdown table, yaml, which prints yaml encoding
|
markdown which prints a Markdown table, yaml, which prints yaml encoding
|
||||||
and CSV mode, which prints a comma separated value file.
|
and CSV mode, which prints a comma separated value file.
|
||||||
|
|
||||||
|
A special output mode ist the Template mode, activated with the option
|
||||||
|
"--template". Template language is the Golang template language:
|
||||||
|
<https://pkg.go.dev/text/template>. You can also use lot's of additional
|
||||||
|
functions from: <https://masterminds.github.io/sprig/>. Here's an
|
||||||
|
example:
|
||||||
|
|
||||||
|
> kubectl get pods | tablizer --template "{{.name}} is {{.status}}"
|
||||||
|
alertmanager-kube-prometheus-alertmanager-0 is Running
|
||||||
|
grafana-fcc54cbc9-bk7s8 is Running
|
||||||
|
|
||||||
|
You can use header names as variables.
|
||||||
|
|
||||||
PUT FIELDS TO CLIPBOARD
|
PUT FIELDS TO CLIPBOARD
|
||||||
You can let tablizer put fields to the clipboard using the option "-y".
|
You can let tablizer put fields to the clipboard using the option "-y".
|
||||||
This best fits the use-case when the result of your filtering yields
|
This best fits the use-case when the result of your filtering yields
|
||||||
@@ -402,7 +477,7 @@ CONFIGURATION AND COLORS
|
|||||||
BUGS
|
BUGS
|
||||||
In order to report a bug, unexpected behavior, feature requests or to
|
In order to report a bug, unexpected behavior, feature requests or to
|
||||||
submit a patch, please open an issue on github:
|
submit a patch, please open an issue on github:
|
||||||
<https://github.com/TLINDEN/tablizer/issues>.
|
<https://codeberg.org/scip/tablizer/issues>.
|
||||||
|
|
||||||
LICENSE
|
LICENSE
|
||||||
This software is licensed under the GNU GENERAL PUBLIC LICENSE version
|
This software is licensed under the GNU GENERAL PUBLIC LICENSE version
|
||||||
@@ -445,45 +520,51 @@ Usage:
|
|||||||
tablizer [regex,...] [-r file] [flags]
|
tablizer [regex,...] [-r 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 ,)
|
||||||
-v, --invert-match select non-matching rows
|
-v, --invert-match select non-matching rows
|
||||||
-n, --numbering Enable header numbering
|
-n, --numbering Enable header numbering
|
||||||
-N, --no-color Disable pattern highlighting
|
-N, --no-color Disable pattern highlighting
|
||||||
-H, --no-headers Disable headers display
|
-H, --no-headers Disable headers display
|
||||||
-s, --separator string Custom field separator
|
-s, --separator <string> Custom field separator (maybe char, string or :class:)
|
||||||
-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
|
||||||
-I, --interactive Interactively filter and select rows
|
-j, --json Read JSON input (must be array of hashes)
|
||||||
|
-I, --interactive Interactively filter and select rows
|
||||||
|
-g, --auto-headers Generate headers if there are none present in input
|
||||||
|
-x, --custom-headers a,b,... Use custom headers, separated by comma
|
||||||
|
|
||||||
Output Flags (mutually exclusive):
|
Output Flags (mutually exclusive):
|
||||||
-X, --extended Enable extended output
|
-X, --extended Enable extended output
|
||||||
-M, --markdown Enable markdown table output
|
-M, --markdown Enable markdown table output
|
||||||
-O, --orgtbl Enable org-mode table output
|
-O, --orgtbl Enable org-mode table output
|
||||||
-S, --shell Enable shell evaluable output
|
-S, --shell Enable shell evaluable output
|
||||||
-Y, --yaml Enable yaml output
|
-Y, --yaml Enable yaml output
|
||||||
-C, --csv Enable CSV output
|
-J, --jsonout Enable JSON output
|
||||||
-A, --ascii Default output mode, ascii tabular
|
-C, --csv Enable CSV output
|
||||||
-L, --hightlight-lines Use alternating background colors for tables
|
-A, --ascii Default output mode, ascii tabular
|
||||||
-y, --yank-columns Yank specified columns (separated by ,) to clipboard,
|
-P, --template <tpl> Enable template mode with template <tpl>
|
||||||
space separated
|
-L, --hightlight-lines Use alternating background colors for tables
|
||||||
|
-o, --ofs <char> Output field separator, used by -A and -C.
|
||||||
|
-y, --yank-columns Yank specified columns (separated by ,) to clipboard,
|
||||||
|
space separated
|
||||||
|
|
||||||
Sort Mode Flags (mutually exclusive):
|
Sort Mode Flags (mutually exclusive):
|
||||||
-a, --sort-age sort according to age (duration) string
|
-a, --sort-age sort according to age (duration) string
|
||||||
-D, --sort-desc Sort in descending order (default: ascending)
|
-D, --sort-desc Sort in descending order (default: ascending)
|
||||||
-i, --sort-numeric sort according to string numerical value
|
-i, --sort-numeric sort according to string numerical value
|
||||||
-t, --sort-time sort according to time string
|
-t, --sort-time sort according to time string
|
||||||
|
|
||||||
Other Flags:
|
Other Flags:
|
||||||
-r --read-file <file> Use <file> as input instead of STDIN
|
-r --read-file <file> Use <file> as input instead of STDIN
|
||||||
--completion <shell> Generate the autocompletion script for <shell>
|
--completion <shell> Generate the autocompletion script for <shell>
|
||||||
-f, --config <file> Configuration file (default: ~/.config/tablizer/config)
|
-f, --config <file> Configuration file (default: ~/.config/tablizer/config)
|
||||||
-d, --debug Enable debugging
|
-d, --debug Enable debugging
|
||||||
-h, --help help for tablizer
|
-h, --help help for tablizer
|
||||||
-m, --man Display manual page
|
-m, --man Display manual page
|
||||||
-V, --version Print program version
|
-V, --version Print program version
|
||||||
|
|
||||||
|
|
||||||
`
|
`
|
||||||
|
|||||||
38
go.mod
38
go.mod
@@ -1,45 +1,53 @@
|
|||||||
module github.com/tlinden/tablizer
|
module codeberg.org/scip/tablizer
|
||||||
|
|
||||||
go 1.23.0
|
go 1.24.0
|
||||||
|
|
||||||
toolchain go1.23.5
|
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/alecthomas/repr v0.5.1
|
github.com/alecthomas/repr v0.5.2
|
||||||
github.com/araddon/dateparse v0.0.0-20210429162001-6b43995a97de
|
github.com/araddon/dateparse v0.0.0-20210429162001-6b43995a97de
|
||||||
github.com/charmbracelet/bubbles v0.21.0
|
github.com/charmbracelet/bubbles v0.21.0
|
||||||
github.com/charmbracelet/bubbletea v1.3.6
|
github.com/charmbracelet/bubbletea v1.3.10
|
||||||
github.com/charmbracelet/lipgloss v1.1.0
|
github.com/charmbracelet/lipgloss v1.1.0
|
||||||
github.com/evertras/bubble-table v0.19.0
|
github.com/evertras/bubble-table v0.19.2
|
||||||
github.com/gookit/color v1.6.0
|
github.com/gookit/color v1.6.0
|
||||||
github.com/hashicorp/hcl/v2 v2.24.0
|
github.com/hashicorp/hcl/v2 v2.24.0
|
||||||
github.com/lithammer/fuzzysearch v1.1.8
|
github.com/lithammer/fuzzysearch v1.1.8
|
||||||
github.com/olekukonko/tablewriter v1.0.9
|
github.com/mattn/go-isatty v0.0.20
|
||||||
|
github.com/olekukonko/tablewriter v1.1.0
|
||||||
github.com/rogpeppe/go-internal v1.14.1
|
github.com/rogpeppe/go-internal v1.14.1
|
||||||
github.com/spf13/cobra v1.9.1
|
github.com/spf13/cobra v1.10.1
|
||||||
|
github.com/stretchr/testify v1.11.1
|
||||||
github.com/tiagomelo/go-clipboard v0.1.2
|
github.com/tiagomelo/go-clipboard v0.1.2
|
||||||
gopkg.in/yaml.v3 v3.0.1
|
gopkg.in/yaml.v3 v3.0.1
|
||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
|
dario.cat/mergo v1.0.1 // indirect
|
||||||
|
github.com/Masterminds/goutils v1.1.1 // indirect
|
||||||
|
github.com/Masterminds/semver/v3 v3.3.0 // indirect
|
||||||
|
github.com/Masterminds/sprig/v3 v3.3.0 // indirect
|
||||||
github.com/agext/levenshtein v1.2.3 // indirect
|
github.com/agext/levenshtein v1.2.3 // indirect
|
||||||
github.com/apparentlymart/go-textseg/v15 v15.0.0 // indirect
|
github.com/apparentlymart/go-textseg/v15 v15.0.0 // indirect
|
||||||
github.com/atotto/clipboard v0.1.4 // indirect
|
github.com/atotto/clipboard v0.1.4 // indirect
|
||||||
github.com/aymanbagabas/go-osc52/v2 v2.0.1 // indirect
|
github.com/aymanbagabas/go-osc52/v2 v2.0.1 // indirect
|
||||||
github.com/charmbracelet/colorprofile v0.3.1 // indirect
|
github.com/charmbracelet/colorprofile v0.3.1 // indirect
|
||||||
github.com/charmbracelet/x/ansi v0.9.3 // indirect
|
github.com/charmbracelet/x/ansi v0.10.1 // indirect
|
||||||
github.com/charmbracelet/x/cellbuf v0.0.13 // indirect
|
github.com/charmbracelet/x/cellbuf v0.0.13 // indirect
|
||||||
github.com/charmbracelet/x/term v0.2.1 // indirect
|
github.com/charmbracelet/x/term v0.2.1 // indirect
|
||||||
|
github.com/davecgh/go-spew v1.1.1 // indirect
|
||||||
github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f // indirect
|
github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f // indirect
|
||||||
github.com/fatih/color v1.18.0 // indirect
|
github.com/fatih/color v1.18.0 // indirect
|
||||||
github.com/google/go-cmp v0.6.0 // indirect
|
github.com/google/go-cmp v0.6.0 // indirect
|
||||||
|
github.com/google/uuid v1.6.0 // indirect
|
||||||
|
github.com/huandu/xstrings v1.5.0 // indirect
|
||||||
github.com/inconshreveable/mousetrap v1.1.0 // indirect
|
github.com/inconshreveable/mousetrap v1.1.0 // indirect
|
||||||
github.com/lucasb-eyer/go-colorful v1.2.0 // indirect
|
github.com/lucasb-eyer/go-colorful v1.2.0 // indirect
|
||||||
github.com/mattn/go-colorable v0.1.14 // indirect
|
github.com/mattn/go-colorable v0.1.14 // indirect
|
||||||
github.com/mattn/go-isatty v0.0.20 // indirect
|
|
||||||
github.com/mattn/go-localereader v0.0.1 // indirect
|
github.com/mattn/go-localereader v0.0.1 // indirect
|
||||||
github.com/mattn/go-runewidth v0.0.16 // indirect
|
github.com/mattn/go-runewidth v0.0.16 // indirect
|
||||||
|
github.com/mitchellh/copystructure v1.2.0 // indirect
|
||||||
github.com/mitchellh/go-wordwrap v1.0.1 // indirect
|
github.com/mitchellh/go-wordwrap v1.0.1 // indirect
|
||||||
|
github.com/mitchellh/reflectwalk v1.0.2 // indirect
|
||||||
github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 // indirect
|
github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 // indirect
|
||||||
github.com/muesli/cancelreader v0.2.2 // indirect
|
github.com/muesli/cancelreader v0.2.2 // indirect
|
||||||
github.com/muesli/reflow v0.3.0 // indirect
|
github.com/muesli/reflow v0.3.0 // indirect
|
||||||
@@ -47,14 +55,18 @@ require (
|
|||||||
github.com/olekukonko/errors v1.1.0 // indirect
|
github.com/olekukonko/errors v1.1.0 // indirect
|
||||||
github.com/olekukonko/ll v0.0.9 // indirect
|
github.com/olekukonko/ll v0.0.9 // indirect
|
||||||
github.com/pkg/errors v0.9.1 // indirect
|
github.com/pkg/errors v0.9.1 // indirect
|
||||||
|
github.com/pmezard/go-difflib v1.0.0 // indirect
|
||||||
github.com/rivo/uniseg v0.4.7 // indirect
|
github.com/rivo/uniseg v0.4.7 // indirect
|
||||||
github.com/spf13/pflag v1.0.6 // indirect
|
github.com/shopspring/decimal v1.4.0 // indirect
|
||||||
|
github.com/spf13/cast v1.7.0 // indirect
|
||||||
|
github.com/spf13/pflag v1.0.9 // indirect
|
||||||
github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e // indirect
|
github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e // indirect
|
||||||
github.com/zclconf/go-cty v1.16.3 // indirect
|
github.com/zclconf/go-cty v1.16.3 // indirect
|
||||||
|
golang.org/x/crypto v0.38.0 // indirect
|
||||||
golang.org/x/exp v0.0.0-20231006140011-7918f672742d // indirect
|
golang.org/x/exp v0.0.0-20231006140011-7918f672742d // indirect
|
||||||
golang.org/x/mod v0.21.0 // indirect
|
golang.org/x/mod v0.21.0 // indirect
|
||||||
golang.org/x/sync v0.15.0 // indirect
|
golang.org/x/sync v0.15.0 // indirect
|
||||||
golang.org/x/sys v0.33.0 // indirect
|
golang.org/x/sys v0.36.0 // indirect
|
||||||
golang.org/x/text v0.25.0 // indirect
|
golang.org/x/text v0.25.0 // indirect
|
||||||
golang.org/x/tools v0.26.0 // indirect
|
golang.org/x/tools v0.26.0 // indirect
|
||||||
)
|
)
|
||||||
|
|||||||
58
go.sum
58
go.sum
@@ -1,7 +1,15 @@
|
|||||||
|
dario.cat/mergo v1.0.1 h1:Ra4+bf83h2ztPIQYNP99R6m+Y7KfnARDfID+a+vLl4s=
|
||||||
|
dario.cat/mergo v1.0.1/go.mod h1:uNxQE+84aUszobStD9th8a29P2fMDhsBdgRYvZOxGmk=
|
||||||
|
github.com/Masterminds/goutils v1.1.1 h1:5nUrii3FMTL5diU80unEVvNevw1nH4+ZV4DSLVJLSYI=
|
||||||
|
github.com/Masterminds/goutils v1.1.1/go.mod h1:8cTjp+g8YejhMuvIA5y2vz3BpJxksy863GQaJW2MFNU=
|
||||||
|
github.com/Masterminds/semver/v3 v3.3.0 h1:B8LGeaivUe71a5qox1ICM/JLl0NqZSW5CHyL+hmvYS0=
|
||||||
|
github.com/Masterminds/semver/v3 v3.3.0/go.mod h1:4V+yj/TJE1HU9XfppCwVMZq3I84lprf4nC11bSS5beM=
|
||||||
|
github.com/Masterminds/sprig/v3 v3.3.0 h1:mQh0Yrg1XPo6vjYXgtf5OtijNAKJRNcTdOOGZe3tPhs=
|
||||||
|
github.com/Masterminds/sprig/v3 v3.3.0/go.mod h1:Zy1iXRYNqNLUolqCpL4uhk6SHUMAOSCzdgBfDb35Lz0=
|
||||||
github.com/agext/levenshtein v1.2.3 h1:YB2fHEn0UJagG8T1rrWknE3ZQzWM06O8AMAatNn7lmo=
|
github.com/agext/levenshtein v1.2.3 h1:YB2fHEn0UJagG8T1rrWknE3ZQzWM06O8AMAatNn7lmo=
|
||||||
github.com/agext/levenshtein v1.2.3/go.mod h1:JEDfjyjHDjOF/1e4FlBE/PkbqA9OfWu2ki2W0IB5558=
|
github.com/agext/levenshtein v1.2.3/go.mod h1:JEDfjyjHDjOF/1e4FlBE/PkbqA9OfWu2ki2W0IB5558=
|
||||||
github.com/alecthomas/repr v0.5.1 h1:E3G4t2QbHTSNpPKBgMTln5KLkZHLOcU7r37J4pXBuIg=
|
github.com/alecthomas/repr v0.5.2 h1:SU73FTI9D1P5UNtvseffFSGmdNci/O6RsqzeXJtP0Qs=
|
||||||
github.com/alecthomas/repr v0.5.1/go.mod h1:Fr0507jx4eOXV7AlPV6AVZLYrLIuIeSOWtW57eE/O/4=
|
github.com/alecthomas/repr v0.5.2/go.mod h1:Fr0507jx4eOXV7AlPV6AVZLYrLIuIeSOWtW57eE/O/4=
|
||||||
github.com/apparentlymart/go-textseg/v15 v15.0.0 h1:uYvfpb3DyLSCGWnctWKGj857c6ew1u1fNQOlOtuGxQY=
|
github.com/apparentlymart/go-textseg/v15 v15.0.0 h1:uYvfpb3DyLSCGWnctWKGj857c6ew1u1fNQOlOtuGxQY=
|
||||||
github.com/apparentlymart/go-textseg/v15 v15.0.0/go.mod h1:K8XmNZdhEBkdlyDdvbmmsvpAG721bKi0joRfFdHIWJ4=
|
github.com/apparentlymart/go-textseg/v15 v15.0.0/go.mod h1:K8XmNZdhEBkdlyDdvbmmsvpAG721bKi0joRfFdHIWJ4=
|
||||||
github.com/araddon/dateparse v0.0.0-20210429162001-6b43995a97de h1:FxWPpzIjnTlhPwqqXc4/vE0f7GvRjuAsbW+HOIe8KnA=
|
github.com/araddon/dateparse v0.0.0-20210429162001-6b43995a97de h1:FxWPpzIjnTlhPwqqXc4/vE0f7GvRjuAsbW+HOIe8KnA=
|
||||||
@@ -12,14 +20,14 @@ github.com/aymanbagabas/go-osc52/v2 v2.0.1 h1:HwpRHbFMcZLEVr42D4p7XBqjyuxQH5SMiE
|
|||||||
github.com/aymanbagabas/go-osc52/v2 v2.0.1/go.mod h1:uYgXzlJ7ZpABp8OJ+exZzJJhRNQ2ASbcXHWsFqH8hp8=
|
github.com/aymanbagabas/go-osc52/v2 v2.0.1/go.mod h1:uYgXzlJ7ZpABp8OJ+exZzJJhRNQ2ASbcXHWsFqH8hp8=
|
||||||
github.com/charmbracelet/bubbles v0.21.0 h1:9TdC97SdRVg/1aaXNVWfFH3nnLAwOXr8Fn6u6mfQdFs=
|
github.com/charmbracelet/bubbles v0.21.0 h1:9TdC97SdRVg/1aaXNVWfFH3nnLAwOXr8Fn6u6mfQdFs=
|
||||||
github.com/charmbracelet/bubbles v0.21.0/go.mod h1:HF+v6QUR4HkEpz62dx7ym2xc71/KBHg+zKwJtMw+qtg=
|
github.com/charmbracelet/bubbles v0.21.0/go.mod h1:HF+v6QUR4HkEpz62dx7ym2xc71/KBHg+zKwJtMw+qtg=
|
||||||
github.com/charmbracelet/bubbletea v1.3.6 h1:VkHIxPJQeDt0aFJIsVxw8BQdh/F/L2KKZGsK6et5taU=
|
github.com/charmbracelet/bubbletea v1.3.10 h1:otUDHWMMzQSB0Pkc87rm691KZ3SWa4KUlvF9nRvCICw=
|
||||||
github.com/charmbracelet/bubbletea v1.3.6/go.mod h1:oQD9VCRQFF8KplacJLo28/jofOI2ToOfGYeFgBBxHOc=
|
github.com/charmbracelet/bubbletea v1.3.10/go.mod h1:ORQfo0fk8U+po9VaNvnV95UPWA1BitP1E0N6xJPlHr4=
|
||||||
github.com/charmbracelet/colorprofile v0.3.1 h1:k8dTHMd7fgw4bnFd7jXTLZrSU/CQrKnL3m+AxCzDz40=
|
github.com/charmbracelet/colorprofile v0.3.1 h1:k8dTHMd7fgw4bnFd7jXTLZrSU/CQrKnL3m+AxCzDz40=
|
||||||
github.com/charmbracelet/colorprofile v0.3.1/go.mod h1:/GkGusxNs8VB/RSOh3fu0TJmQ4ICMMPApIIVn0KszZ0=
|
github.com/charmbracelet/colorprofile v0.3.1/go.mod h1:/GkGusxNs8VB/RSOh3fu0TJmQ4ICMMPApIIVn0KszZ0=
|
||||||
github.com/charmbracelet/lipgloss v1.1.0 h1:vYXsiLHVkK7fp74RkV7b2kq9+zDLoEU4MZoFqR/noCY=
|
github.com/charmbracelet/lipgloss v1.1.0 h1:vYXsiLHVkK7fp74RkV7b2kq9+zDLoEU4MZoFqR/noCY=
|
||||||
github.com/charmbracelet/lipgloss v1.1.0/go.mod h1:/6Q8FR2o+kj8rz4Dq0zQc3vYf7X+B0binUUBwA0aL30=
|
github.com/charmbracelet/lipgloss v1.1.0/go.mod h1:/6Q8FR2o+kj8rz4Dq0zQc3vYf7X+B0binUUBwA0aL30=
|
||||||
github.com/charmbracelet/x/ansi v0.9.3 h1:BXt5DHS/MKF+LjuK4huWrC6NCvHtexww7dMayh6GXd0=
|
github.com/charmbracelet/x/ansi v0.10.1 h1:rL3Koar5XvX0pHGfovN03f5cxLbCF2YvLeyz7D2jVDQ=
|
||||||
github.com/charmbracelet/x/ansi v0.9.3/go.mod h1:3RQDQ6lDnROptfpWuUVIUG64bD2g2BgntdxH0Ya5TeE=
|
github.com/charmbracelet/x/ansi v0.10.1/go.mod h1:3RQDQ6lDnROptfpWuUVIUG64bD2g2BgntdxH0Ya5TeE=
|
||||||
github.com/charmbracelet/x/cellbuf v0.0.13 h1:/KBBKHuVRbq1lYx5BzEHBAFBP8VcQzJejZ/IA3iR28k=
|
github.com/charmbracelet/x/cellbuf v0.0.13 h1:/KBBKHuVRbq1lYx5BzEHBAFBP8VcQzJejZ/IA3iR28k=
|
||||||
github.com/charmbracelet/x/cellbuf v0.0.13/go.mod h1:xe0nKWGd3eJgtqZRaN9RjMtK7xUYchjzPr7q6kcvCCs=
|
github.com/charmbracelet/x/cellbuf v0.0.13/go.mod h1:xe0nKWGd3eJgtqZRaN9RjMtK7xUYchjzPr7q6kcvCCs=
|
||||||
github.com/charmbracelet/x/term v0.2.1 h1:AQeHeLZ1OqSXhrAWpYUtZyX1T3zVxfpZuEQMIQaGIAQ=
|
github.com/charmbracelet/x/term v0.2.1 h1:AQeHeLZ1OqSXhrAWpYUtZyX1T3zVxfpZuEQMIQaGIAQ=
|
||||||
@@ -30,20 +38,24 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c
|
|||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f h1:Y/CXytFA4m6baUTXGLOoWe4PQhGxaX0KpnayAqC48p4=
|
github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f h1:Y/CXytFA4m6baUTXGLOoWe4PQhGxaX0KpnayAqC48p4=
|
||||||
github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f/go.mod h1:vw97MGsxSvLiUE2X8qFplwetxpGLQrlU1Q9AUEIzCaM=
|
github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f/go.mod h1:vw97MGsxSvLiUE2X8qFplwetxpGLQrlU1Q9AUEIzCaM=
|
||||||
github.com/evertras/bubble-table v0.19.0 h1:+JlXRUjNuBN1JI7XU1PapmW1wglbcqZUKkiPnVKPgrc=
|
github.com/evertras/bubble-table v0.19.2 h1:u77oiM6JlRR+CvS5FZc3Hz+J6iEsvEDcR5kO8OFb1Yw=
|
||||||
github.com/evertras/bubble-table v0.19.0/go.mod h1:ifHujS1YxwnYSOgcR2+m3GnJ84f7CVU/4kUOxUCjEbQ=
|
github.com/evertras/bubble-table v0.19.2/go.mod h1:ifHujS1YxwnYSOgcR2+m3GnJ84f7CVU/4kUOxUCjEbQ=
|
||||||
github.com/fatih/color v1.18.0 h1:S8gINlzdQ840/4pfAwic/ZE0djQEH3wM94VfqLTZcOM=
|
github.com/fatih/color v1.18.0 h1:S8gINlzdQ840/4pfAwic/ZE0djQEH3wM94VfqLTZcOM=
|
||||||
github.com/fatih/color v1.18.0/go.mod h1:4FelSpRwEGDpQ12mAdzqdOukCy4u8WUtOY6lkT/6HfU=
|
github.com/fatih/color v1.18.0/go.mod h1:4FelSpRwEGDpQ12mAdzqdOukCy4u8WUtOY6lkT/6HfU=
|
||||||
github.com/go-test/deep v1.0.3 h1:ZrJSEWsXzPOxaZnFteGEfooLba+ju3FYIbOrS+rQd68=
|
github.com/go-test/deep v1.0.3 h1:ZrJSEWsXzPOxaZnFteGEfooLba+ju3FYIbOrS+rQd68=
|
||||||
github.com/go-test/deep v1.0.3/go.mod h1:wGDj63lr65AM2AQyKZd/NYHGb0R+1RLqB8NKt3aSFNA=
|
github.com/go-test/deep v1.0.3/go.mod h1:wGDj63lr65AM2AQyKZd/NYHGb0R+1RLqB8NKt3aSFNA=
|
||||||
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
|
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
|
||||||
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
|
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
|
||||||
|
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
|
||||||
|
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
||||||
github.com/gookit/assert v0.1.1 h1:lh3GcawXe/p+cU7ESTZ5Ui3Sm/x8JWpIis4/1aF0mY0=
|
github.com/gookit/assert v0.1.1 h1:lh3GcawXe/p+cU7ESTZ5Ui3Sm/x8JWpIis4/1aF0mY0=
|
||||||
github.com/gookit/assert v0.1.1/go.mod h1:jS5bmIVQZTIwk42uXl4lyj4iaaxx32tqH16CFj0VX2E=
|
github.com/gookit/assert v0.1.1/go.mod h1:jS5bmIVQZTIwk42uXl4lyj4iaaxx32tqH16CFj0VX2E=
|
||||||
github.com/gookit/color v1.6.0 h1:JjJXBTk1ETNyqyilJhkTXJYYigHG24TM9Xa2M1xAhRA=
|
github.com/gookit/color v1.6.0 h1:JjJXBTk1ETNyqyilJhkTXJYYigHG24TM9Xa2M1xAhRA=
|
||||||
github.com/gookit/color v1.6.0/go.mod h1:9ACFc7/1IpHGBW8RwuDm/0YEnhg3dwwXpoMsmtyHfjs=
|
github.com/gookit/color v1.6.0/go.mod h1:9ACFc7/1IpHGBW8RwuDm/0YEnhg3dwwXpoMsmtyHfjs=
|
||||||
github.com/hashicorp/hcl/v2 v2.24.0 h1:2QJdZ454DSsYGoaE6QheQZjtKZSUs9Nh2izTWiwQxvE=
|
github.com/hashicorp/hcl/v2 v2.24.0 h1:2QJdZ454DSsYGoaE6QheQZjtKZSUs9Nh2izTWiwQxvE=
|
||||||
github.com/hashicorp/hcl/v2 v2.24.0/go.mod h1:oGoO1FIQYfn/AgyOhlg9qLC6/nOJPX3qGbkZpYAcqfM=
|
github.com/hashicorp/hcl/v2 v2.24.0/go.mod h1:oGoO1FIQYfn/AgyOhlg9qLC6/nOJPX3qGbkZpYAcqfM=
|
||||||
|
github.com/huandu/xstrings v1.5.0 h1:2ag3IFq9ZDANvthTwTiqSSZLjDc+BedvHPAp5tJy2TI=
|
||||||
|
github.com/huandu/xstrings v1.5.0/go.mod h1:y5/lhBue+AyNmUVz9RLU9xbLR0o4KIIExikq4ovT0aE=
|
||||||
github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8=
|
github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8=
|
||||||
github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw=
|
github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw=
|
||||||
github.com/lithammer/fuzzysearch v1.1.8 h1:/HIuJnjHuXS8bKaiTMeeDlW2/AyIWk2brx1V8LFgLN4=
|
github.com/lithammer/fuzzysearch v1.1.8 h1:/HIuJnjHuXS8bKaiTMeeDlW2/AyIWk2brx1V8LFgLN4=
|
||||||
@@ -60,8 +72,12 @@ github.com/mattn/go-runewidth v0.0.10/go.mod h1:RAqKPSqVFrSLVXbA8x7dzmKdmGzieGRC
|
|||||||
github.com/mattn/go-runewidth v0.0.12/go.mod h1:RAqKPSqVFrSLVXbA8x7dzmKdmGzieGRCM46jaSJTDAk=
|
github.com/mattn/go-runewidth v0.0.12/go.mod h1:RAqKPSqVFrSLVXbA8x7dzmKdmGzieGRCM46jaSJTDAk=
|
||||||
github.com/mattn/go-runewidth v0.0.16 h1:E5ScNMtiwvlvB5paMFdw9p4kSQzbXFikJ5SQO6TULQc=
|
github.com/mattn/go-runewidth v0.0.16 h1:E5ScNMtiwvlvB5paMFdw9p4kSQzbXFikJ5SQO6TULQc=
|
||||||
github.com/mattn/go-runewidth v0.0.16/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w=
|
github.com/mattn/go-runewidth v0.0.16/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w=
|
||||||
|
github.com/mitchellh/copystructure v1.2.0 h1:vpKXTN4ewci03Vljg/q9QvCGUDttBOGBIa15WveJJGw=
|
||||||
|
github.com/mitchellh/copystructure v1.2.0/go.mod h1:qLl+cE2AmVv+CoeAwDPye/v+N2HKCj9FbZEVFJRxO9s=
|
||||||
github.com/mitchellh/go-wordwrap v1.0.1 h1:TLuKupo69TCn6TQSyGxwI1EblZZEsQ0vMlAFQflz0v0=
|
github.com/mitchellh/go-wordwrap v1.0.1 h1:TLuKupo69TCn6TQSyGxwI1EblZZEsQ0vMlAFQflz0v0=
|
||||||
github.com/mitchellh/go-wordwrap v1.0.1/go.mod h1:R62XHJLzvMFRBbcrT7m7WgmE1eOyTSsCt+hzestvNj0=
|
github.com/mitchellh/go-wordwrap v1.0.1/go.mod h1:R62XHJLzvMFRBbcrT7m7WgmE1eOyTSsCt+hzestvNj0=
|
||||||
|
github.com/mitchellh/reflectwalk v1.0.2 h1:G2LzWKi524PWgd3mLHV8Y5k7s6XUvT0Gef6zxSIeXaQ=
|
||||||
|
github.com/mitchellh/reflectwalk v1.0.2/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw=
|
||||||
github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 h1:ZK8zHtRHOkbHy6Mmr5D264iyp3TiX5OmNcI5cIARiQI=
|
github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 h1:ZK8zHtRHOkbHy6Mmr5D264iyp3TiX5OmNcI5cIARiQI=
|
||||||
github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6/go.mod h1:CJlz5H+gyd6CUWT45Oy4q24RdLyn7Md9Vj2/ldJBSIo=
|
github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6/go.mod h1:CJlz5H+gyd6CUWT45Oy4q24RdLyn7Md9Vj2/ldJBSIo=
|
||||||
github.com/muesli/cancelreader v0.2.2 h1:3I4Kt4BQjOR54NavqnDogx/MIoWBFa0StPA8ELUXHmA=
|
github.com/muesli/cancelreader v0.2.2 h1:3I4Kt4BQjOR54NavqnDogx/MIoWBFa0StPA8ELUXHmA=
|
||||||
@@ -74,8 +90,8 @@ github.com/olekukonko/errors v1.1.0 h1:RNuGIh15QdDenh+hNvKrJkmxxjV4hcS50Db478Ou5
|
|||||||
github.com/olekukonko/errors v1.1.0/go.mod h1:ppzxA5jBKcO1vIpCXQ9ZqgDh8iwODz6OXIGKU8r5m4Y=
|
github.com/olekukonko/errors v1.1.0/go.mod h1:ppzxA5jBKcO1vIpCXQ9ZqgDh8iwODz6OXIGKU8r5m4Y=
|
||||||
github.com/olekukonko/ll v0.0.9 h1:Y+1YqDfVkqMWuEQMclsF9HUR5+a82+dxJuL1HHSRpxI=
|
github.com/olekukonko/ll v0.0.9 h1:Y+1YqDfVkqMWuEQMclsF9HUR5+a82+dxJuL1HHSRpxI=
|
||||||
github.com/olekukonko/ll v0.0.9/go.mod h1:En+sEW0JNETl26+K8eZ6/W4UQ7CYSrrgg/EdIYT2H8g=
|
github.com/olekukonko/ll v0.0.9/go.mod h1:En+sEW0JNETl26+K8eZ6/W4UQ7CYSrrgg/EdIYT2H8g=
|
||||||
github.com/olekukonko/tablewriter v1.0.9 h1:XGwRsYLC2bY7bNd93Dk51bcPZksWZmLYuaTHR0FqfL8=
|
github.com/olekukonko/tablewriter v1.1.0 h1:N0LHrshF4T39KvI96fn6GT8HEjXRXYNDrDjKFDB7RIY=
|
||||||
github.com/olekukonko/tablewriter v1.0.9/go.mod h1:5c+EBPeSqvXnLLgkm9isDdzR3wjfBkHR9Nhfp3NWrzo=
|
github.com/olekukonko/tablewriter v1.1.0/go.mod h1:5c+EBPeSqvXnLLgkm9isDdzR3wjfBkHR9Nhfp3NWrzo=
|
||||||
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
||||||
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
||||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||||
@@ -88,14 +104,18 @@ github.com/rogpeppe/go-internal v1.14.1 h1:UQB4HGPB6osV0SQTLymcB4TgvyWu6ZyliaW0t
|
|||||||
github.com/rogpeppe/go-internal v1.14.1/go.mod h1:MaRKkUm5W0goXpeCfT7UZI6fk/L7L7so1lCWt35ZSgc=
|
github.com/rogpeppe/go-internal v1.14.1/go.mod h1:MaRKkUm5W0goXpeCfT7UZI6fk/L7L7so1lCWt35ZSgc=
|
||||||
github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
|
github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
|
||||||
github.com/scylladb/termtables v0.0.0-20191203121021-c4c0b6d42ff4/go.mod h1:C1a7PQSMz9NShzorzCiG2fk9+xuCgLkPeCvMHYR2OWg=
|
github.com/scylladb/termtables v0.0.0-20191203121021-c4c0b6d42ff4/go.mod h1:C1a7PQSMz9NShzorzCiG2fk9+xuCgLkPeCvMHYR2OWg=
|
||||||
github.com/spf13/cobra v1.9.1 h1:CXSaggrXdbHK9CF+8ywj8Amf7PBRmPCOJugH954Nnlo=
|
github.com/shopspring/decimal v1.4.0 h1:bxl37RwXBklmTi0C79JfXCEBD1cqqHt0bbgBAGFp81k=
|
||||||
github.com/spf13/cobra v1.9.1/go.mod h1:nDyEzZ8ogv936Cinf6g1RU9MRY64Ir93oCnqb9wxYW0=
|
github.com/shopspring/decimal v1.4.0/go.mod h1:gawqmDU56v4yIKSwfBSFip1HdCCXN8/+DMd9qYNcwME=
|
||||||
github.com/spf13/pflag v1.0.6 h1:jFzHGLGAlb3ruxLB8MhbI6A8+AQX/2eW4qeyNZXNp2o=
|
github.com/spf13/cast v1.7.0 h1:ntdiHjuueXFgm5nzDRdOS4yfT43P5Fnud6DH50rz/7w=
|
||||||
github.com/spf13/pflag v1.0.6/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
|
github.com/spf13/cast v1.7.0/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo=
|
||||||
|
github.com/spf13/cobra v1.10.1 h1:lJeBwCfmrnXthfAupyUTzJ/J4Nc1RsHC/mSRU2dll/s=
|
||||||
|
github.com/spf13/cobra v1.10.1/go.mod h1:7SmJGaTHFVBY0jW4NXGluQoLvhqFQM+6XSKD+P4XaB0=
|
||||||
|
github.com/spf13/pflag v1.0.9 h1:9exaQaMOCwffKiiiYk6/BndUBv+iRViNW+4lEMi0PvY=
|
||||||
|
github.com/spf13/pflag v1.0.9/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
|
||||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||||
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||||
github.com/stretchr/testify v1.8.4 h1:CcVxjf3Q8PM0mHUKJCdn+eZZtm5yQwehR5yeSVQQcUk=
|
github.com/stretchr/testify v1.11.1 h1:7s2iGBzp5EwR7/aIZr8ao5+dra3wiQyKjjFuvgVKu7U=
|
||||||
github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
|
github.com/stretchr/testify v1.11.1/go.mod h1:wZwfW3scLgRK+23gO65QZefKpKQRnfz6sD981Nm4B6U=
|
||||||
github.com/tiagomelo/go-clipboard v0.1.2 h1:Ph2icR0vZRIj3v5ExvsGweBwsbbDUTlS6HoF40MkQD8=
|
github.com/tiagomelo/go-clipboard v0.1.2 h1:Ph2icR0vZRIj3v5ExvsGweBwsbbDUTlS6HoF40MkQD8=
|
||||||
github.com/tiagomelo/go-clipboard v0.1.2/go.mod h1:kXtjJBIMimZaGbxmcKZ8+JqK+acSNf5tAJiChlZBOr8=
|
github.com/tiagomelo/go-clipboard v0.1.2/go.mod h1:kXtjJBIMimZaGbxmcKZ8+JqK+acSNf5tAJiChlZBOr8=
|
||||||
github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e h1:JVG44RsyaB9T2KIHavMF/ppJZNG9ZpyihvCd0w101no=
|
github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e h1:JVG44RsyaB9T2KIHavMF/ppJZNG9ZpyihvCd0w101no=
|
||||||
@@ -107,6 +127,8 @@ github.com/zclconf/go-cty-debug v0.0.0-20240509010212-0d6042c53940 h1:4r45xpDWB6
|
|||||||
github.com/zclconf/go-cty-debug v0.0.0-20240509010212-0d6042c53940/go.mod h1:CmBdvvj3nqzfzJ6nTCIwDTPZ56aVGvDrmztiO5g3qrM=
|
github.com/zclconf/go-cty-debug v0.0.0-20240509010212-0d6042c53940/go.mod h1:CmBdvvj3nqzfzJ6nTCIwDTPZ56aVGvDrmztiO5g3qrM=
|
||||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
||||||
golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
|
golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
|
||||||
|
golang.org/x/crypto v0.38.0 h1:jt+WWG8IZlBnVbomuhg2Mdq0+BBQaHbtqHEFEigjUV8=
|
||||||
|
golang.org/x/crypto v0.38.0/go.mod h1:MvrbAqul58NNYPKnOra203SB9vpuZW0e+RRZV+Ggqjw=
|
||||||
golang.org/x/exp v0.0.0-20231006140011-7918f672742d h1:jtJma62tbqLibJ5sFQz8bKtEM8rJBtfilJ2qTU199MI=
|
golang.org/x/exp v0.0.0-20231006140011-7918f672742d h1:jtJma62tbqLibJ5sFQz8bKtEM8rJBtfilJ2qTU199MI=
|
||||||
golang.org/x/exp v0.0.0-20231006140011-7918f672742d/go.mod h1:ldy0pHrwJyGW56pPQzzkH36rKxoZW1tw7ZJpeKx+hdo=
|
golang.org/x/exp v0.0.0-20231006140011-7918f672742d/go.mod h1:ldy0pHrwJyGW56pPQzzkH36rKxoZW1tw7ZJpeKx+hdo=
|
||||||
golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
|
golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
|
||||||
@@ -130,8 +152,8 @@ golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBc
|
|||||||
golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.33.0 h1:q3i8TbbEz+JRD9ywIRlyRAQbM0qF7hu24q3teo2hbuw=
|
golang.org/x/sys v0.36.0 h1:KVRy2GtZBrk1cBYA7MKu5bEZFxQk4NIDV6RLVcC8o0k=
|
||||||
golang.org/x/sys v0.33.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k=
|
golang.org/x/sys v0.36.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
|
||||||
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
||||||
golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
|
golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
|
||||||
golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
|
golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/lithammer/fuzzysearch/fuzzy"
|
"github.com/lithammer/fuzzysearch/fuzzy"
|
||||||
"github.com/tlinden/tablizer/cfg"
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
)
|
)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|||||||
@@ -19,10 +19,10 @@ package lib
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"reflect"
|
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/tlinden/tablizer/cfg"
|
"github.com/stretchr/testify/assert"
|
||||||
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestMatchPattern(t *testing.T) {
|
func TestMatchPattern(t *testing.T) {
|
||||||
@@ -56,13 +56,11 @@ func TestMatchPattern(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
err := conf.PreparePattern(inputdata.patterns)
|
err := conf.PreparePattern(inputdata.patterns)
|
||||||
if err != nil {
|
|
||||||
t.Errorf("PreparePattern returned error: %s", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if !matchPattern(conf, inputdata.line) {
|
assert.NoError(t, err)
|
||||||
t.Errorf("matchPattern() did not match\nExp: true\nGot: false\n")
|
|
||||||
}
|
res := matchPattern(conf, inputdata.line)
|
||||||
|
assert.EqualValues(t, true, res)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -163,14 +161,12 @@ func TestFilterByFields(t *testing.T) {
|
|||||||
conf := cfg.Config{Rawfilters: inputdata.filter, InvertMatch: inputdata.invert}
|
conf := cfg.Config{Rawfilters: inputdata.filter, InvertMatch: inputdata.invert}
|
||||||
|
|
||||||
err := conf.PrepareFilters()
|
err := conf.PrepareFilters()
|
||||||
if err != nil {
|
|
||||||
t.Errorf("PrepareFilters returned error: %s", err)
|
assert.NoError(t, err)
|
||||||
}
|
|
||||||
|
|
||||||
data, _, _ := FilterByFields(conf, &data)
|
data, _, _ := FilterByFields(conf, &data)
|
||||||
if !reflect.DeepEqual(*data, inputdata.expect) {
|
|
||||||
t.Errorf("Filtered data does not match expected data:\ngot: %+v\nexp: %+v", data, inputdata.expect)
|
assert.EqualValues(t, inputdata.expect, *data)
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,24 +22,14 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"regexp"
|
"regexp"
|
||||||
"sort"
|
"slices"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/gookit/color"
|
"github.com/gookit/color"
|
||||||
"github.com/tlinden/tablizer/cfg"
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
)
|
)
|
||||||
|
|
||||||
func contains(s []int, e int) bool {
|
|
||||||
for _, a := range s {
|
|
||||||
if a == e {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func findindex(s []int, e int) (int, bool) {
|
func findindex(s []int, e int) (int, bool) {
|
||||||
for i, a := range s {
|
for i, a := range s {
|
||||||
if a == e {
|
if a == e {
|
||||||
@@ -172,48 +162,32 @@ func PrepareColumnVars(columns string, data *Tabdata) ([]int, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// deduplicate: put all values into a map (value gets map key)
|
// deduplicate columns, preserve order
|
||||||
// thereby removing duplicates, extract keys into new slice
|
deduped := []int{}
|
||||||
// and sort it
|
|
||||||
imap := make(map[int]int, len(usecolumns))
|
|
||||||
for _, i := range usecolumns {
|
for _, i := range usecolumns {
|
||||||
imap[i] = 0
|
if !slices.Contains(deduped, i) {
|
||||||
|
deduped = append(deduped, i)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// fill with deduplicated columns
|
return deduped, nil
|
||||||
usecolumns = nil
|
|
||||||
|
|
||||||
for k := range imap {
|
|
||||||
usecolumns = append(usecolumns, k)
|
|
||||||
}
|
|
||||||
|
|
||||||
sort.Ints(usecolumns)
|
|
||||||
|
|
||||||
return usecolumns, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// prepare headers: add numbers to headers
|
// prepare headers: add numbers to headers
|
||||||
func numberizeAndReduceHeaders(conf cfg.Config, data *Tabdata) {
|
func numberizeAndReduceHeaders(conf cfg.Config, data *Tabdata) {
|
||||||
numberedHeaders := []string{}
|
numberedHeaders := make([]string, len(data.headers))
|
||||||
|
|
||||||
maxwidth := 0 // start from scratch, so we only look at displayed column widths
|
maxwidth := 0 // start from scratch, so we only look at displayed column widths
|
||||||
|
|
||||||
|
// add numbers to headers if needed, get widest cell width
|
||||||
for idx, head := range data.headers {
|
for idx, head := range data.headers {
|
||||||
var headlen int
|
var headlen int
|
||||||
|
|
||||||
if len(conf.Columns) > 0 {
|
|
||||||
// -c specified
|
|
||||||
if !contains(conf.UseColumns, idx+1) {
|
|
||||||
// ignore this one
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if conf.Numbering {
|
if conf.Numbering {
|
||||||
numhead := fmt.Sprintf("%s(%d)", head, idx+1)
|
newhead := fmt.Sprintf("%s(%d)", head, idx+1)
|
||||||
headlen = len(numhead)
|
numberedHeaders[idx] = newhead
|
||||||
numberedHeaders = append(numberedHeaders, numhead)
|
headlen = len(newhead)
|
||||||
} else {
|
} else {
|
||||||
numberedHeaders = append(numberedHeaders, head)
|
|
||||||
headlen = len(head)
|
headlen = len(head)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -222,7 +196,24 @@ func numberizeAndReduceHeaders(conf cfg.Config, data *Tabdata) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
data.headers = numberedHeaders
|
if conf.Numbering {
|
||||||
|
data.headers = numberedHeaders
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(conf.UseColumns) > 0 {
|
||||||
|
// re-align headers based on user requested column list
|
||||||
|
headers := make([]string, len(conf.UseColumns))
|
||||||
|
|
||||||
|
for i, col := range conf.UseColumns {
|
||||||
|
for idx := range data.headers {
|
||||||
|
if col-1 == idx {
|
||||||
|
headers[i] = data.headers[col-1]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
data.headers = headers
|
||||||
|
}
|
||||||
|
|
||||||
if data.maxwidthHeader != maxwidth && maxwidth > 0 {
|
if data.maxwidthHeader != maxwidth && maxwidth > 0 {
|
||||||
data.maxwidthHeader = maxwidth
|
data.maxwidthHeader = maxwidth
|
||||||
@@ -234,17 +225,17 @@ func reduceColumns(conf cfg.Config, data *Tabdata) {
|
|||||||
if len(conf.Columns) > 0 {
|
if len(conf.Columns) > 0 {
|
||||||
reducedEntries := [][]string{}
|
reducedEntries := [][]string{}
|
||||||
|
|
||||||
var reducedEntry []string
|
|
||||||
|
|
||||||
for _, entry := range data.entries {
|
for _, entry := range data.entries {
|
||||||
reducedEntry = nil
|
var reducedEntry []string
|
||||||
|
|
||||||
for i, value := range entry {
|
for _, col := range conf.UseColumns {
|
||||||
if !contains(conf.UseColumns, i+1) {
|
col--
|
||||||
continue
|
|
||||||
|
for idx, value := range entry {
|
||||||
|
if idx == col {
|
||||||
|
reducedEntry = append(reducedEntry, value)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
reducedEntry = append(reducedEntry, value)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
reducedEntries = append(reducedEntries, reducedEntry)
|
reducedEntries = append(reducedEntries, reducedEntry)
|
||||||
|
|||||||
@@ -19,10 +19,11 @@ package lib
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"reflect"
|
"slices"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/tlinden/tablizer/cfg"
|
"github.com/stretchr/testify/assert"
|
||||||
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestContains(t *testing.T) {
|
func TestContains(t *testing.T) {
|
||||||
@@ -38,10 +39,9 @@ func TestContains(t *testing.T) {
|
|||||||
for _, tt := range tests {
|
for _, tt := range tests {
|
||||||
testname := fmt.Sprintf("contains-%d,%d,%t", tt.list, tt.search, tt.want)
|
testname := fmt.Sprintf("contains-%d,%d,%t", tt.list, tt.search, tt.want)
|
||||||
t.Run(testname, func(t *testing.T) {
|
t.Run(testname, func(t *testing.T) {
|
||||||
answer := contains(tt.list, tt.search)
|
answer := slices.Contains(tt.list, tt.search)
|
||||||
if answer != tt.want {
|
|
||||||
t.Errorf("got %t, want %t", answer, tt.want)
|
assert.EqualValues(t, tt.want, answer)
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -73,18 +73,17 @@ func TestPrepareColumns(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for _, testdata := range tests {
|
for _, testdata := range tests {
|
||||||
testname := fmt.Sprintf("PrepareColumns-%s-%t", testdata.input, testdata.wanterror)
|
testname := fmt.Sprintf("PrepareColumns-%s-%t",
|
||||||
|
testdata.input, testdata.wanterror)
|
||||||
t.Run(testname, func(t *testing.T) {
|
t.Run(testname, func(t *testing.T) {
|
||||||
conf := cfg.Config{Columns: testdata.input}
|
conf := cfg.Config{Columns: testdata.input}
|
||||||
err := PrepareColumns(&conf, &data)
|
err := PrepareColumns(&conf, &data)
|
||||||
if err != nil {
|
|
||||||
if !testdata.wanterror {
|
if testdata.wanterror {
|
||||||
t.Errorf("got error: %v", err)
|
assert.Error(t, err)
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
if !reflect.DeepEqual(conf.UseColumns, testdata.exp) {
|
assert.NoError(t, err)
|
||||||
t.Errorf("got: %v, expected: %v", conf.UseColumns, testdata.exp)
|
assert.EqualValues(t, testdata.exp, conf.UseColumns)
|
||||||
}
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@@ -153,18 +152,13 @@ func TestPrepareTransposerColumns(t *testing.T) {
|
|||||||
t.Run(testname, func(t *testing.T) {
|
t.Run(testname, func(t *testing.T) {
|
||||||
conf := cfg.Config{TransposeColumns: testdata.input, Transposers: testdata.transp}
|
conf := cfg.Config{TransposeColumns: testdata.input, Transposers: testdata.transp}
|
||||||
err := PrepareTransposerColumns(&conf, &data)
|
err := PrepareTransposerColumns(&conf, &data)
|
||||||
if err != nil {
|
|
||||||
if !testdata.wanterror {
|
|
||||||
t.Errorf("got error: %v", err)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if len(conf.UseTransposeColumns) != testdata.exp {
|
|
||||||
t.Errorf("got %d, want %d", conf.UseTransposeColumns, testdata.exp)
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(conf.Transposers) != len(conf.UseTransposeColumns) {
|
if testdata.wanterror {
|
||||||
t.Errorf("got %d, want %d", conf.UseTransposeColumns, testdata.exp)
|
assert.Error(t, err)
|
||||||
}
|
} else {
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.EqualValues(t, testdata.exp, len(conf.UseTransposeColumns))
|
||||||
|
assert.EqualValues(t, len(conf.UseTransposeColumns), len(conf.Transposers))
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@@ -202,10 +196,8 @@ func TestReduceColumns(t *testing.T) {
|
|||||||
c := cfg.Config{Columns: "x", UseColumns: testdata.columns}
|
c := cfg.Config{Columns: "x", UseColumns: testdata.columns}
|
||||||
data := Tabdata{entries: input}
|
data := Tabdata{entries: input}
|
||||||
reduceColumns(c, &data)
|
reduceColumns(c, &data)
|
||||||
if !reflect.DeepEqual(data.entries, testdata.expect) {
|
|
||||||
t.Errorf("reduceColumns returned invalid data:\ngot: %+v\nexp: %+v",
|
assert.EqualValues(t, testdata.expect, data.entries)
|
||||||
data.entries, testdata.expect)
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -233,10 +225,8 @@ func TestNumberizeHeaders(t *testing.T) {
|
|||||||
conf := cfg.Config{Columns: "x", UseColumns: testdata.columns, Numbering: testdata.numberize}
|
conf := cfg.Config{Columns: "x", UseColumns: testdata.columns, Numbering: testdata.numberize}
|
||||||
usedata := data
|
usedata := data
|
||||||
numberizeAndReduceHeaders(conf, &usedata)
|
numberizeAndReduceHeaders(conf, &usedata)
|
||||||
if !reflect.DeepEqual(usedata.headers, testdata.expect) {
|
|
||||||
t.Errorf("numberizeAndReduceHeaders returned invalid data:\ngot: %+v\nexp: %+v",
|
assert.EqualValues(t, testdata.expect, usedata.headers)
|
||||||
usedata.headers, testdata.expect)
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ import (
|
|||||||
"io"
|
"io"
|
||||||
"os"
|
"os"
|
||||||
|
|
||||||
"github.com/tlinden/tablizer/cfg"
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
)
|
)
|
||||||
|
|
||||||
const RWRR = 0755
|
const RWRR = 0755
|
||||||
|
|||||||
206
lib/parser.go
206
lib/parser.go
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
Copyright © 2022-2024 Thomas von Dein
|
Copyright © 2022-2025 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
|
||||||
@@ -20,13 +20,17 @@ package lib
|
|||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
"encoding/csv"
|
"encoding/csv"
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"log"
|
||||||
|
"math"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/alecthomas/repr"
|
"github.com/alecthomas/repr"
|
||||||
"github.com/tlinden/tablizer/cfg"
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
)
|
)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -39,6 +43,8 @@ func Parse(conf cfg.Config, input io.Reader) (Tabdata, error) {
|
|||||||
// first step, parse the data
|
// first step, parse the data
|
||||||
if len(conf.Separator) == 1 {
|
if len(conf.Separator) == 1 {
|
||||||
data, err = parseCSV(conf, input)
|
data, err = parseCSV(conf, input)
|
||||||
|
} else if conf.InputJSON {
|
||||||
|
data, err = parseJSON(conf, input)
|
||||||
} else {
|
} else {
|
||||||
data, err = parseTabular(conf, input)
|
data, err = parseTabular(conf, input)
|
||||||
}
|
}
|
||||||
@@ -60,6 +66,43 @@ func Parse(conf cfg.Config, input io.Reader) (Tabdata, error) {
|
|||||||
return data, err
|
return data, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Setup headers, given headers might be usable headers or just the
|
||||||
|
* first row, which we use to determine how many headers to generate,
|
||||||
|
* if enabled.
|
||||||
|
*/
|
||||||
|
func SetHeaders(conf cfg.Config, headers []string) []string {
|
||||||
|
if !conf.AutoHeaders && len(conf.CustomHeaders) == 0 {
|
||||||
|
return headers
|
||||||
|
}
|
||||||
|
|
||||||
|
if conf.AutoHeaders {
|
||||||
|
heads := make([]string, len(headers))
|
||||||
|
for idx := range headers {
|
||||||
|
heads[idx] = fmt.Sprintf("%d", idx+1)
|
||||||
|
}
|
||||||
|
|
||||||
|
return heads
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(conf.CustomHeaders) == len(headers) {
|
||||||
|
return conf.CustomHeaders
|
||||||
|
}
|
||||||
|
|
||||||
|
// use as much custom ones we have, generate the remainder
|
||||||
|
heads := make([]string, len(headers))
|
||||||
|
|
||||||
|
for idx := range headers {
|
||||||
|
if idx < len(conf.CustomHeaders) {
|
||||||
|
heads[idx] = conf.CustomHeaders[idx]
|
||||||
|
} else {
|
||||||
|
heads[idx] = fmt.Sprintf("%d", idx+1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return heads
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Parse CSV input.
|
Parse CSV input.
|
||||||
*/
|
*/
|
||||||
@@ -81,7 +124,7 @@ func parseCSV(conf cfg.Config, input io.Reader) (Tabdata, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if len(records) >= 1 {
|
if len(records) >= 1 {
|
||||||
data.headers = records[0]
|
data.headers = SetHeaders(conf, records[0])
|
||||||
data.columns = len(records)
|
data.columns = len(records)
|
||||||
|
|
||||||
for _, head := range data.headers {
|
for _, head := range data.headers {
|
||||||
@@ -92,9 +135,14 @@ func parseCSV(conf cfg.Config, input io.Reader) (Tabdata, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(records) > 1 {
|
if len(records) >= 1 {
|
||||||
data.entries = records[1:]
|
if conf.AutoHeaders || len(conf.CustomHeaders) > 0 {
|
||||||
|
data.entries = records
|
||||||
|
} else {
|
||||||
|
data.entries = records[1:]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return data, nil
|
return data, nil
|
||||||
@@ -122,7 +170,9 @@ func parseTabular(conf cfg.Config, input io.Reader) (Tabdata, error) {
|
|||||||
data.columns = len(parts)
|
data.columns = len(parts)
|
||||||
|
|
||||||
// process all header fields
|
// process all header fields
|
||||||
for _, part := range parts {
|
firstrow := make([]string, len(parts))
|
||||||
|
|
||||||
|
for idx, part := range parts {
|
||||||
// register widest header field
|
// register widest header field
|
||||||
headerlen := len(part)
|
headerlen := len(part)
|
||||||
if headerlen > data.maxwidthHeader {
|
if headerlen > data.maxwidthHeader {
|
||||||
@@ -130,11 +180,22 @@ func parseTabular(conf cfg.Config, input io.Reader) (Tabdata, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// register fields data
|
// register fields data
|
||||||
data.headers = append(data.headers, strings.TrimSpace(part))
|
firstrow[idx] = strings.TrimSpace(part)
|
||||||
|
|
||||||
// done
|
// done
|
||||||
hadFirst = true
|
hadFirst = true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
data.headers = SetHeaders(conf, firstrow)
|
||||||
|
|
||||||
|
if conf.AutoHeaders || len(conf.CustomHeaders) > 0 {
|
||||||
|
// we do not use generated headers, consider as row
|
||||||
|
if matchPattern(conf, line) == conf.InvertMatch {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
data.entries = append(data.entries, firstrow)
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
// data processing
|
// data processing
|
||||||
if matchPattern(conf, line) == conf.InvertMatch {
|
if matchPattern(conf, line) == conf.InvertMatch {
|
||||||
@@ -172,6 +233,137 @@ func parseTabular(conf cfg.Config, input io.Reader) (Tabdata, error) {
|
|||||||
return data, nil
|
return data, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Parse JSON input. We only support an array of maps.
|
||||||
|
*/
|
||||||
|
func parseRawJSON(conf cfg.Config, input io.Reader) (Tabdata, error) {
|
||||||
|
dec := json.NewDecoder(input)
|
||||||
|
headers := []string{}
|
||||||
|
idxmap := map[string]int{}
|
||||||
|
data := [][]string{}
|
||||||
|
row := []string{}
|
||||||
|
iskey := true
|
||||||
|
haveheaders := false
|
||||||
|
var currentfield string
|
||||||
|
var idx int
|
||||||
|
var isjson bool
|
||||||
|
|
||||||
|
for {
|
||||||
|
t, err := dec.Token()
|
||||||
|
if err == io.EOF {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
switch val := t.(type) {
|
||||||
|
case string:
|
||||||
|
if iskey {
|
||||||
|
if !haveheaders {
|
||||||
|
// consider only the keys of the first item as headers
|
||||||
|
headers = append(headers, val)
|
||||||
|
}
|
||||||
|
currentfield = val
|
||||||
|
} else {
|
||||||
|
if !haveheaders {
|
||||||
|
// the first row uses the order as it comes in
|
||||||
|
row = append(row, val)
|
||||||
|
} else {
|
||||||
|
// use the pre-determined order, that way items
|
||||||
|
// can be in any order as long as they contain all
|
||||||
|
// neccessary fields. They may also contain less
|
||||||
|
// fields than the first item, these will contain
|
||||||
|
// the empty string
|
||||||
|
row[idxmap[currentfield]] = val
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
case float64:
|
||||||
|
var value string
|
||||||
|
|
||||||
|
// we set precision to 0 if the float is a whole number
|
||||||
|
if val == math.Trunc(val) {
|
||||||
|
value = fmt.Sprintf("%.f", val)
|
||||||
|
} else {
|
||||||
|
value = fmt.Sprintf("%f", val)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !haveheaders {
|
||||||
|
row = append(row, value)
|
||||||
|
} else {
|
||||||
|
row[idxmap[currentfield]] = value
|
||||||
|
}
|
||||||
|
|
||||||
|
case nil:
|
||||||
|
// we ignore here if a value shall be an int or a string,
|
||||||
|
// because tablizer only works with strings anyway
|
||||||
|
if !haveheaders {
|
||||||
|
row = append(row, "")
|
||||||
|
} else {
|
||||||
|
row[idxmap[currentfield]] = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
case json.Delim:
|
||||||
|
if val.String() == "}" {
|
||||||
|
data = append(data, row)
|
||||||
|
row = make([]string, len(headers))
|
||||||
|
idx++
|
||||||
|
|
||||||
|
if !haveheaders {
|
||||||
|
// remember the array position of header fields,
|
||||||
|
// which we use to assign elements to the correct
|
||||||
|
// row index
|
||||||
|
for i, header := range headers {
|
||||||
|
idxmap[header] = i
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
haveheaders = true
|
||||||
|
}
|
||||||
|
isjson = true
|
||||||
|
default:
|
||||||
|
fmt.Printf("unknown token: %v type: %T\n", t, t)
|
||||||
|
}
|
||||||
|
|
||||||
|
iskey = !iskey
|
||||||
|
}
|
||||||
|
|
||||||
|
if isjson && (len(headers) == 0 || len(data) == 0) {
|
||||||
|
return Tabdata{}, errors.New("failed to parse JSON, input did not contain array of hashes")
|
||||||
|
}
|
||||||
|
|
||||||
|
return Tabdata{headers: headers, entries: data, columns: len(headers)}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseJSON(conf cfg.Config, input io.Reader) (Tabdata, error) {
|
||||||
|
// parse raw json
|
||||||
|
data, err := parseRawJSON(conf, input)
|
||||||
|
if err != nil {
|
||||||
|
return data, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// apply filter, if any
|
||||||
|
filtered := [][]string{}
|
||||||
|
var line string
|
||||||
|
|
||||||
|
for _, row := range data.entries {
|
||||||
|
line = strings.Join(row, " ")
|
||||||
|
|
||||||
|
if matchPattern(conf, line) == conf.InvertMatch {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
filtered = append(filtered, row)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(filtered) != len(data.entries) {
|
||||||
|
data.entries = filtered
|
||||||
|
}
|
||||||
|
|
||||||
|
return data, nil
|
||||||
|
}
|
||||||
|
|
||||||
func PostProcess(conf cfg.Config, data *Tabdata) (*Tabdata, bool, error) {
|
func PostProcess(conf cfg.Config, data *Tabdata) (*Tabdata, bool, error) {
|
||||||
var modified bool
|
var modified bool
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
Copyright © 2022 Thomas von Dein
|
Copyright © 2022-2025 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
|
||||||
@@ -19,11 +19,12 @@ package lib
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"reflect"
|
"io"
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/tlinden/tablizer/cfg"
|
"github.com/stretchr/testify/assert"
|
||||||
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
)
|
)
|
||||||
|
|
||||||
var input = []struct {
|
var input = []struct {
|
||||||
@@ -33,7 +34,7 @@ var input = []struct {
|
|||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
name: "tabular-data",
|
name: "tabular-data",
|
||||||
separator: cfg.DefaultSeparator,
|
separator: cfg.SeparatorTemplates[":default:"],
|
||||||
text: `
|
text: `
|
||||||
ONE TWO THREE
|
ONE TWO THREE
|
||||||
asd igig cxxxncnc
|
asd igig cxxxncnc
|
||||||
@@ -67,27 +68,21 @@ func TestParser(t *testing.T) {
|
|||||||
t.Run(testname, func(t *testing.T) {
|
t.Run(testname, func(t *testing.T) {
|
||||||
readFd := strings.NewReader(strings.TrimSpace(testdata.text))
|
readFd := strings.NewReader(strings.TrimSpace(testdata.text))
|
||||||
conf := cfg.Config{Separator: testdata.separator}
|
conf := cfg.Config{Separator: testdata.separator}
|
||||||
gotdata, err := Parse(conf, readFd)
|
gotdata, err := wrapValidateParser(conf, readFd)
|
||||||
|
|
||||||
if err != nil {
|
assert.NoError(t, err)
|
||||||
t.Errorf("Parser returned error: %s\nData processed so far: %+v", err, gotdata)
|
assert.EqualValues(t, data, gotdata)
|
||||||
}
|
|
||||||
|
|
||||||
if !reflect.DeepEqual(data, gotdata) {
|
|
||||||
t.Errorf("Parser returned invalid data\nExp: %+v\nGot: %+v\n",
|
|
||||||
data, gotdata)
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestParserPatternmatching(t *testing.T) {
|
func TestParserPatternmatching(t *testing.T) {
|
||||||
var tests = []struct {
|
var tests = []struct {
|
||||||
name string
|
name string
|
||||||
entries [][]string
|
entries [][]string
|
||||||
patterns []*cfg.Pattern
|
patterns []*cfg.Pattern
|
||||||
invert bool
|
invert bool
|
||||||
want bool
|
wanterror bool
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
name: "match",
|
name: "match",
|
||||||
@@ -121,18 +116,13 @@ func TestParserPatternmatching(t *testing.T) {
|
|||||||
_ = conf.PreparePattern(testdata.patterns)
|
_ = conf.PreparePattern(testdata.patterns)
|
||||||
|
|
||||||
readFd := strings.NewReader(strings.TrimSpace(inputdata.text))
|
readFd := strings.NewReader(strings.TrimSpace(inputdata.text))
|
||||||
gotdata, err := Parse(conf, readFd)
|
data, err := wrapValidateParser(conf, readFd)
|
||||||
|
|
||||||
if err != nil {
|
if testdata.wanterror {
|
||||||
if !testdata.want {
|
assert.Error(t, err)
|
||||||
t.Errorf("Parser returned error: %s\nData processed so far: %+v",
|
|
||||||
err, gotdata)
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
if !reflect.DeepEqual(testdata.entries, gotdata.entries) {
|
assert.NoError(t, err)
|
||||||
t.Errorf("Parser returned invalid data (pattern: %s, invert: %t)\nExp: %+v\nGot: %+v\n",
|
assert.EqualValues(t, testdata.entries, data.entries)
|
||||||
testdata.name, testdata.invert, testdata.entries, gotdata.entries)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@@ -158,15 +148,282 @@ asd igig
|
|||||||
19191 EDD 1 X`
|
19191 EDD 1 X`
|
||||||
|
|
||||||
readFd := strings.NewReader(strings.TrimSpace(table))
|
readFd := strings.NewReader(strings.TrimSpace(table))
|
||||||
conf := cfg.Config{Separator: cfg.DefaultSeparator}
|
conf := cfg.Config{Separator: cfg.SeparatorTemplates[":default:"]}
|
||||||
gotdata, err := Parse(conf, readFd)
|
gotdata, err := wrapValidateParser(conf, readFd)
|
||||||
|
|
||||||
if err != nil {
|
assert.NoError(t, err)
|
||||||
t.Errorf("Parser returned error: %s\nData processed so far: %+v", err, gotdata)
|
assert.EqualValues(t, data, gotdata)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParserJSONInput(t *testing.T) {
|
||||||
|
var tests = []struct {
|
||||||
|
name string
|
||||||
|
input string
|
||||||
|
expect Tabdata
|
||||||
|
wanterror bool // true: expect fail, false: expect success
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
// too deep nesting
|
||||||
|
name: "invalidjson",
|
||||||
|
wanterror: true,
|
||||||
|
input: `[
|
||||||
|
{
|
||||||
|
"item": {
|
||||||
|
"NAME": "postgres-operator-7f4c7c8485-ntlns",
|
||||||
|
"READY": "1/1",
|
||||||
|
"STATUS": "Running",
|
||||||
|
"RESTARTS": "0",
|
||||||
|
"AGE": "24h"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
expect: Tabdata{},
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
// contains nil, int and float values
|
||||||
|
name: "niljson",
|
||||||
|
wanterror: false,
|
||||||
|
input: `[
|
||||||
|
{
|
||||||
|
"NAME": "postgres-operator-7f4c7c8485-ntlns",
|
||||||
|
"READY": "1/1",
|
||||||
|
"STATUS": "Running",
|
||||||
|
"RESTARTS": 0,
|
||||||
|
"AGE": null,
|
||||||
|
"X": 12,
|
||||||
|
"Y": 34.222
|
||||||
|
}
|
||||||
|
]`,
|
||||||
|
expect: Tabdata{
|
||||||
|
columns: 7,
|
||||||
|
headers: []string{"NAME", "READY", "STATUS", "RESTARTS", "AGE", "X", "Y"},
|
||||||
|
entries: [][]string{
|
||||||
|
[]string{
|
||||||
|
"postgres-operator-7f4c7c8485-ntlns",
|
||||||
|
"1/1",
|
||||||
|
"Running",
|
||||||
|
"0",
|
||||||
|
"",
|
||||||
|
"12",
|
||||||
|
"34.222000",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
// one field missing + different order
|
||||||
|
// but shall not fail
|
||||||
|
name: "kgpfail",
|
||||||
|
wanterror: false,
|
||||||
|
input: `[
|
||||||
|
{
|
||||||
|
"NAME": "postgres-operator-7f4c7c8485-ntlns",
|
||||||
|
"READY": "1/1",
|
||||||
|
"STATUS": "Running",
|
||||||
|
"RESTARTS": "0",
|
||||||
|
"AGE": "24h"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"NAME": "wal-g-exporter-778dcd95f5-wcjzn",
|
||||||
|
"RESTARTS": "0",
|
||||||
|
"READY": "1/1",
|
||||||
|
"AGE": "24h"
|
||||||
|
}
|
||||||
|
]`,
|
||||||
|
expect: Tabdata{
|
||||||
|
columns: 5,
|
||||||
|
headers: []string{"NAME", "READY", "STATUS", "RESTARTS", "AGE"},
|
||||||
|
entries: [][]string{
|
||||||
|
[]string{
|
||||||
|
"postgres-operator-7f4c7c8485-ntlns",
|
||||||
|
"1/1",
|
||||||
|
"Running",
|
||||||
|
"0",
|
||||||
|
"24h",
|
||||||
|
},
|
||||||
|
[]string{
|
||||||
|
"wal-g-exporter-778dcd95f5-wcjzn",
|
||||||
|
"1/1",
|
||||||
|
"",
|
||||||
|
"0",
|
||||||
|
"24h",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
name: "kgp",
|
||||||
|
wanterror: false,
|
||||||
|
input: `[
|
||||||
|
{
|
||||||
|
"NAME": "postgres-operator-7f4c7c8485-ntlns",
|
||||||
|
"READY": "1/1",
|
||||||
|
"STATUS": "Running",
|
||||||
|
"RESTARTS": "0",
|
||||||
|
"AGE": "24h"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"NAME": "wal-g-exporter-778dcd95f5-wcjzn",
|
||||||
|
"STATUS": "Running",
|
||||||
|
"READY": "1/1",
|
||||||
|
"RESTARTS": "0",
|
||||||
|
"AGE": "24h"
|
||||||
|
}
|
||||||
|
]`,
|
||||||
|
expect: Tabdata{
|
||||||
|
columns: 5,
|
||||||
|
headers: []string{"NAME", "READY", "STATUS", "RESTARTS", "AGE"},
|
||||||
|
entries: [][]string{
|
||||||
|
[]string{
|
||||||
|
"postgres-operator-7f4c7c8485-ntlns",
|
||||||
|
"1/1",
|
||||||
|
"Running",
|
||||||
|
"0",
|
||||||
|
"24h",
|
||||||
|
},
|
||||||
|
[]string{
|
||||||
|
"wal-g-exporter-778dcd95f5-wcjzn",
|
||||||
|
"1/1",
|
||||||
|
"Running",
|
||||||
|
"0",
|
||||||
|
"24h",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
if !reflect.DeepEqual(data, gotdata) {
|
for _, testdata := range tests {
|
||||||
t.Errorf("Parser returned invalid data, Regex: %s\nExp: %+v\nGot: %+v\n",
|
testname := fmt.Sprintf("parse-json-%s", testdata.name)
|
||||||
conf.Separator, data, gotdata)
|
t.Run(testname, func(t *testing.T) {
|
||||||
|
conf := cfg.Config{InputJSON: true}
|
||||||
|
|
||||||
|
readFd := strings.NewReader(strings.TrimSpace(testdata.input))
|
||||||
|
data, err := wrapValidateParser(conf, readFd)
|
||||||
|
|
||||||
|
if testdata.wanterror {
|
||||||
|
assert.Error(t, err)
|
||||||
|
} else {
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.EqualValues(t, testdata.expect, data)
|
||||||
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestParserSeparators(t *testing.T) {
|
||||||
|
list := []string{"alpha", "beta", "delta"}
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
input string
|
||||||
|
sep string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
input: `🎲`,
|
||||||
|
sep: ":nonprint:",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: `|`,
|
||||||
|
sep: ":pipe:",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: ` `,
|
||||||
|
sep: ":spaces:",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: " \t ",
|
||||||
|
sep: ":tab:",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: `-`,
|
||||||
|
sep: ":nonword:",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
input: `//$`,
|
||||||
|
sep: ":special:",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, testdata := range tests {
|
||||||
|
testname := fmt.Sprintf("parse-%s", testdata.sep)
|
||||||
|
t.Run(testname, func(t *testing.T) {
|
||||||
|
header := strings.Join(list, testdata.input)
|
||||||
|
row := header
|
||||||
|
content := header + "\n" + row
|
||||||
|
|
||||||
|
readFd := strings.NewReader(strings.TrimSpace(content))
|
||||||
|
conf := cfg.Config{Separator: testdata.sep}
|
||||||
|
conf.ApplyDefaults()
|
||||||
|
|
||||||
|
gotdata, err := wrapValidateParser(conf, readFd)
|
||||||
|
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.EqualValues(t, [][]string{list}, gotdata.entries)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParserSetHeaders(t *testing.T) {
|
||||||
|
row := []string{"c", "b", "c", "d", "e"}
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
custom []string
|
||||||
|
expect []string
|
||||||
|
auto bool
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "default",
|
||||||
|
expect: row,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "auto",
|
||||||
|
expect: strings.Split("1 2 3 4 5", " "),
|
||||||
|
auto: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "custom-complete",
|
||||||
|
custom: strings.Split("A B C D E", " "),
|
||||||
|
expect: strings.Split("A B C D E", " "),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "custom-too-short",
|
||||||
|
custom: strings.Split("A B", " "),
|
||||||
|
expect: strings.Split("A B 3 4 5", " "),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "custom-too-long",
|
||||||
|
custom: strings.Split("A B C D E F G", " "),
|
||||||
|
expect: strings.Split("A B C D E", " "),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, testdata := range tests {
|
||||||
|
testname := fmt.Sprintf("parse-%s", testdata.name)
|
||||||
|
t.Run(testname, func(t *testing.T) {
|
||||||
|
conf := cfg.Config{
|
||||||
|
AutoHeaders: testdata.auto,
|
||||||
|
CustomHeaders: testdata.custom,
|
||||||
|
}
|
||||||
|
headers := SetHeaders(conf, row)
|
||||||
|
|
||||||
|
assert.NotNil(t, headers)
|
||||||
|
assert.EqualValues(t, testdata.expect, headers)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func wrapValidateParser(conf cfg.Config, input io.Reader) (Tabdata, error) {
|
||||||
|
data, err := Parse(conf, input)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return data, err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = ValidateConsistency(&data)
|
||||||
|
|
||||||
|
return data, err
|
||||||
|
}
|
||||||
|
|||||||
@@ -19,17 +19,20 @@ package lib
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/csv"
|
"encoding/csv"
|
||||||
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
"log"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
"text/template"
|
||||||
|
|
||||||
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
|
"github.com/Masterminds/sprig/v3"
|
||||||
"github.com/gookit/color"
|
"github.com/gookit/color"
|
||||||
"github.com/olekukonko/tablewriter"
|
"github.com/olekukonko/tablewriter"
|
||||||
"github.com/olekukonko/tablewriter/renderer"
|
"github.com/olekukonko/tablewriter/renderer"
|
||||||
"github.com/olekukonko/tablewriter/tw"
|
"github.com/olekukonko/tablewriter/tw"
|
||||||
"github.com/tlinden/tablizer/cfg"
|
|
||||||
"gopkg.in/yaml.v3"
|
"gopkg.in/yaml.v3"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -61,15 +64,22 @@ func printData(writer io.Writer, conf cfg.Config, data *Tabdata) {
|
|||||||
printShellData(writer, data)
|
printShellData(writer, data)
|
||||||
case cfg.Yaml:
|
case cfg.Yaml:
|
||||||
printYamlData(writer, data)
|
printYamlData(writer, data)
|
||||||
|
case cfg.Json:
|
||||||
|
printJsonData(writer, data)
|
||||||
case cfg.CSV:
|
case cfg.CSV:
|
||||||
printCSVData(writer, data)
|
printCSVData(writer, conf, data)
|
||||||
|
case cfg.Template:
|
||||||
|
printTemplateData(writer, conf, data)
|
||||||
default:
|
default:
|
||||||
printASCIIData(writer, conf, data)
|
printASCIIData(writer, conf, data)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func output(writer io.Writer, str string) {
|
func output(writer io.Writer, str string) {
|
||||||
fmt.Fprint(writer, str)
|
_, err := fmt.Fprint(writer, str)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("failed to print output: %s", err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -194,6 +204,11 @@ func printMarkdownData(writer io.Writer, conf cfg.Config, data *Tabdata) {
|
|||||||
Simple ASCII table without any borders etc, just like the input we expect
|
Simple ASCII table without any borders etc, just like the input we expect
|
||||||
*/
|
*/
|
||||||
func printASCIIData(writer io.Writer, conf cfg.Config, data *Tabdata) {
|
func printASCIIData(writer io.Writer, conf cfg.Config, data *Tabdata) {
|
||||||
|
OFS := " "
|
||||||
|
if conf.OFS != "" {
|
||||||
|
OFS = conf.OFS
|
||||||
|
}
|
||||||
|
|
||||||
tableString := &strings.Builder{}
|
tableString := &strings.Builder{}
|
||||||
|
|
||||||
styleTSV := tw.NewSymbolCustom("space").WithColumn("\t")
|
styleTSV := tw.NewSymbolCustom("space").WithColumn("\t")
|
||||||
@@ -204,8 +219,8 @@ func printASCIIData(writer io.Writer, conf cfg.Config, data *Tabdata) {
|
|||||||
Borders: tw.BorderNone,
|
Borders: tw.BorderNone,
|
||||||
Symbols: styleTSV,
|
Symbols: styleTSV,
|
||||||
Settings: tw.Settings{
|
Settings: tw.Settings{
|
||||||
Separators: tw.Separators{BetweenRows: tw.Off, BetweenColumns: tw.On},
|
Separators: tw.SeparatorsNone,
|
||||||
Lines: tw.Lines{ShowFooterLine: tw.Off, ShowHeaderLine: tw.Off},
|
Lines: tw.LinesNone,
|
||||||
},
|
},
|
||||||
})),
|
})),
|
||||||
tablewriter.WithConfig(tablewriter.Config{
|
tablewriter.WithConfig(tablewriter.Config{
|
||||||
@@ -213,23 +228,18 @@ func printASCIIData(writer io.Writer, conf cfg.Config, data *Tabdata) {
|
|||||||
Formatting: tw.CellFormatting{
|
Formatting: tw.CellFormatting{
|
||||||
AutoFormat: tw.Off,
|
AutoFormat: tw.Off,
|
||||||
},
|
},
|
||||||
Padding: tw.CellPadding{
|
Padding: tw.CellPadding{Global: tw.Padding{Left: "", Right: OFS}},
|
||||||
Global: tw.Padding{Left: "", Right: ""},
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
Row: tw.CellConfig{
|
Row: tw.CellConfig{
|
||||||
Formatting: tw.CellFormatting{
|
Formatting: tw.CellFormatting{
|
||||||
AutoWrap: tw.WrapNone,
|
AutoWrap: tw.WrapNone,
|
||||||
Alignment: tw.AlignLeft,
|
Alignment: tw.AlignLeft,
|
||||||
},
|
},
|
||||||
Padding: tw.CellPadding{
|
Padding: tw.CellPadding{Global: tw.Padding{Right: OFS}},
|
||||||
Global: tw.Padding{Left: "", Right: ""},
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
|
||||||
Debug: true,
|
Debug: true,
|
||||||
}),
|
}),
|
||||||
tablewriter.WithPadding(tw.PaddingNone),
|
|
||||||
)
|
)
|
||||||
|
|
||||||
if !conf.NoHeaders {
|
if !conf.NoHeaders {
|
||||||
@@ -291,6 +301,35 @@ func printShellData(writer io.Writer, data *Tabdata) {
|
|||||||
output(writer, out)
|
output(writer, out)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func printJsonData(writer io.Writer, data *Tabdata) {
|
||||||
|
objlist := make([]map[string]any, len(data.entries))
|
||||||
|
|
||||||
|
if len(data.entries) > 0 {
|
||||||
|
for i, entry := range data.entries {
|
||||||
|
obj := make(map[string]any, len(entry))
|
||||||
|
|
||||||
|
for idx, value := range entry {
|
||||||
|
num, err := strconv.Atoi(value)
|
||||||
|
if err == nil {
|
||||||
|
obj[data.headers[idx]] = num
|
||||||
|
} else {
|
||||||
|
obj[data.headers[idx]] = value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
objlist[i] = obj
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
jsonstr, err := json.MarshalIndent(&objlist, "", " ")
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
output(writer, string(jsonstr))
|
||||||
|
}
|
||||||
|
|
||||||
func printYamlData(writer io.Writer, data *Tabdata) {
|
func printYamlData(writer io.Writer, data *Tabdata) {
|
||||||
type Data struct {
|
type Data struct {
|
||||||
Entries []map[string]interface{} `yaml:"entries"`
|
Entries []map[string]interface{} `yaml:"entries"`
|
||||||
@@ -328,8 +367,14 @@ func printYamlData(writer io.Writer, data *Tabdata) {
|
|||||||
output(writer, string(yamlstr))
|
output(writer, string(yamlstr))
|
||||||
}
|
}
|
||||||
|
|
||||||
func printCSVData(writer io.Writer, data *Tabdata) {
|
func printCSVData(writer io.Writer, conf cfg.Config, data *Tabdata) {
|
||||||
|
OFS := ","
|
||||||
|
if conf.OFS != "" {
|
||||||
|
OFS = conf.OFS
|
||||||
|
}
|
||||||
|
|
||||||
csvout := csv.NewWriter(writer)
|
csvout := csv.NewWriter(writer)
|
||||||
|
csvout.Comma = []rune(OFS)[0]
|
||||||
|
|
||||||
if err := csvout.Write(data.headers); err != nil {
|
if err := csvout.Write(data.headers); err != nil {
|
||||||
log.Fatalln("error writing record to csv:", err)
|
log.Fatalln("error writing record to csv:", err)
|
||||||
@@ -347,3 +392,25 @@ func printCSVData(writer io.Writer, data *Tabdata) {
|
|||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func printTemplateData(writer io.Writer, conf cfg.Config, data *Tabdata) {
|
||||||
|
tmpl, err := template.New("printer").Funcs(sprig.TxtFuncMap()).Parse(conf.Template)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("failed to parse template: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
buf := strings.Builder{}
|
||||||
|
|
||||||
|
for line, dict := range data.ToMap() {
|
||||||
|
err = tmpl.Execute(&buf, dict)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("failed to execute template in line %d: %s", line, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
buf.WriteString("\n")
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := fmt.Fprintln(writer, buf.String()); err != nil {
|
||||||
|
log.Fatalf("failed to print output: %s", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -23,7 +23,8 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/tlinden/tablizer/cfg"
|
"github.com/stretchr/testify/assert"
|
||||||
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
)
|
)
|
||||||
|
|
||||||
func newData() Tabdata {
|
func newData() Tabdata {
|
||||||
@@ -77,10 +78,10 @@ var tests = []struct {
|
|||||||
numberize: true,
|
numberize: true,
|
||||||
name: "default",
|
name: "default",
|
||||||
expect: `
|
expect: `
|
||||||
NAME(1) DURATION(2) COUNT(3) WHEN(4)
|
NAME(1) DURATION(2) COUNT(3) WHEN(4)
|
||||||
beta 1d10h5m1s 33 3/1/2014
|
beta 1d10h5m1s 33 3/1/2014
|
||||||
alpha 4h35m 170 2013-Feb-03
|
alpha 4h35m 170 2013-Feb-03
|
||||||
ceta 33d12h 9 06/Jan/2008 15:04:05 -0700`,
|
ceta 33d12h 9 06/Jan/2008 15:04:05 -0700`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
mode: cfg.CSV,
|
mode: cfg.CSV,
|
||||||
@@ -124,6 +125,31 @@ ceta,33d12h,9,06/Jan/2008 15:04:05 -0700`,
|
|||||||
NAME="beta" DURATION="1d10h5m1s" COUNT="33" WHEN="3/1/2014"
|
NAME="beta" DURATION="1d10h5m1s" COUNT="33" WHEN="3/1/2014"
|
||||||
NAME="alpha" DURATION="4h35m" COUNT="170" WHEN="2013-Feb-03"
|
NAME="alpha" DURATION="4h35m" COUNT="170" WHEN="2013-Feb-03"
|
||||||
NAME="ceta" DURATION="33d12h" COUNT="9" WHEN="06/Jan/2008 15:04:05 -0700"`,
|
NAME="ceta" DURATION="33d12h" COUNT="9" WHEN="06/Jan/2008 15:04:05 -0700"`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "json",
|
||||||
|
mode: cfg.Json,
|
||||||
|
numberize: false,
|
||||||
|
expect: `[
|
||||||
|
{
|
||||||
|
"COUNT": 33,
|
||||||
|
"DURATION": "1d10h5m1s",
|
||||||
|
"NAME": "beta",
|
||||||
|
"WHEN": "3/1/2014"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"COUNT": 170,
|
||||||
|
"DURATION": "4h35m",
|
||||||
|
"NAME": "alpha",
|
||||||
|
"WHEN": "2013-Feb-03"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"COUNT": 9,
|
||||||
|
"DURATION": "33d12h",
|
||||||
|
"NAME": "ceta",
|
||||||
|
"WHEN": "06/Jan/2008 15:04:05 -0700"
|
||||||
|
}
|
||||||
|
]`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "yaml",
|
name: "yaml",
|
||||||
@@ -173,10 +199,10 @@ DURATION(2): 33d12h
|
|||||||
numberize: true,
|
numberize: true,
|
||||||
desc: false,
|
desc: false,
|
||||||
expect: `
|
expect: `
|
||||||
NAME(1) DURATION(2) COUNT(3) WHEN(4)
|
NAME(1) DURATION(2) COUNT(3) WHEN(4)
|
||||||
ceta 33d12h 9 06/Jan/2008 15:04:05 -0700
|
ceta 33d12h 9 06/Jan/2008 15:04:05 -0700
|
||||||
beta 1d10h5m1s 33 3/1/2014
|
beta 1d10h5m1s 33 3/1/2014
|
||||||
alpha 4h35m 170 2013-Feb-03`,
|
alpha 4h35m 170 2013-Feb-03`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "sortbycolumn4",
|
name: "sortbycolumn4",
|
||||||
@@ -185,10 +211,10 @@ alpha 4h35m 170 2013-Feb-03`,
|
|||||||
desc: false,
|
desc: false,
|
||||||
numberize: true,
|
numberize: true,
|
||||||
expect: `
|
expect: `
|
||||||
NAME(1) DURATION(2) COUNT(3) WHEN(4)
|
NAME(1) DURATION(2) COUNT(3) WHEN(4)
|
||||||
ceta 33d12h 9 06/Jan/2008 15:04:05 -0700
|
ceta 33d12h 9 06/Jan/2008 15:04:05 -0700
|
||||||
alpha 4h35m 170 2013-Feb-03
|
alpha 4h35m 170 2013-Feb-03
|
||||||
beta 1d10h5m1s 33 3/1/2014`,
|
beta 1d10h5m1s 33 3/1/2014`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "sortbycolumn2",
|
name: "sortbycolumn2",
|
||||||
@@ -197,10 +223,10 @@ beta 1d10h5m1s 33 3/1/2014`,
|
|||||||
numberize: true,
|
numberize: true,
|
||||||
desc: false,
|
desc: false,
|
||||||
expect: `
|
expect: `
|
||||||
NAME(1) DURATION(2) COUNT(3) WHEN(4)
|
NAME(1) DURATION(2) COUNT(3) WHEN(4)
|
||||||
alpha 4h35m 170 2013-Feb-03
|
alpha 4h35m 170 2013-Feb-03
|
||||||
beta 1d10h5m1s 33 3/1/2014
|
beta 1d10h5m1s 33 3/1/2014
|
||||||
ceta 33d12h 9 06/Jan/2008 15:04:05 -0700`,
|
ceta 33d12h 9 06/Jan/2008 15:04:05 -0700`,
|
||||||
},
|
},
|
||||||
|
|
||||||
// ----------------------- UseColumns Tests
|
// ----------------------- UseColumns Tests
|
||||||
@@ -210,44 +236,44 @@ ceta 33d12h 9 06/Jan/2008 15:04:05 -0700`,
|
|||||||
numberize: true,
|
numberize: true,
|
||||||
usecolstr: "1,4",
|
usecolstr: "1,4",
|
||||||
expect: `
|
expect: `
|
||||||
NAME(1) WHEN(4)
|
NAME(1) WHEN(4)
|
||||||
beta 3/1/2014
|
beta 3/1/2014
|
||||||
alpha 2013-Feb-03
|
alpha 2013-Feb-03
|
||||||
ceta 06/Jan/2008 15:04:05 -0700`,
|
ceta 06/Jan/2008 15:04:05 -0700`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "usecolumns",
|
name: "usecolumns2",
|
||||||
usecol: []int{2},
|
usecol: []int{2},
|
||||||
numberize: true,
|
numberize: true,
|
||||||
usecolstr: "2",
|
usecolstr: "2",
|
||||||
expect: `
|
expect: `
|
||||||
DURATION(2)
|
DURATION(2)
|
||||||
1d10h5m1s
|
1d10h5m1s
|
||||||
4h35m
|
4h35m
|
||||||
33d12h`,
|
33d12h`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "usecolumns",
|
name: "usecolumns3",
|
||||||
usecol: []int{3},
|
usecol: []int{3},
|
||||||
numberize: true,
|
numberize: true,
|
||||||
usecolstr: "3",
|
usecolstr: "3",
|
||||||
expect: `
|
expect: `
|
||||||
COUNT(3)
|
COUNT(3)
|
||||||
33
|
33
|
||||||
170
|
170
|
||||||
9`,
|
9`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "usecolumns",
|
name: "usecolumns4",
|
||||||
column: 0,
|
column: 0,
|
||||||
usecol: []int{1, 3},
|
usecol: []int{1, 3},
|
||||||
numberize: true,
|
numberize: true,
|
||||||
usecolstr: "1,3",
|
usecolstr: "1,3",
|
||||||
expect: `
|
expect: `
|
||||||
NAME(1) COUNT(3)
|
NAME(1) COUNT(3)
|
||||||
beta 33
|
beta 33
|
||||||
alpha 170
|
alpha 170
|
||||||
ceta 9`,
|
ceta 9`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "usecolumns",
|
name: "usecolumns",
|
||||||
@@ -255,10 +281,10 @@ ceta 9`,
|
|||||||
numberize: true,
|
numberize: true,
|
||||||
usecolstr: "2,4",
|
usecolstr: "2,4",
|
||||||
expect: `
|
expect: `
|
||||||
DURATION(2) WHEN(4)
|
DURATION(2) WHEN(4)
|
||||||
1d10h5m1s 3/1/2014
|
1d10h5m1s 3/1/2014
|
||||||
4h35m 2013-Feb-03
|
4h35m 2013-Feb-03
|
||||||
33d12h 06/Jan/2008 15:04:05 -0700`,
|
33d12h 06/Jan/2008 15:04:05 -0700`,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -280,12 +306,18 @@ func TestPrinter(t *testing.T) {
|
|||||||
Numbering: testdata.numberize,
|
Numbering: testdata.numberize,
|
||||||
UseColumns: testdata.usecol,
|
UseColumns: testdata.usecol,
|
||||||
NoColor: true,
|
NoColor: true,
|
||||||
|
OFS: " ",
|
||||||
|
}
|
||||||
|
|
||||||
|
if conf.OutputMode == cfg.CSV {
|
||||||
|
conf.OFS = ","
|
||||||
}
|
}
|
||||||
|
|
||||||
if testdata.column > 0 {
|
if testdata.column > 0 {
|
||||||
conf.UseSortByColumn = []int{testdata.column}
|
conf.UseSortByColumn = []int{testdata.column}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
conf.Separator = cfg.SeparatorTemplates[":default:"]
|
||||||
conf.ApplyDefaults()
|
conf.ApplyDefaults()
|
||||||
|
|
||||||
// the test checks the len!
|
// the test checks the len!
|
||||||
@@ -302,10 +334,7 @@ func TestPrinter(t *testing.T) {
|
|||||||
|
|
||||||
got := strings.TrimSpace(writer.String())
|
got := strings.TrimSpace(writer.String())
|
||||||
|
|
||||||
if got != exp {
|
assert.EqualValues(t, exp, got)
|
||||||
t.Errorf("not rendered correctly:\n+++ got:\n%s\n+++ want:\n%s",
|
|
||||||
got, exp)
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,8 +23,8 @@ import (
|
|||||||
"sort"
|
"sort"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
|
||||||
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
"github.com/araddon/dateparse"
|
"github.com/araddon/dateparse"
|
||||||
"github.com/tlinden/tablizer/cfg"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func sortTable(conf cfg.Config, data *Tabdata) {
|
func sortTable(conf cfg.Config, data *Tabdata) {
|
||||||
@@ -38,6 +38,12 @@ func sortTable(conf cfg.Config, data *Tabdata) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for _, column := range conf.UseSortByColumn {
|
||||||
|
if column > len(data.headers) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// actual sorting
|
// actual sorting
|
||||||
sort.SliceStable(data.entries, func(i, j int) bool {
|
sort.SliceStable(data.entries, func(i, j int) bool {
|
||||||
// holds the result of a sort of one column
|
// holds the result of a sort of one column
|
||||||
|
|||||||
@@ -21,7 +21,8 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/tlinden/tablizer/cfg"
|
"github.com/stretchr/testify/assert"
|
||||||
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestDuration2Seconds(t *testing.T) {
|
func TestDuration2Seconds(t *testing.T) {
|
||||||
@@ -41,9 +42,7 @@ func TestDuration2Seconds(t *testing.T) {
|
|||||||
testname := fmt.Sprintf("duration-%s", testdata.dur)
|
testname := fmt.Sprintf("duration-%s", testdata.dur)
|
||||||
t.Run(testname, func(t *testing.T) {
|
t.Run(testname, func(t *testing.T) {
|
||||||
seconds := duration2int(testdata.dur)
|
seconds := duration2int(testdata.dur)
|
||||||
if seconds != testdata.expect {
|
assert.EqualValues(t, testdata.expect, seconds)
|
||||||
t.Errorf("got %d, want %d", seconds, testdata.expect)
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -74,9 +73,7 @@ func TestCompare(t *testing.T) {
|
|||||||
t.Run(testname, func(t *testing.T) {
|
t.Run(testname, func(t *testing.T) {
|
||||||
c := cfg.Config{SortMode: testdata.mode, SortDescending: testdata.desc}
|
c := cfg.Config{SortMode: testdata.mode, SortDescending: testdata.desc}
|
||||||
got := compare(&c, testdata.a, testdata.b)
|
got := compare(&c, testdata.a, testdata.b)
|
||||||
if got != testdata.want {
|
assert.EqualValues(t, testdata.want, got)
|
||||||
t.Errorf("got %d, want %d", got, testdata.want)
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
Copyright © 2022-2024 Thomas von Dein
|
Copyright © 2022-2025 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
|
||||||
@@ -17,6 +17,11 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|||||||
|
|
||||||
package lib
|
package lib
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
// contains a whole parsed table
|
// contains a whole parsed table
|
||||||
type Tabdata struct {
|
type Tabdata struct {
|
||||||
maxwidthHeader int // longest header
|
maxwidthHeader int // longest header
|
||||||
@@ -34,3 +39,41 @@ func (data *Tabdata) CloneEmpty() Tabdata {
|
|||||||
|
|
||||||
return newdata
|
return newdata
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// convert our internal data to a list of maps
|
||||||
|
func (data *Tabdata) ToMap() []map[string]any {
|
||||||
|
dictlist := make([]map[string]any, len(data.entries))
|
||||||
|
headers := make([]string, len(data.headers))
|
||||||
|
|
||||||
|
for idx, header := range data.headers {
|
||||||
|
headers[idx] = strings.ToLower(header)
|
||||||
|
}
|
||||||
|
|
||||||
|
for line, row := range data.entries {
|
||||||
|
dict := make(map[string]any, len(data.headers))
|
||||||
|
|
||||||
|
for col, cell := range row {
|
||||||
|
n, ok := strconv.Atoi(cell)
|
||||||
|
if ok == nil {
|
||||||
|
dict[headers[col]] = n
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.ToLower(cell) == "true" {
|
||||||
|
dict[headers[col]] = true
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.ToLower(cell) == "false" {
|
||||||
|
dict[headers[col]] = false
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
dict[headers[col]] = cell
|
||||||
|
}
|
||||||
|
|
||||||
|
dictlist[line] = dict
|
||||||
|
}
|
||||||
|
|
||||||
|
return dictlist
|
||||||
|
}
|
||||||
@@ -26,7 +26,7 @@ import (
|
|||||||
"github.com/charmbracelet/lipgloss"
|
"github.com/charmbracelet/lipgloss"
|
||||||
"github.com/evertras/bubble-table/table"
|
"github.com/evertras/bubble-table/table"
|
||||||
"github.com/mattn/go-isatty"
|
"github.com/mattn/go-isatty"
|
||||||
"github.com/tlinden/tablizer/cfg"
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
)
|
)
|
||||||
|
|
||||||
// The context exists outside of the bubble loop, and is being used as
|
// The context exists outside of the bubble loop, and is being used as
|
||||||
|
|||||||
@@ -22,7 +22,7 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/tiagomelo/go-clipboard/clipboard"
|
"github.com/tiagomelo/go-clipboard/clipboard"
|
||||||
"github.com/tlinden/tablizer/cfg"
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
)
|
)
|
||||||
|
|
||||||
func yankColumns(conf cfg.Config, data *Tabdata) {
|
func yankColumns(conf cfg.Config, data *Tabdata) {
|
||||||
|
|||||||
@@ -22,8 +22,9 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
"github.com/tiagomelo/go-clipboard/clipboard"
|
"github.com/tiagomelo/go-clipboard/clipboard"
|
||||||
"github.com/tlinden/tablizer/cfg"
|
"codeberg.org/scip/tablizer/cfg"
|
||||||
)
|
)
|
||||||
|
|
||||||
var yanktests = []struct {
|
var yanktests = []struct {
|
||||||
@@ -59,14 +60,9 @@ func DISABLED_TestYankColumns(t *testing.T) {
|
|||||||
printData(&writer, conf, &data)
|
printData(&writer, conf, &data)
|
||||||
|
|
||||||
got, err := cb.PasteText()
|
got, err := cb.PasteText()
|
||||||
if err != nil {
|
|
||||||
t.Errorf("failed to fetch yanked text from clipboard")
|
|
||||||
}
|
|
||||||
|
|
||||||
if got != testdata.expect {
|
assert.NoError(t, err)
|
||||||
t.Errorf("not yanked correctly:\n+++ got:\n%s\n+++ want:\n%s",
|
assert.EqualValues(t, testdata.expect, got)
|
||||||
got, testdata.expect)
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
2
main.go
2
main.go
@@ -20,7 +20,7 @@ package main
|
|||||||
import (
|
import (
|
||||||
"os"
|
"os"
|
||||||
|
|
||||||
"github.com/tlinden/tablizer/cmd"
|
"codeberg.org/scip/tablizer/cmd"
|
||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
|
|||||||
2
mkrel.sh
2
mkrel.sh
@@ -49,7 +49,7 @@ for D in $DIST; do
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
set -x
|
set -x
|
||||||
GOOS=${os} GOARCH=${arch} go build -tags osusergo,netgo -ldflags "-extldflags=-static -w -X 'github.com/tlinden/tablizer/cfg.VERSION=${version}'" --trimpath $pie -o ${binfile}
|
GOOS=${os} GOARCH=${arch} go build -tags osusergo,netgo -ldflags "-extldflags=-static -w -X 'codeberg.org/scip/tablizer/cfg.VERSION=${version}'" --trimpath $pie -o ${binfile}
|
||||||
strip --strip-all ${binfile}
|
strip --strip-all ${binfile}
|
||||||
mkdir -p ${tardir}
|
mkdir -p ${tardir}
|
||||||
cp ${binfile} README.md LICENSE ${tardir}/
|
cp ${binfile} README.md LICENSE ${tardir}/
|
||||||
|
|||||||
@@ -1,7 +1,10 @@
|
|||||||
# usage
|
# usage
|
||||||
exec tablizer -h
|
exec tablizer --help
|
||||||
stdout Usage
|
stdout Usage
|
||||||
|
|
||||||
|
exec tablizer -h
|
||||||
|
stdout show
|
||||||
|
|
||||||
# version
|
# version
|
||||||
exec tablizer -V
|
exec tablizer -V
|
||||||
stdout version
|
stdout version
|
||||||
|
|||||||
16
t/test-template.txtar
Normal file
16
t/test-template.txtar
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
exec tablizer -r testtable.txt -P '{{.species | title}}s are {{.type }}'
|
||||||
|
stdout 'Humans are invasive'
|
||||||
|
|
||||||
|
exec tablizer -r testtable.txt -P '{{.species | sha256sum}}'
|
||||||
|
stdout '79a5478768d2447431a90f7f4549df735f50ad541371464c248abc7522dc3a01'
|
||||||
|
|
||||||
|
exec tablizer -r testtable.txt -P '{{add 100 .count}}'
|
||||||
|
stdout 104
|
||||||
|
|
||||||
|
-- testtable.txt --
|
||||||
|
SPECIES TYPE HOME COUNT STAGE
|
||||||
|
human invasive earth 4 brink
|
||||||
|
riedl peaceful keauna 33 civilized
|
||||||
|
namak invasive namak 123 imperium
|
||||||
|
heduu peaceful iu 66 imperium
|
||||||
|
kenaha peaceful kohi 3333 hunter-gatherer
|
||||||
187
tablizer.1
187
tablizer.1
@@ -133,7 +133,7 @@
|
|||||||
.\" ========================================================================
|
.\" ========================================================================
|
||||||
.\"
|
.\"
|
||||||
.IX Title "TABLIZER 1"
|
.IX Title "TABLIZER 1"
|
||||||
.TH TABLIZER 1 "2025-09-11" "1" "User Commands"
|
.TH TABLIZER 1 "2025-12-08" "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
|
||||||
@@ -147,45 +147,51 @@ tablizer \- Manipulate tabular output of other programs
|
|||||||
\& tablizer [regex,...] [\-r file] [flags]
|
\& tablizer [regex,...] [\-r 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 ,)
|
||||||
\& \-v, \-\-invert\-match select non\-matching rows
|
\& \-v, \-\-invert\-match select non\-matching rows
|
||||||
\& \-n, \-\-numbering Enable header numbering
|
\& \-n, \-\-numbering Enable header numbering
|
||||||
\& \-N, \-\-no\-color Disable pattern highlighting
|
\& \-N, \-\-no\-color Disable pattern highlighting
|
||||||
\& \-H, \-\-no\-headers Disable headers display
|
\& \-H, \-\-no\-headers Disable headers display
|
||||||
\& \-s, \-\-separator string Custom field separator
|
\& \-s, \-\-separator <string> Custom field separator (maybe char, string or :class:)
|
||||||
\& \-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
|
||||||
\& \-I, \-\-interactive Interactively filter and select rows
|
\& \-j, \-\-json Read JSON input (must be array of hashes)
|
||||||
|
\& \-I, \-\-interactive Interactively filter and select rows
|
||||||
|
\& \-g, \-\-auto\-headers Generate headers if there are none present in input
|
||||||
|
\& \-x, \-\-custom\-headers a,b,... Use custom headers, separated by comma
|
||||||
\&
|
\&
|
||||||
\& Output Flags (mutually exclusive):
|
\& Output Flags (mutually exclusive):
|
||||||
\& \-X, \-\-extended Enable extended output
|
\& \-X, \-\-extended Enable extended output
|
||||||
\& \-M, \-\-markdown Enable markdown table output
|
\& \-M, \-\-markdown Enable markdown table output
|
||||||
\& \-O, \-\-orgtbl Enable org\-mode table output
|
\& \-O, \-\-orgtbl Enable org\-mode table output
|
||||||
\& \-S, \-\-shell Enable shell evaluable output
|
\& \-S, \-\-shell Enable shell evaluable output
|
||||||
\& \-Y, \-\-yaml Enable yaml output
|
\& \-Y, \-\-yaml Enable yaml output
|
||||||
\& \-C, \-\-csv Enable CSV output
|
\& \-J, \-\-jsonout Enable JSON output
|
||||||
\& \-A, \-\-ascii Default output mode, ascii tabular
|
\& \-C, \-\-csv Enable CSV output
|
||||||
\& \-L, \-\-hightlight\-lines Use alternating background colors for tables
|
\& \-A, \-\-ascii Default output mode, ascii tabular
|
||||||
\& \-y, \-\-yank\-columns Yank specified columns (separated by ,) to clipboard,
|
\& \-P, \-\-template <tpl> Enable template mode with template <tpl>
|
||||||
\& space separated
|
\& \-L, \-\-hightlight\-lines Use alternating background colors for tables
|
||||||
|
\& \-o, \-\-ofs <char> Output field separator, used by \-A and \-C.
|
||||||
|
\& \-y, \-\-yank\-columns Yank specified columns (separated by ,) to clipboard,
|
||||||
|
\& space separated
|
||||||
\&
|
\&
|
||||||
\& Sort Mode Flags (mutually exclusive):
|
\& Sort Mode Flags (mutually exclusive):
|
||||||
\& \-a, \-\-sort\-age sort according to age (duration) string
|
\& \-a, \-\-sort\-age sort according to age (duration) string
|
||||||
\& \-D, \-\-sort\-desc Sort in descending order (default: ascending)
|
\& \-D, \-\-sort\-desc Sort in descending order (default: ascending)
|
||||||
\& \-i, \-\-sort\-numeric sort according to string numerical value
|
\& \-i, \-\-sort\-numeric sort according to string numerical value
|
||||||
\& \-t, \-\-sort\-time sort according to time string
|
\& \-t, \-\-sort\-time sort according to time string
|
||||||
\&
|
\&
|
||||||
\& Other Flags:
|
\& Other Flags:
|
||||||
\& \-r \-\-read\-file <file> Use <file> as input instead of STDIN
|
\& \-r \-\-read\-file <file> Use <file> as input instead of STDIN
|
||||||
\& \-\-completion <shell> Generate the autocompletion script for <shell>
|
\& \-\-completion <shell> Generate the autocompletion script for <shell>
|
||||||
\& \-f, \-\-config <file> Configuration file (default: ~/.config/tablizer/config)
|
\& \-f, \-\-config <file> Configuration file (default: ~/.config/tablizer/config)
|
||||||
\& \-d, \-\-debug Enable debugging
|
\& \-d, \-\-debug Enable debugging
|
||||||
\& \-h, \-\-help help for tablizer
|
\& \-h, \-\-help help for tablizer
|
||||||
\& \-m, \-\-man Display manual page
|
\& \-m, \-\-man Display manual page
|
||||||
\& \-V, \-\-version Print program version
|
\& \-V, \-\-version Print program version
|
||||||
.Ve
|
.Ve
|
||||||
.SH "DESCRIPTION"
|
.SH "DESCRIPTION"
|
||||||
.IX Header "DESCRIPTION"
|
.IX Header "DESCRIPTION"
|
||||||
@@ -214,20 +220,20 @@ pattern. Hence:
|
|||||||
.PP
|
.PP
|
||||||
.Vb 2
|
.Vb 2
|
||||||
\& # read from STDIN
|
\& # read from STDIN
|
||||||
\& kubectl get pods | tablizer
|
\& > kubectl get pods | tablizer
|
||||||
\&
|
\&
|
||||||
\& # read a file
|
\& # read a file
|
||||||
\& tablizer filename
|
\& > tablizer \-r filename
|
||||||
\&
|
\&
|
||||||
\& # search for pattern in a file (works like grep)
|
\& # search for pattern in a file (works like grep)
|
||||||
\& tablizer regex filename
|
\& > tablizer regex \-r filename
|
||||||
\&
|
\&
|
||||||
\& # search for pattern in STDIN
|
\& # search for pattern in STDIN
|
||||||
\& kubectl get pods | tablizer regex
|
\& > kubectl get pods | tablizer regex
|
||||||
.Ve
|
.Ve
|
||||||
.PP
|
.PP
|
||||||
The output looks like the original one but every header field will
|
The output looks like the original one. You can add the option \fB\-n\fR,
|
||||||
have a numer associated with it, e.g.:
|
then every header field will have a numer associated with it, e.g.:
|
||||||
.PP
|
.PP
|
||||||
.Vb 1
|
.Vb 1
|
||||||
\& NAME(1) READY(2) STATUS(3) RESTARTS(4) AGE(5)
|
\& NAME(1) READY(2) STATUS(3) RESTARTS(4) AGE(5)
|
||||||
@@ -237,13 +243,24 @@ These numbers denote the column and you can use them to specify which
|
|||||||
columns you want to have in your output (see \s-1COLUMNS\s0:
|
columns you want to have in your output (see \s-1COLUMNS\s0:
|
||||||
.PP
|
.PP
|
||||||
.Vb 1
|
.Vb 1
|
||||||
\& kubectl get pods | tablizer \-c1,3
|
\& > kubectl get pods | tablizer \-c1,3
|
||||||
.Ve
|
.Ve
|
||||||
.PP
|
.PP
|
||||||
You can specify the numbers in any order but output will always follow
|
You can specify the numbers in any order but output will always follow
|
||||||
the original order.
|
the original order.
|
||||||
.PP
|
.PP
|
||||||
The numbering can be suppressed by using the \fB\-n\fR option.
|
However, you may also just use the header names instead of numbers,
|
||||||
|
eg:
|
||||||
|
.PP
|
||||||
|
.Vb 1
|
||||||
|
\& > kubectl get pods | tablizer \-cname,status
|
||||||
|
.Ve
|
||||||
|
.PP
|
||||||
|
You can also use regular expressions with \fB\-c\fR, eg:
|
||||||
|
.PP
|
||||||
|
.Vb 1
|
||||||
|
\& > kubectl get pods | tablizer \-c \*(Aq[ae]\*(Aq
|
||||||
|
.Ve
|
||||||
.PP
|
.PP
|
||||||
By default tablizer shows a header containing the names of each
|
By default tablizer shows a header containing the names of each
|
||||||
column. This can be disabled using the \fB\-H\fR option. Be aware that
|
column. This can be disabled using the \fB\-H\fR option. Be aware that
|
||||||
@@ -280,6 +297,62 @@ Sorts timestamps.
|
|||||||
.PP
|
.PP
|
||||||
Finally the \fB\-d\fR option enables debugging output which is mostly
|
Finally the \fB\-d\fR option enables debugging output which is mostly
|
||||||
useful for the developer.
|
useful for the developer.
|
||||||
|
.SS "\s-1SEPARATOR\s0"
|
||||||
|
.IX Subsection "SEPARATOR"
|
||||||
|
The option \fB\-s\fR can be a single character, in which case the \s-1CSV\s0
|
||||||
|
parser will be invoked. You can also specify a string as
|
||||||
|
separator. The string will be interpreted as literal string unless it
|
||||||
|
is a valid go regular expression. For example:
|
||||||
|
.PP
|
||||||
|
.Vb 1
|
||||||
|
\& \-s \*(Aq\et{2,}\e\*(Aq
|
||||||
|
.Ve
|
||||||
|
.PP
|
||||||
|
is being used as a regexp and will match two or more consecutive tabs.
|
||||||
|
.PP
|
||||||
|
.Vb 1
|
||||||
|
\& \-s \*(Aqfoo\*(Aq
|
||||||
|
.Ve
|
||||||
|
.PP
|
||||||
|
on the other hand is no regular expression and will be used literally.
|
||||||
|
.PP
|
||||||
|
To make live easier, there are a couple of predefined regular
|
||||||
|
expressions, which you can specify as classes:
|
||||||
|
.Sp
|
||||||
|
.RS 4
|
||||||
|
* :tab:
|
||||||
|
.Sp
|
||||||
|
Matches a tab and eats spaces around it.
|
||||||
|
.Sp
|
||||||
|
* :spaces:
|
||||||
|
.Sp
|
||||||
|
Matches 2 or more spaces.
|
||||||
|
.Sp
|
||||||
|
* :pipe:
|
||||||
|
.Sp
|
||||||
|
Matches a pipe character and eats spaces around it.
|
||||||
|
.Sp
|
||||||
|
* :default:
|
||||||
|
.Sp
|
||||||
|
Matches 2 or more spaces or tab. This is the default separator if none
|
||||||
|
is specified.
|
||||||
|
.Sp
|
||||||
|
* :nonword:
|
||||||
|
.Sp
|
||||||
|
Matches a non-word character.
|
||||||
|
.Sp
|
||||||
|
* :nondigit:
|
||||||
|
.Sp
|
||||||
|
Matches a non-digit character.
|
||||||
|
.Sp
|
||||||
|
* :special:
|
||||||
|
.Sp
|
||||||
|
Matches one or more special chars like brackets, dollar sign, slashes etc.
|
||||||
|
.Sp
|
||||||
|
* :nonprint:
|
||||||
|
.Sp
|
||||||
|
Matches one or more non-printable characters.
|
||||||
|
.RE
|
||||||
.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 one or more regular
|
||||||
@@ -309,7 +382,7 @@ and append the flag. The following flags are supported:
|
|||||||
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 "/account/i"
|
||||||
.Ve
|
.Ve
|
||||||
.PP
|
.PP
|
||||||
If you use the \f(CW\*(C`!\*(C'\fR flag, then the regex match will be negated, that
|
If you use the \f(CW\*(C`!\*(C'\fR flag, then the regex match will be negated, that
|
||||||
@@ -388,7 +461,7 @@ Lets take this table:
|
|||||||
We want to see only the \s-1CMD\s0 column and use a regex for this:
|
We want to see only the \s-1CMD\s0 column and use a regex for this:
|
||||||
.PP
|
.PP
|
||||||
.Vb 6
|
.Vb 6
|
||||||
\& ps | tablizer \-s \*(Aq\es+\*(Aq \-c C
|
\& > ps | tablizer \-s \*(Aq\es+\*(Aq \-c C
|
||||||
\& CMD(4)
|
\& CMD(4)
|
||||||
\& bash
|
\& bash
|
||||||
\& ps
|
\& ps
|
||||||
@@ -425,7 +498,7 @@ use a regexp containing the \f(CW\*(C`/\*(C'\fR character, eg:
|
|||||||
Example:
|
Example:
|
||||||
.PP
|
.PP
|
||||||
.Vb 7
|
.Vb 7
|
||||||
\& cat t/testtable2
|
\& > cat t/testtable2
|
||||||
\& NAME DURATION
|
\& NAME DURATION
|
||||||
\& x 10
|
\& x 10
|
||||||
\& a 100
|
\& a 100
|
||||||
@@ -433,7 +506,7 @@ Example:
|
|||||||
\& u 4
|
\& u 4
|
||||||
\& k 6
|
\& k 6
|
||||||
\&
|
\&
|
||||||
\& cat t/testtable2 | tablizer \-T2 \-R \*(Aq/^\ed/4/\*(Aq \-n
|
\& > cat t/testtable2 | tablizer \-T2 \-R \*(Aq/^\ed/4/\*(Aq \-n
|
||||||
\& NAME DURATION
|
\& NAME DURATION
|
||||||
\& x 40
|
\& x 40
|
||||||
\& a 400
|
\& a 400
|
||||||
@@ -451,7 +524,7 @@ printed vertically, header left, value right, aligned by the field
|
|||||||
widths. Here's an example:
|
widths. Here's an example:
|
||||||
.PP
|
.PP
|
||||||
.Vb 6
|
.Vb 6
|
||||||
\& kubectl get pods | ./tablizer \-o extended
|
\& > kubectl get pods | ./tablizer \-o extended
|
||||||
\& NAME: repldepl\-7bcd8d5b64\-7zq4l
|
\& NAME: repldepl\-7bcd8d5b64\-7zq4l
|
||||||
\& READY: 1/1
|
\& READY: 1/1
|
||||||
\& STATUS: Running
|
\& STATUS: Running
|
||||||
@@ -467,7 +540,7 @@ by the shell, it prints variable assignments for each cell, one line
|
|||||||
per row:
|
per row:
|
||||||
.PP
|
.PP
|
||||||
.Vb 4
|
.Vb 4
|
||||||
\& kubectl get pods | ./tablizer \-o extended ./tablizer \-o shell
|
\& > kubectl get pods | ./tablizer \-o extended ./tablizer \-o shell
|
||||||
\& NAME="repldepl\-7bcd8d5b64\-7zq4l" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
\& NAME="repldepl\-7bcd8d5b64\-7zq4l" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
||||||
\& NAME="repldepl\-7bcd8d5b64\-m48n8" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
\& NAME="repldepl\-7bcd8d5b64\-m48n8" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
||||||
\& NAME="repldepl\-7bcd8d5b64\-q2bf4" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
\& NAME="repldepl\-7bcd8d5b64\-q2bf4" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
||||||
@@ -478,8 +551,22 @@ You can use this in an eval loop.
|
|||||||
Beside normal ascii mode (the default) and extended mode there are
|
Beside normal ascii mode (the default) and extended mode there are
|
||||||
more output modes available: \fBorgtbl\fR which prints an Emacs org-mode
|
more output modes available: \fBorgtbl\fR which prints an Emacs org-mode
|
||||||
table and \fBmarkdown\fR which prints a Markdown table, \fByaml\fR, which
|
table and \fBmarkdown\fR which prints a Markdown table, \fByaml\fR, which
|
||||||
prints yaml encoding and \s-1CSV\s0 mode, which prints a comma separated
|
prints yaml encoding and \fB\s-1CSV\s0\fR mode, which prints a comma separated
|
||||||
value file.
|
value file.
|
||||||
|
.PP
|
||||||
|
A special output mode ist the \fBTemplate\fR mode, activated with the
|
||||||
|
option \f(CW\*(C`\-\-template\*(C'\fR. Template language is the Golang template
|
||||||
|
language: <https://pkg.go.dev/text/template>. You can also use lot's
|
||||||
|
of additional functions from:
|
||||||
|
<https://masterminds.github.io/sprig/>. Here's an example:
|
||||||
|
.PP
|
||||||
|
.Vb 3
|
||||||
|
\& > kubectl get pods | tablizer \-\-template "{{.name}} is {{.status}}"
|
||||||
|
\& alertmanager\-kube\-prometheus\-alertmanager\-0 is Running
|
||||||
|
\& grafana\-fcc54cbc9\-bk7s8 is Running
|
||||||
|
.Ve
|
||||||
|
.PP
|
||||||
|
You can use header names as variables.
|
||||||
.SS "\s-1PUT FIELDS TO CLIPBOARD\s0"
|
.SS "\s-1PUT FIELDS TO CLIPBOARD\s0"
|
||||||
.IX Subsection "PUT FIELDS TO CLIPBOARD"
|
.IX Subsection "PUT FIELDS TO CLIPBOARD"
|
||||||
You can let tablizer put fields to the clipboard using the option
|
You can let tablizer put fields to the clipboard using the option
|
||||||
@@ -605,7 +692,7 @@ parameter \f(CW\*(C`\-N\*(C'\fR or the environment variable \fB\s-1NO_COLOR\s0\f
|
|||||||
.IX Header "BUGS"
|
.IX Header "BUGS"
|
||||||
In order to report a bug, unexpected behavior, feature requests
|
In order to report a bug, unexpected behavior, feature requests
|
||||||
or to submit a patch, please open an issue on github:
|
or to submit a patch, please open an issue on github:
|
||||||
<https://github.com/TLINDEN/tablizer/issues>.
|
<https://codeberg.org/scip/tablizer/issues>.
|
||||||
.SH "LICENSE"
|
.SH "LICENSE"
|
||||||
.IX Header "LICENSE"
|
.IX Header "LICENSE"
|
||||||
This software is licensed under the \s-1GNU GENERAL PUBLIC LICENSE\s0 version 3.
|
This software is licensed under the \s-1GNU GENERAL PUBLIC LICENSE\s0 version 3.
|
||||||
|
|||||||
180
tablizer.pod
180
tablizer.pod
@@ -8,45 +8,51 @@ tablizer - Manipulate tabular output of other programs
|
|||||||
tablizer [regex,...] [-r file] [flags]
|
tablizer [regex,...] [-r 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 ,)
|
||||||
-v, --invert-match select non-matching rows
|
-v, --invert-match select non-matching rows
|
||||||
-n, --numbering Enable header numbering
|
-n, --numbering Enable header numbering
|
||||||
-N, --no-color Disable pattern highlighting
|
-N, --no-color Disable pattern highlighting
|
||||||
-H, --no-headers Disable headers display
|
-H, --no-headers Disable headers display
|
||||||
-s, --separator string Custom field separator
|
-s, --separator <string> Custom field separator (maybe char, string or :class:)
|
||||||
-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
|
||||||
-I, --interactive Interactively filter and select rows
|
-j, --json Read JSON input (must be array of hashes)
|
||||||
|
-I, --interactive Interactively filter and select rows
|
||||||
|
-g, --auto-headers Generate headers if there are none present in input
|
||||||
|
-x, --custom-headers a,b,... Use custom headers, separated by comma
|
||||||
|
|
||||||
Output Flags (mutually exclusive):
|
Output Flags (mutually exclusive):
|
||||||
-X, --extended Enable extended output
|
-X, --extended Enable extended output
|
||||||
-M, --markdown Enable markdown table output
|
-M, --markdown Enable markdown table output
|
||||||
-O, --orgtbl Enable org-mode table output
|
-O, --orgtbl Enable org-mode table output
|
||||||
-S, --shell Enable shell evaluable output
|
-S, --shell Enable shell evaluable output
|
||||||
-Y, --yaml Enable yaml output
|
-Y, --yaml Enable yaml output
|
||||||
-C, --csv Enable CSV output
|
-J, --jsonout Enable JSON output
|
||||||
-A, --ascii Default output mode, ascii tabular
|
-C, --csv Enable CSV output
|
||||||
-L, --hightlight-lines Use alternating background colors for tables
|
-A, --ascii Default output mode, ascii tabular
|
||||||
-y, --yank-columns Yank specified columns (separated by ,) to clipboard,
|
-P, --template <tpl> Enable template mode with template <tpl>
|
||||||
space separated
|
-L, --hightlight-lines Use alternating background colors for tables
|
||||||
|
-o, --ofs <char> Output field separator, used by -A and -C.
|
||||||
|
-y, --yank-columns Yank specified columns (separated by ,) to clipboard,
|
||||||
|
space separated
|
||||||
|
|
||||||
Sort Mode Flags (mutually exclusive):
|
Sort Mode Flags (mutually exclusive):
|
||||||
-a, --sort-age sort according to age (duration) string
|
-a, --sort-age sort according to age (duration) string
|
||||||
-D, --sort-desc Sort in descending order (default: ascending)
|
-D, --sort-desc Sort in descending order (default: ascending)
|
||||||
-i, --sort-numeric sort according to string numerical value
|
-i, --sort-numeric sort according to string numerical value
|
||||||
-t, --sort-time sort according to time string
|
-t, --sort-time sort according to time string
|
||||||
|
|
||||||
Other Flags:
|
Other Flags:
|
||||||
-r --read-file <file> Use <file> as input instead of STDIN
|
-r --read-file <file> Use <file> as input instead of STDIN
|
||||||
--completion <shell> Generate the autocompletion script for <shell>
|
--completion <shell> Generate the autocompletion script for <shell>
|
||||||
-f, --config <file> Configuration file (default: ~/.config/tablizer/config)
|
-f, --config <file> Configuration file (default: ~/.config/tablizer/config)
|
||||||
-d, --debug Enable debugging
|
-d, --debug Enable debugging
|
||||||
-h, --help help for tablizer
|
-h, --help help for tablizer
|
||||||
-m, --man Display manual page
|
-m, --man Display manual page
|
||||||
-V, --version Print program version
|
-V, --version Print program version
|
||||||
|
|
||||||
|
|
||||||
=head1 DESCRIPTION
|
=head1 DESCRIPTION
|
||||||
@@ -75,31 +81,38 @@ may also use the B<-v> option to exclude all rows which match the
|
|||||||
pattern. Hence:
|
pattern. Hence:
|
||||||
|
|
||||||
# read from STDIN
|
# read from STDIN
|
||||||
kubectl get pods | tablizer
|
> kubectl get pods | tablizer
|
||||||
|
|
||||||
# read a file
|
# read a file
|
||||||
tablizer filename
|
> tablizer -r filename
|
||||||
|
|
||||||
# search for pattern in a file (works like grep)
|
# search for pattern in a file (works like grep)
|
||||||
tablizer regex filename
|
> tablizer regex -r filename
|
||||||
|
|
||||||
# search for pattern in STDIN
|
# search for pattern in STDIN
|
||||||
kubectl get pods | tablizer regex
|
> kubectl get pods | tablizer regex
|
||||||
|
|
||||||
The output looks like the original one but every header field will
|
The output looks like the original one. You can add the option B<-n>,
|
||||||
have a numer associated with it, e.g.:
|
then every header field will have a numer associated with it, e.g.:
|
||||||
|
|
||||||
NAME(1) READY(2) STATUS(3) RESTARTS(4) AGE(5)
|
NAME(1) READY(2) STATUS(3) RESTARTS(4) AGE(5)
|
||||||
|
|
||||||
These numbers denote the column and you can use them to specify which
|
These numbers denote the column and you can use them to specify which
|
||||||
columns you want to have in your output (see L<COLUMNS>:
|
columns you want to have in your output (see L<COLUMNS>:
|
||||||
|
|
||||||
kubectl get pods | tablizer -c1,3
|
> kubectl get pods | tablizer -c1,3
|
||||||
|
|
||||||
You can specify the numbers in any order but output will always follow
|
You can specify the numbers in any order but output will always follow
|
||||||
the original order.
|
the original order.
|
||||||
|
|
||||||
The numbering can be suppressed by using the B<-n> option.
|
However, you may also just use the header names instead of numbers,
|
||||||
|
eg:
|
||||||
|
|
||||||
|
> kubectl get pods | tablizer -cname,status
|
||||||
|
|
||||||
|
You can also use regular expressions with B<-c>, eg:
|
||||||
|
|
||||||
|
> kubectl get pods | tablizer -c '[ae]'
|
||||||
|
|
||||||
By default tablizer shows a header containing the names of each
|
By default tablizer shows a header containing the names of each
|
||||||
column. This can be disabled using the B<-H> option. Be aware that
|
column. This can be disabled using the B<-H> option. Be aware that
|
||||||
@@ -144,6 +157,62 @@ Sorts timestamps.
|
|||||||
Finally the B<-d> option enables debugging output which is mostly
|
Finally the B<-d> option enables debugging output which is mostly
|
||||||
useful for the developer.
|
useful for the developer.
|
||||||
|
|
||||||
|
=head2 SEPARATOR
|
||||||
|
|
||||||
|
The option B<-s> can be a single character, in which case the CSV
|
||||||
|
parser will be invoked. You can also specify a string as
|
||||||
|
separator. The string will be interpreted as literal string unless it
|
||||||
|
is a valid go regular expression. For example:
|
||||||
|
|
||||||
|
-s '\t{2,}\'
|
||||||
|
|
||||||
|
is being used as a regexp and will match two or more consecutive tabs.
|
||||||
|
|
||||||
|
-s 'foo'
|
||||||
|
|
||||||
|
on the other hand is no regular expression and will be used literally.
|
||||||
|
|
||||||
|
To make live easier, there are a couple of predefined regular
|
||||||
|
expressions, which you can specify as classes:
|
||||||
|
|
||||||
|
=over
|
||||||
|
|
||||||
|
* :tab:
|
||||||
|
|
||||||
|
Matches a tab and eats spaces around it.
|
||||||
|
|
||||||
|
* :spaces:
|
||||||
|
|
||||||
|
Matches 2 or more spaces.
|
||||||
|
|
||||||
|
* :pipe:
|
||||||
|
|
||||||
|
Matches a pipe character and eats spaces around it.
|
||||||
|
|
||||||
|
* :default:
|
||||||
|
|
||||||
|
Matches 2 or more spaces or tab. This is the default separator if none
|
||||||
|
is specified.
|
||||||
|
|
||||||
|
* :nonword:
|
||||||
|
|
||||||
|
Matches a non-word character.
|
||||||
|
|
||||||
|
* :nondigit:
|
||||||
|
|
||||||
|
Matches a non-digit character.
|
||||||
|
|
||||||
|
* :special:
|
||||||
|
|
||||||
|
Matches one or more special chars like brackets, dollar sign, slashes etc.
|
||||||
|
|
||||||
|
* :nonprint:
|
||||||
|
|
||||||
|
Matches one or more non-printable characters.
|
||||||
|
|
||||||
|
|
||||||
|
=back
|
||||||
|
|
||||||
=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 one or more regular
|
||||||
@@ -168,7 +237,7 @@ and append the flag. The following flags are supported:
|
|||||||
|
|
||||||
Example for a case insensitive search:
|
Example for a case insensitive search:
|
||||||
|
|
||||||
kubectl get pods -A | tablizer "/account/i"
|
> kubectl get pods -A | tablizer "/account/i"
|
||||||
|
|
||||||
If you use the C<!> flag, then the regex match will be negated, that
|
If you use the C<!> flag, then the regex match will be negated, that
|
||||||
is, if a line in the input matches the given regex, but C<!> is
|
is, if a line in the input matches the given regex, but C<!> is
|
||||||
@@ -239,7 +308,7 @@ Lets take this table:
|
|||||||
|
|
||||||
We want to see only the CMD column and use a regex for this:
|
We want to see only the CMD column and use a regex for this:
|
||||||
|
|
||||||
ps | tablizer -s '\s+' -c C
|
> ps | tablizer -s '\s+' -c C
|
||||||
CMD(4)
|
CMD(4)
|
||||||
bash
|
bash
|
||||||
ps
|
ps
|
||||||
@@ -272,7 +341,7 @@ use a regexp containing the C</> character, eg:
|
|||||||
|
|
||||||
Example:
|
Example:
|
||||||
|
|
||||||
cat t/testtable2
|
> cat t/testtable2
|
||||||
NAME DURATION
|
NAME DURATION
|
||||||
x 10
|
x 10
|
||||||
a 100
|
a 100
|
||||||
@@ -280,7 +349,7 @@ Example:
|
|||||||
u 4
|
u 4
|
||||||
k 6
|
k 6
|
||||||
|
|
||||||
cat t/testtable2 | tablizer -T2 -R '/^\d/4/' -n
|
> cat t/testtable2 | tablizer -T2 -R '/^\d/4/' -n
|
||||||
NAME DURATION
|
NAME DURATION
|
||||||
x 40
|
x 40
|
||||||
a 400
|
a 400
|
||||||
@@ -297,7 +366,7 @@ useful which enables I<extended mode>. In this mode, each row will be
|
|||||||
printed vertically, header left, value right, aligned by the field
|
printed vertically, header left, value right, aligned by the field
|
||||||
widths. Here's an example:
|
widths. Here's an example:
|
||||||
|
|
||||||
kubectl get pods | ./tablizer -o extended
|
> kubectl get pods | ./tablizer -o extended
|
||||||
NAME: repldepl-7bcd8d5b64-7zq4l
|
NAME: repldepl-7bcd8d5b64-7zq4l
|
||||||
READY: 1/1
|
READY: 1/1
|
||||||
STATUS: Running
|
STATUS: Running
|
||||||
@@ -311,7 +380,7 @@ The option B<-o shell> can be used if the output has to be processed
|
|||||||
by the shell, it prints variable assignments for each cell, one line
|
by the shell, it prints variable assignments for each cell, one line
|
||||||
per row:
|
per row:
|
||||||
|
|
||||||
kubectl get pods | ./tablizer -o extended ./tablizer -o shell
|
> kubectl get pods | ./tablizer -o extended ./tablizer -o shell
|
||||||
NAME="repldepl-7bcd8d5b64-7zq4l" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
NAME="repldepl-7bcd8d5b64-7zq4l" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
||||||
NAME="repldepl-7bcd8d5b64-m48n8" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
NAME="repldepl-7bcd8d5b64-m48n8" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
||||||
NAME="repldepl-7bcd8d5b64-q2bf4" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
NAME="repldepl-7bcd8d5b64-q2bf4" READY="1/1" STATUS="Running" RESTARTS="9 (47m ago)" AGE="4d23h"
|
||||||
@@ -321,9 +390,22 @@ You can use this in an eval loop.
|
|||||||
Beside normal ascii mode (the default) and extended mode there are
|
Beside normal ascii mode (the default) and extended mode there are
|
||||||
more output modes available: B<orgtbl> which prints an Emacs org-mode
|
more output modes available: B<orgtbl> which prints an Emacs org-mode
|
||||||
table and B<markdown> which prints a Markdown table, B<yaml>, which
|
table and B<markdown> which prints a Markdown table, B<yaml>, which
|
||||||
prints yaml encoding and CSV mode, which prints a comma separated
|
prints yaml encoding and B<CSV> mode, which prints a comma separated
|
||||||
value file.
|
value file.
|
||||||
|
|
||||||
|
A special output mode ist the B<Template> mode, activated with the
|
||||||
|
option C<--template>. Template language is the Golang template
|
||||||
|
language: L<https://pkg.go.dev/text/template>. You can also use lot's
|
||||||
|
of additional functions from:
|
||||||
|
L<https://masterminds.github.io/sprig/>. Here's an example:
|
||||||
|
|
||||||
|
> kubectl get pods | tablizer --template "{{.name}} is {{.status}}"
|
||||||
|
alertmanager-kube-prometheus-alertmanager-0 is Running
|
||||||
|
grafana-fcc54cbc9-bk7s8 is Running
|
||||||
|
|
||||||
|
|
||||||
|
You can use header names as variables.
|
||||||
|
|
||||||
=head2 PUT FIELDS TO CLIPBOARD
|
=head2 PUT FIELDS TO CLIPBOARD
|
||||||
|
|
||||||
You can let tablizer put fields to the clipboard using the option
|
You can let tablizer put fields to the clipboard using the option
|
||||||
@@ -445,7 +527,7 @@ parameter C<-N> or the environment variable B<NO_COLOR> to a true value.
|
|||||||
|
|
||||||
In order to report a bug, unexpected behavior, feature requests
|
In order to report a bug, unexpected behavior, feature requests
|
||||||
or to submit a patch, please open an issue on github:
|
or to submit a patch, please open an issue on github:
|
||||||
L<https://github.com/TLINDEN/tablizer/issues>.
|
L<https://codeberg.org/scip/tablizer/issues>.
|
||||||
|
|
||||||
=head1 LICENSE
|
=head1 LICENSE
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user